blob: a23ef4a5a57b9de43e37c61fe53ec912b1ce23d5 [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
commit-bot@chromium.org9cb671a2014-02-16 14:45:45 +0000232 void setEmpty();
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000233
234 // All Element methods below are only used within SkClipStack.cpp
235 inline void checkEmpty() const;
bsalomon@google.com8a98e3b2012-11-29 21:05:13 +0000236 inline bool canBeIntersectedInPlace(int saveCount, SkRegion::Op op) const;
237 /* 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 +0000238 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 +0000239 anti-aliasing. */
240 bool rectRectIntersectAllowed(const SkRect& newR, bool newAA) const;
241 /** Determines possible finite bounds for the Element given the previous element of the
242 stack */
243 void updateBoundAndGenID(const Element* prior);
244 // The different combination of fill & inverse fill when combining bounding boxes
245 enum FillCombo {
246 kPrev_Cur_FillCombo,
247 kPrev_InvCur_FillCombo,
248 kInvPrev_Cur_FillCombo,
249 kInvPrev_InvCur_FillCombo
250 };
251 // per-set operation functions used by updateBoundAndGenID().
252 inline void combineBoundsDiff(FillCombo combination, const SkRect& prevFinite);
253 inline void combineBoundsXOR(int combination, const SkRect& prevFinite);
254 inline void combineBoundsUnion(int combination, const SkRect& prevFinite);
255 inline void combineBoundsIntersection(int combination, const SkRect& prevFinite);
256 inline void combineBoundsRevDiff(int combination, const SkRect& prevFinite);
257 };
258
reed@google.com5c3d1472011-02-22 19:12:23 +0000259 SkClipStack();
vandebo@chromium.org1e1c36f2011-05-03 16:26:09 +0000260 SkClipStack(const SkClipStack& b);
robertphillips@google.comcc6493b2012-07-26 18:39:13 +0000261 explicit SkClipStack(const SkRect& r);
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000262 explicit SkClipStack(const SkIRect& r);
vandebo@chromium.org610f7162012-03-14 18:34:15 +0000263 ~SkClipStack();
reed@google.com5c3d1472011-02-22 19:12:23 +0000264
vandebo@chromium.org1e1c36f2011-05-03 16:26:09 +0000265 SkClipStack& operator=(const SkClipStack& b);
266 bool operator==(const SkClipStack& b) const;
267 bool operator!=(const SkClipStack& b) const { return !(*this == b); }
268
reed@google.com5c3d1472011-02-22 19:12:23 +0000269 void reset();
270
271 int getSaveCount() const { return fSaveCount; }
272 void save();
273 void restore();
274
robertphillips@google.com607fe072012-07-24 13:54:00 +0000275 /**
276 * getBounds places the current finite bound in its first parameter. In its
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000277 * second, it indicates which kind of bound is being returned. If
robertphillips@google.com7b112892012-07-31 15:18:21 +0000278 * 'canvFiniteBound' is a normal bounding box then it encloses all writeable
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000279 * pixels. If 'canvFiniteBound' is an inside out bounding box then it
robertphillips@google.com607fe072012-07-24 13:54:00 +0000280 * encloses all the un-writeable pixels and the true/normal bound is the
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +0000281 * infinite plane. isIntersectionOfRects is an optional parameter
robertphillips@google.com7b112892012-07-31 15:18:21 +0000282 * that is true if 'canvFiniteBound' resulted from an intersection of rects.
robertphillips@google.com607fe072012-07-24 13:54:00 +0000283 */
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000284 void getBounds(SkRect* canvFiniteBound,
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +0000285 BoundsType* boundType,
286 bool* isIntersectionOfRects = NULL) const;
robertphillips@google.com607fe072012-07-24 13:54:00 +0000287
bsalomon@google.com3ab43d52012-10-11 19:39:09 +0000288 /**
289 * Takes an input rect in device space and conservatively clips it to the
290 * clip-stack. If false is returned then the rect does not intersect the
291 * clip and is unmodified.
292 */
293 bool intersectRectWithClip(SkRect* devRect) const;
294
junov@chromium.org8cdf0f52012-12-12 17:58:15 +0000295 /**
296 * Returns true if the input rect in device space is entirely contained
297 * by the clip. A return value of false does not guarantee that the rect
298 * is not contained by the clip.
299 */
300 bool quickContains(const SkRect& devRect) const;
301
reed@google.com115d9312012-05-16 18:50:40 +0000302 void clipDevRect(const SkIRect& ir, SkRegion::Op op) {
reed@google.com5c3d1472011-02-22 19:12:23 +0000303 SkRect r;
304 r.set(ir);
reed@google.com00177082011-10-12 14:34:30 +0000305 this->clipDevRect(r, op, false);
reed@google.com5c3d1472011-02-22 19:12:23 +0000306 }
reed@google.com00177082011-10-12 14:34:30 +0000307 void clipDevRect(const SkRect&, SkRegion::Op, bool doAA);
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000308 void clipDevRRect(const SkRRect&, SkRegion::Op, bool doAA);
reed@google.com00177082011-10-12 14:34:30 +0000309 void clipDevPath(const SkPath&, SkRegion::Op, bool doAA);
reed@google.com0557d9e2012-08-16 15:59:59 +0000310 // An optimized version of clipDevRect(emptyRect, kIntersect, ...)
311 void clipEmpty();
reed@google.com5c3d1472011-02-22 19:12:23 +0000312
robertphillips@google.comcc6493b2012-07-26 18:39:13 +0000313 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000314 * isWideOpen returns true if the clip state corresponds to the infinite
robertphillips@google.comcc6493b2012-07-26 18:39:13 +0000315 * plane (i.e., draws are not limited at all)
316 */
317 bool isWideOpen() const;
318
robertphillips@google.com46f93502012-08-07 15:38:08 +0000319 /**
robertphillips@google.com46f93502012-08-07 15:38:08 +0000320 * The generation ID has three reserved values to indicate special
bsalomon@google.come8ca6c62012-11-07 21:19:10 +0000321 * (potentially ignorable) cases
robertphillips@google.com46f93502012-08-07 15:38:08 +0000322 */
commit-bot@chromium.orgd3e58422013-11-05 15:03:08 +0000323 static const int32_t kInvalidGenID = 0; //!< Invalid id that is never returned by
324 //!< SkClipStack. Useful when caching clips
325 //!< based on GenID.
robertphillips@google.com46f93502012-08-07 15:38:08 +0000326 static const int32_t kEmptyGenID = 1; // no pixels writeable
327 static const int32_t kWideOpenGenID = 2; // all pixels writeable
328
robertphillips@google.com73e71022012-08-09 18:10:49 +0000329 int32_t getTopmostGenID() const;
330
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000331public:
332 class Iter {
reed@google.com5c3d1472011-02-22 19:12:23 +0000333 public:
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000334 enum IterStart {
robertphillips@google.com80214e22012-07-20 15:33:18 +0000335 kBottom_IterStart = SkDeque::Iter::kFront_IterStart,
336 kTop_IterStart = SkDeque::Iter::kBack_IterStart
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000337 };
338
bsalomon@google.comd302f142011-03-03 13:54:13 +0000339 /**
340 * Creates an uninitialized iterator. Must be reset()
341 */
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000342 Iter();
bsalomon@google.comd302f142011-03-03 13:54:13 +0000343
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000344 Iter(const SkClipStack& stack, IterStart startLoc);
reed@google.com5c3d1472011-02-22 19:12:23 +0000345
bsalomon@google.com8182fa02012-12-04 14:06:06 +0000346 /**
347 * Return the clip element for this iterator. If next()/prev() returns NULL, then the
348 * iterator is done.
349 */
350 const Element* next();
351 const Element* prev();
reed@google.com5c3d1472011-02-22 19:12:23 +0000352
353 /**
bsalomon@google.com8182fa02012-12-04 14:06:06 +0000354 * Moves the iterator to the topmost element with the specified RegionOp and returns that
355 * element. If no clip element with that op is found, the first element is returned.
reed@google.com5c3d1472011-02-22 19:12:23 +0000356 */
bsalomon@google.com8182fa02012-12-04 14:06:06 +0000357 const Element* skipToTopmost(SkRegion::Op op);
robertphillips@google.com5836b6d2012-07-18 12:06:15 +0000358
359 /**
bsalomon@google.comd302f142011-03-03 13:54:13 +0000360 * Restarts the iterator on a clip stack.
361 */
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000362 void reset(const SkClipStack& stack, IterStart startLoc);
bsalomon@google.comd302f142011-03-03 13:54:13 +0000363
reed@google.com5c3d1472011-02-22 19:12:23 +0000364 private:
robertphillips@google.com5836b6d2012-07-18 12:06:15 +0000365 const SkClipStack* fStack;
robertphillips@google.com5836b6d2012-07-18 12:06:15 +0000366 SkDeque::Iter fIter;
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000367 };
368
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000369 /**
robertphillips@google.com80214e22012-07-20 15:33:18 +0000370 * The B2TIter iterates from the bottom of the stack to the top.
371 * It inherits privately from Iter to prevent access to reverse iteration.
372 */
373 class B2TIter : private Iter {
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000374 public:
robertphillips@google.com80214e22012-07-20 15:33:18 +0000375 B2TIter() {}
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000376
377 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000378 * Wrap Iter's 2 parameter ctor to force initialization to the
robertphillips@google.com80214e22012-07-20 15:33:18 +0000379 * beginning of the deque/bottom of the stack
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000380 */
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000381 B2TIter(const SkClipStack& stack)
robertphillips@google.com80214e22012-07-20 15:33:18 +0000382 : INHERITED(stack, kBottom_IterStart) {
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000383 }
384
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000385 using Iter::next;
386
387 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000388 * Wrap Iter::reset 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 */
391 void reset(const SkClipStack& stack) {
robertphillips@google.com80214e22012-07-20 15:33:18 +0000392 this->INHERITED::reset(stack, kBottom_IterStart);
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000393 }
394
395 private:
396
397 typedef Iter INHERITED;
reed@google.com5c3d1472011-02-22 19:12:23 +0000398 };
399
robertphillips@google.com607fe072012-07-24 13:54:00 +0000400 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000401 * GetConservativeBounds returns a conservative bound of the current clip.
402 * Since this could be the infinite plane (if inverse fills were involved) the
403 * maxWidth and maxHeight parameters can be used to limit the returned bound
robertphillips@google.com607fe072012-07-24 13:54:00 +0000404 * to the expected drawing area. Similarly, the offsetX and offsetY parameters
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000405 * allow the caller to offset the returned bound to account for translated
robertphillips@google.com607fe072012-07-24 13:54:00 +0000406 * drawing areas (i.e., those resulting from a saveLayer). For finite bounds,
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000407 * the translation (+offsetX, +offsetY) is applied before the clamp to the
robertphillips@google.com607fe072012-07-24 13:54:00 +0000408 * maximum rectangle: [0,maxWidth) x [0,maxHeight).
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000409 * isIntersectionOfRects is an optional parameter that is true when
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000410 * 'devBounds' is the result of an intersection of rects. In this case
411 * 'devBounds' is the exact answer/clip.
robertphillips@google.com607fe072012-07-24 13:54:00 +0000412 */
413 void getConservativeBounds(int offsetX,
414 int offsetY,
415 int maxWidth,
416 int maxHeight,
robertphillips@google.com7b112892012-07-31 15:18:21 +0000417 SkRect* devBounds,
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +0000418 bool* isIntersectionOfRects = NULL) const;
robertphillips@google.com607fe072012-07-24 13:54:00 +0000419
reed@google.com5c3d1472011-02-22 19:12:23 +0000420private:
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000421 friend class Iter;
reed@google.com5c3d1472011-02-22 19:12:23 +0000422
423 SkDeque fDeque;
424 int fSaveCount;
robertphillips@google.com46f93502012-08-07 15:38:08 +0000425
426 // Generation ID for the clip stack. This is incremented for each
427 // clipDevRect and clipDevPath call. 0 is reserved to indicate an
428 // invalid ID.
429 static int32_t gGenID;
430
robertphillips@google.com46f93502012-08-07 15:38:08 +0000431 /**
commit-bot@chromium.orge5b2af92014-02-16 13:25:24 +0000432 * Helper for clipDevPath, etc.
433 */
434 void pushElement(const Element& element);
435
436 /**
commit-bot@chromium.org6fbe54c2013-06-11 11:01:48 +0000437 * Restore the stack back to the specified save count.
438 */
439 void restoreTo(int saveCount);
440
441 /**
robertphillips@google.com46f93502012-08-07 15:38:08 +0000442 * Return the next unique generation ID.
443 */
444 static int32_t GetNextGenID();
reed@google.com5c3d1472011-02-22 19:12:23 +0000445};
446
447#endif