blob: 0896db4284c9b28c979ff3654a66f6f40f1e8962 [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
2/*
3 * Copyright 2011 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
reed@google.com5c3d1472011-02-22 19:12:23 +00008#ifndef SkClipStack_DEFINED
9#define SkClipStack_DEFINED
10
11#include "SkDeque.h"
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +000012#include "SkPath.h"
13#include "SkRect.h"
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +000014#include "SkRRect.h"
reed@google.com5c3d1472011-02-22 19:12:23 +000015#include "SkRegion.h"
robertphillips@google.com46f93502012-08-07 15:38:08 +000016#include "SkTDArray.h"
reed@google.com5c3d1472011-02-22 19:12:23 +000017
reed@google.com5c3d1472011-02-22 19:12:23 +000018
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +000019// Because a single save/restore state can have multiple clips, this class
20// stores the stack depth (fSaveCount) and clips (fDeque) separately.
21// Each clip in fDeque stores the stack state to which it belongs
22// (i.e., the fSaveCount in force when it was added). Restores are thus
23// implemented by removing clips from fDeque that have an fSaveCount larger
24// then the freshly decremented count.
ctguil@chromium.org7ffb1b22011-03-15 21:27:08 +000025class SK_API SkClipStack {
reed@google.com5c3d1472011-02-22 19:12:23 +000026public:
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +000027 enum BoundsType {
28 // The bounding box contains all the pixels that can be written to
29 kNormal_BoundsType,
30 // The bounding box contains all the pixels that cannot be written to.
31 // The real bound extends out to infinity and all the pixels outside
32 // of the bound can be written to. Note that some of the pixels inside
33 // the bound may also be writeable but all pixels that cannot be
34 // written to are guaranteed to be inside.
35 kInsideOut_BoundsType
36 };
37
38 class Element {
39 public:
40 enum Type {
41 //!< This element makes the clip empty (regardless of previous elements).
42 kEmpty_Type,
43 //!< This element combines a rect with the current clip using a set operation
44 kRect_Type,
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +000045 //!< This element combines a round-rect with the current clip using a set operation
46 kRRect_Type,
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +000047 //!< This element combines a path with the current clip using a set operation
48 kPath_Type,
49 };
50
51 Element() {
52 this->initCommon(0, SkRegion::kReplace_Op, false);
53 this->setEmpty();
54 }
55
56 Element(const SkRect& rect, SkRegion::Op op, bool doAA) {
57 this->initRect(0, rect, op, doAA);
58 }
59
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +000060 Element(const SkRRect& rrect, SkRegion::Op op, bool doAA) {
61 this->initRRect(0, rrect, op, doAA);
62 }
63
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +000064 Element(const SkPath& path, SkRegion::Op op, bool doAA) {
65 this->initPath(0, path, op, doAA);
66 }
67
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +000068 bool operator== (const Element& element) const;
bsalomon@google.com8182fa02012-12-04 14:06:06 +000069 bool operator!= (const Element& element) const { return !(*this == element); }
70
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +000071 //!< Call to get the type of the clip element.
72 Type getType() const { return fType; }
73
74 //!< Call if getType() is kPath to get the path.
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +000075 const SkPath& getPath() const { SkASSERT(kPath_Type == fType); return fPath; }
76
77 //!< Call if getType() is kRRect to get the round-rect.
78 const SkRRect& getRRect() const { SkASSERT(kRRect_Type == fType); return fRRect; }
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +000079
80 //!< Call if getType() is kRect to get the rect.
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +000081 const SkRect& getRect() const { SkASSERT(kRect_Type == fType); return fRect; }
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +000082
83 //!< Call if getType() is not kEmpty to get the set operation used to combine this element.
84 SkRegion::Op getOp() const { return fOp; }
85
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +000086 //!< Call to get the element as a path, regardless of its type.
87 void asPath(SkPath* path) const;
88
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +000089 /** If getType() is not kEmpty this indicates whether the clip shape should be anti-aliased
90 when it is rasterized. */
91 bool isAA() const { return fDoAA; }
92
bsalomon@google.comc6b3e482012-12-07 20:43:52 +000093 //!< Inverts the fill of the clip shape. Note that a kEmpty element remains kEmpty.
94 void invertShapeFillType();
95
96 //!< Sets the set operation represented by the element.
bsalomon@google.com8182fa02012-12-04 14:06:06 +000097 void setOp(SkRegion::Op op) { fOp = op; }
98
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +000099 /** The GenID can be used by clip stack clients to cache representations of the clip. The
100 ID corresponds to the set of clip elements up to and including this element within the
101 stack not to the element itself. That is the same clip path in different stacks will
102 have a different ID since the elements produce different clip result in the context of
103 their stacks. */
commit-bot@chromium.orgd3e58422013-11-05 15:03:08 +0000104 int32_t getGenID() const { SkASSERT(kInvalidGenID != fGenID); return fGenID; }
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000105
bsalomon@google.com8182fa02012-12-04 14:06:06 +0000106 /**
107 * Gets the bounds of the clip element, either the rect or path bounds. (Whether the shape
108 * is inverse filled is not considered.)
109 */
110 const SkRect& getBounds() const {
111 static const SkRect kEmpty = { 0, 0, 0, 0 };
112 switch (fType) {
113 case kRect_Type:
114 return fRect;
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000115 case kRRect_Type:
116 return fRRect.getBounds();
bsalomon@google.com8182fa02012-12-04 14:06:06 +0000117 case kPath_Type:
118 return fPath.getBounds();
119 case kEmpty_Type:
120 return kEmpty;
121 default:
122 SkDEBUGFAIL("Unexpected type.");
123 return kEmpty;
124 }
125 }
126
127 /**
128 * Conservatively checks whether the clip shape contains the rect param. (Whether the shape
129 * is inverse filled is not considered.)
130 */
131 bool contains(const SkRect& rect) const {
132 switch (fType) {
133 case kRect_Type:
134 return fRect.contains(rect);
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000135 case kRRect_Type:
136 return fRRect.contains(rect);
bsalomon@google.com8182fa02012-12-04 14:06:06 +0000137 case kPath_Type:
138 return fPath.conservativelyContainsRect(rect);
139 case kEmpty_Type:
140 return false;
141 default:
142 SkDEBUGFAIL("Unexpected type.");
143 return false;
144 }
145 }
146
147 /**
148 * Is the clip shape inverse filled.
149 */
150 bool isInverseFilled() const {
151 return kPath_Type == fType && fPath.isInverseFillType();
152 }
153
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000154 private:
155 friend class SkClipStack;
156
157 SkPath fPath;
158 SkRect fRect;
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000159 SkRRect fRRect;
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000160 int fSaveCount; // save count of stack when this element was added.
161 SkRegion::Op fOp;
162 Type fType;
163 bool fDoAA;
164
165 /* fFiniteBoundType and fFiniteBound are used to incrementally update the clip stack's
166 bound. When fFiniteBoundType is kNormal_BoundsType, fFiniteBound represents the
167 conservative bounding box of the pixels that aren't clipped (i.e., any pixels that can be
168 drawn to are inside the bound). When fFiniteBoundType is kInsideOut_BoundsType (which
169 occurs when a clip is inverse filled), fFiniteBound represents the conservative bounding
170 box of the pixels that _are_ clipped (i.e., any pixels that cannot be drawn to are inside
171 the bound). When fFiniteBoundType is kInsideOut_BoundsType the actual bound is the
172 infinite plane. This behavior of fFiniteBoundType and fFiniteBound is required so that we
173 can capture the cancelling out of the extensions to infinity when two inverse filled
174 clips are Booleaned together. */
175 SkClipStack::BoundsType fFiniteBoundType;
176 SkRect fFiniteBound;
skia.committer@gmail.comc3d7d902012-11-30 02:01:24 +0000177
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000178 // When element is applied to the previous elements in the stack is the result known to be
179 // equivalent to a single rect intersection? IIOW, is the clip effectively a rectangle.
180 bool fIsIntersectionOfRects;
181
182 int fGenID;
183
184 Element(int saveCount) {
185 this->initCommon(saveCount, SkRegion::kReplace_Op, false);
186 this->setEmpty();
187 }
188
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000189 Element(int saveCount, const SkRRect& rrect, SkRegion::Op op, bool doAA) {
190 this->initRRect(saveCount, rrect, op, doAA);
191 }
192
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000193 Element(int saveCount, const SkRect& rect, SkRegion::Op op, bool doAA) {
194 this->initRect(saveCount, rect, op, doAA);
195 }
196
197 Element(int saveCount, const SkPath& path, SkRegion::Op op, bool doAA) {
198 this->initPath(saveCount, path, op, doAA);
199 }
200
201 void initCommon(int saveCount, SkRegion::Op op, bool doAA) {
202 fSaveCount = saveCount;
203 fOp = op;
204 fDoAA = doAA;
205 // A default of inside-out and empty bounds means the bounds are effectively void as it
206 // indicates that nothing is known to be outside the clip.
207 fFiniteBoundType = kInsideOut_BoundsType;
208 fFiniteBound.setEmpty();
209 fIsIntersectionOfRects = false;
210 fGenID = kInvalidGenID;
211 }
212
213 void initRect(int saveCount, const SkRect& rect, SkRegion::Op op, bool doAA) {
214 fRect = rect;
215 fType = kRect_Type;
216 this->initCommon(saveCount, op, doAA);
217 }
218
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000219 void initRRect(int saveCount, const SkRRect& rrect, SkRegion::Op op, bool doAA) {
220 if (rrect.isRect()) {
221 fRect = rrect.getBounds();
222 fType = kRect_Type;
223 } else {
224 fRRect = rrect;
225 fType = kRRect_Type;
226 }
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000227 this->initCommon(saveCount, op, doAA);
228 }
229
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000230 void initPath(int saveCount, const SkPath& path, SkRegion::Op op, bool doAA);
231
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000232 void setEmpty() {
233 fType = kEmpty_Type;
234 fFiniteBound.setEmpty();
235 fFiniteBoundType = kNormal_BoundsType;
236 fIsIntersectionOfRects = false;
237 fRect.setEmpty();
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000238 fRRect.setEmpty();
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000239 fPath.reset();
240 fGenID = kEmptyGenID;
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000241 SkDEBUGCODE(this->checkEmpty();)
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000242 }
243
244 // All Element methods below are only used within SkClipStack.cpp
245 inline void checkEmpty() const;
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000246 inline bool canBeIntersectedInPlace(int saveCount, SkRegion::Op op) const;
247 /* This method checks to see if two rect clips can be safely merged into one. The issue here
skia.committer@gmail.comc3d7d902012-11-30 02:01:24 +0000248 is that to be strictly correct all the edges of the resulting rect must have the same
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000249 anti-aliasing. */
250 bool rectRectIntersectAllowed(const SkRect& newR, bool newAA) const;
251 /** Determines possible finite bounds for the Element given the previous element of the
252 stack */
253 void updateBoundAndGenID(const Element* prior);
254 // The different combination of fill & inverse fill when combining bounding boxes
255 enum FillCombo {
256 kPrev_Cur_FillCombo,
257 kPrev_InvCur_FillCombo,
258 kInvPrev_Cur_FillCombo,
259 kInvPrev_InvCur_FillCombo
260 };
261 // per-set operation functions used by updateBoundAndGenID().
262 inline void combineBoundsDiff(FillCombo combination, const SkRect& prevFinite);
263 inline void combineBoundsXOR(int combination, const SkRect& prevFinite);
264 inline void combineBoundsUnion(int combination, const SkRect& prevFinite);
265 inline void combineBoundsIntersection(int combination, const SkRect& prevFinite);
266 inline void combineBoundsRevDiff(int combination, const SkRect& prevFinite);
267 };
268
reed@google.com5c3d1472011-02-22 19:12:23 +0000269 SkClipStack();
vandebo@chromium.org1e1c36f2011-05-03 16:26:09 +0000270 SkClipStack(const SkClipStack& b);
robertphillips@google.comcc6493b2012-07-26 18:39:13 +0000271 explicit SkClipStack(const SkRect& r);
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000272 explicit SkClipStack(const SkIRect& r);
vandebo@chromium.org610f7162012-03-14 18:34:15 +0000273 ~SkClipStack();
reed@google.com5c3d1472011-02-22 19:12:23 +0000274
vandebo@chromium.org1e1c36f2011-05-03 16:26:09 +0000275 SkClipStack& operator=(const SkClipStack& b);
276 bool operator==(const SkClipStack& b) const;
277 bool operator!=(const SkClipStack& b) const { return !(*this == b); }
278
reed@google.com5c3d1472011-02-22 19:12:23 +0000279 void reset();
280
281 int getSaveCount() const { return fSaveCount; }
282 void save();
283 void restore();
284
robertphillips@google.com607fe072012-07-24 13:54:00 +0000285 /**
286 * getBounds places the current finite bound in its first parameter. In its
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000287 * second, it indicates which kind of bound is being returned. If
robertphillips@google.com7b112892012-07-31 15:18:21 +0000288 * 'canvFiniteBound' is a normal bounding box then it encloses all writeable
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000289 * pixels. If 'canvFiniteBound' is an inside out bounding box then it
robertphillips@google.com607fe072012-07-24 13:54:00 +0000290 * encloses all the un-writeable pixels and the true/normal bound is the
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +0000291 * infinite plane. isIntersectionOfRects is an optional parameter
robertphillips@google.com7b112892012-07-31 15:18:21 +0000292 * that is true if 'canvFiniteBound' resulted from an intersection of rects.
robertphillips@google.com607fe072012-07-24 13:54:00 +0000293 */
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000294 void getBounds(SkRect* canvFiniteBound,
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +0000295 BoundsType* boundType,
296 bool* isIntersectionOfRects = NULL) const;
robertphillips@google.com607fe072012-07-24 13:54:00 +0000297
bsalomon@google.com3ab43d52012-10-11 19:39:09 +0000298 /**
299 * Takes an input rect in device space and conservatively clips it to the
300 * clip-stack. If false is returned then the rect does not intersect the
301 * clip and is unmodified.
302 */
303 bool intersectRectWithClip(SkRect* devRect) const;
304
junov@chromium.org8cdf0f52012-12-12 17:58:15 +0000305 /**
306 * Returns true if the input rect in device space is entirely contained
307 * by the clip. A return value of false does not guarantee that the rect
308 * is not contained by the clip.
309 */
310 bool quickContains(const SkRect& devRect) const;
311
reed@google.com115d9312012-05-16 18:50:40 +0000312 void clipDevRect(const SkIRect& ir, SkRegion::Op op) {
reed@google.com5c3d1472011-02-22 19:12:23 +0000313 SkRect r;
314 r.set(ir);
reed@google.com00177082011-10-12 14:34:30 +0000315 this->clipDevRect(r, op, false);
reed@google.com5c3d1472011-02-22 19:12:23 +0000316 }
reed@google.com00177082011-10-12 14:34:30 +0000317 void clipDevRect(const SkRect&, SkRegion::Op, bool doAA);
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000318 void clipDevRRect(const SkRRect&, SkRegion::Op, bool doAA);
reed@google.com00177082011-10-12 14:34:30 +0000319 void clipDevPath(const SkPath&, SkRegion::Op, bool doAA);
reed@google.com0557d9e2012-08-16 15:59:59 +0000320 // An optimized version of clipDevRect(emptyRect, kIntersect, ...)
321 void clipEmpty();
reed@google.com5c3d1472011-02-22 19:12:23 +0000322
robertphillips@google.comcc6493b2012-07-26 18:39:13 +0000323 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000324 * isWideOpen returns true if the clip state corresponds to the infinite
robertphillips@google.comcc6493b2012-07-26 18:39:13 +0000325 * plane (i.e., draws are not limited at all)
326 */
327 bool isWideOpen() const;
328
robertphillips@google.com46f93502012-08-07 15:38:08 +0000329 /**
robertphillips@google.com46f93502012-08-07 15:38:08 +0000330 * The generation ID has three reserved values to indicate special
bsalomon@google.come8ca6c62012-11-07 21:19:10 +0000331 * (potentially ignorable) cases
robertphillips@google.com46f93502012-08-07 15:38:08 +0000332 */
commit-bot@chromium.orgd3e58422013-11-05 15:03:08 +0000333 static const int32_t kInvalidGenID = 0; //!< Invalid id that is never returned by
334 //!< SkClipStack. Useful when caching clips
335 //!< based on GenID.
robertphillips@google.com46f93502012-08-07 15:38:08 +0000336 static const int32_t kEmptyGenID = 1; // no pixels writeable
337 static const int32_t kWideOpenGenID = 2; // all pixels writeable
338
robertphillips@google.com73e71022012-08-09 18:10:49 +0000339 int32_t getTopmostGenID() const;
340
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000341public:
342 class Iter {
reed@google.com5c3d1472011-02-22 19:12:23 +0000343 public:
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000344 enum IterStart {
robertphillips@google.com80214e22012-07-20 15:33:18 +0000345 kBottom_IterStart = SkDeque::Iter::kFront_IterStart,
346 kTop_IterStart = SkDeque::Iter::kBack_IterStart
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000347 };
348
bsalomon@google.comd302f142011-03-03 13:54:13 +0000349 /**
350 * Creates an uninitialized iterator. Must be reset()
351 */
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000352 Iter();
bsalomon@google.comd302f142011-03-03 13:54:13 +0000353
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000354 Iter(const SkClipStack& stack, IterStart startLoc);
reed@google.com5c3d1472011-02-22 19:12:23 +0000355
bsalomon@google.com8182fa02012-12-04 14:06:06 +0000356 /**
357 * Return the clip element for this iterator. If next()/prev() returns NULL, then the
358 * iterator is done.
359 */
360 const Element* next();
361 const Element* prev();
reed@google.com5c3d1472011-02-22 19:12:23 +0000362
363 /**
bsalomon@google.com8182fa02012-12-04 14:06:06 +0000364 * Moves the iterator to the topmost element with the specified RegionOp and returns that
365 * element. If no clip element with that op is found, the first element is returned.
reed@google.com5c3d1472011-02-22 19:12:23 +0000366 */
bsalomon@google.com8182fa02012-12-04 14:06:06 +0000367 const Element* skipToTopmost(SkRegion::Op op);
robertphillips@google.com5836b6d2012-07-18 12:06:15 +0000368
369 /**
bsalomon@google.comd302f142011-03-03 13:54:13 +0000370 * Restarts the iterator on a clip stack.
371 */
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000372 void reset(const SkClipStack& stack, IterStart startLoc);
bsalomon@google.comd302f142011-03-03 13:54:13 +0000373
reed@google.com5c3d1472011-02-22 19:12:23 +0000374 private:
robertphillips@google.com5836b6d2012-07-18 12:06:15 +0000375 const SkClipStack* fStack;
robertphillips@google.com5836b6d2012-07-18 12:06:15 +0000376 SkDeque::Iter fIter;
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000377 };
378
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000379 /**
robertphillips@google.com80214e22012-07-20 15:33:18 +0000380 * The B2TIter iterates from the bottom of the stack to the top.
381 * It inherits privately from Iter to prevent access to reverse iteration.
382 */
383 class B2TIter : private Iter {
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000384 public:
robertphillips@google.com80214e22012-07-20 15:33:18 +0000385 B2TIter() {}
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000386
387 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000388 * Wrap Iter's 2 parameter ctor to force initialization to the
robertphillips@google.com80214e22012-07-20 15:33:18 +0000389 * beginning of the deque/bottom of the stack
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000390 */
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000391 B2TIter(const SkClipStack& stack)
robertphillips@google.com80214e22012-07-20 15:33:18 +0000392 : INHERITED(stack, kBottom_IterStart) {
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000393 }
394
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000395 using Iter::next;
396
397 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000398 * Wrap Iter::reset to force initialization to the
robertphillips@google.com80214e22012-07-20 15:33:18 +0000399 * beginning of the deque/bottom of the stack
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000400 */
401 void reset(const SkClipStack& stack) {
robertphillips@google.com80214e22012-07-20 15:33:18 +0000402 this->INHERITED::reset(stack, kBottom_IterStart);
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000403 }
404
405 private:
406
407 typedef Iter INHERITED;
reed@google.com5c3d1472011-02-22 19:12:23 +0000408 };
409
robertphillips@google.com607fe072012-07-24 13:54:00 +0000410 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000411 * GetConservativeBounds returns a conservative bound of the current clip.
412 * Since this could be the infinite plane (if inverse fills were involved) the
413 * maxWidth and maxHeight parameters can be used to limit the returned bound
robertphillips@google.com607fe072012-07-24 13:54:00 +0000414 * to the expected drawing area. Similarly, the offsetX and offsetY parameters
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000415 * allow the caller to offset the returned bound to account for translated
robertphillips@google.com607fe072012-07-24 13:54:00 +0000416 * drawing areas (i.e., those resulting from a saveLayer). For finite bounds,
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000417 * the translation (+offsetX, +offsetY) is applied before the clamp to the
robertphillips@google.com607fe072012-07-24 13:54:00 +0000418 * maximum rectangle: [0,maxWidth) x [0,maxHeight).
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000419 * isIntersectionOfRects is an optional parameter that is true when
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000420 * 'devBounds' is the result of an intersection of rects. In this case
421 * 'devBounds' is the exact answer/clip.
robertphillips@google.com607fe072012-07-24 13:54:00 +0000422 */
423 void getConservativeBounds(int offsetX,
424 int offsetY,
425 int maxWidth,
426 int maxHeight,
robertphillips@google.com7b112892012-07-31 15:18:21 +0000427 SkRect* devBounds,
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +0000428 bool* isIntersectionOfRects = NULL) const;
robertphillips@google.com607fe072012-07-24 13:54:00 +0000429
reed@google.com5c3d1472011-02-22 19:12:23 +0000430private:
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000431 friend class Iter;
reed@google.com5c3d1472011-02-22 19:12:23 +0000432
433 SkDeque fDeque;
434 int fSaveCount;
robertphillips@google.com46f93502012-08-07 15:38:08 +0000435
436 // Generation ID for the clip stack. This is incremented for each
437 // clipDevRect and clipDevPath call. 0 is reserved to indicate an
438 // invalid ID.
439 static int32_t gGenID;
440
robertphillips@google.com46f93502012-08-07 15:38:08 +0000441 /**
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000442 * Helper for clipDevPath, etc.
443 */
444 void pushElement(const Element& element);
445
446 /**
commit-bot@chromium.org6fbe54c2013-06-11 11:01:48 +0000447 * Restore the stack back to the specified save count.
448 */
449 void restoreTo(int saveCount);
450
451 /**
robertphillips@google.com46f93502012-08-07 15:38:08 +0000452 * Return the next unique generation ID.
453 */
454 static int32_t GetNextGenID();
reed@google.com5c3d1472011-02-22 19:12:23 +0000455};
456
457#endif