blob: 443e6ad7d9464eef63ff33886da07cd47bb33391 [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"
12#include "SkRegion.h"
robertphillips@google.com46f93502012-08-07 15:38:08 +000013#include "SkTDArray.h"
reed@google.com5c3d1472011-02-22 19:12:23 +000014
bsalomon@google.com57788b52011-02-22 21:00:31 +000015struct SkRect;
reed@google.com5c3d1472011-02-22 19:12:23 +000016class SkPath;
17
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +000018// Because a single save/restore state can have multiple clips, this class
19// stores the stack depth (fSaveCount) and clips (fDeque) separately.
20// Each clip in fDeque stores the stack state to which it belongs
21// (i.e., the fSaveCount in force when it was added). Restores are thus
22// implemented by removing clips from fDeque that have an fSaveCount larger
23// then the freshly decremented count.
ctguil@chromium.org7ffb1b22011-03-15 21:27:08 +000024class SK_API SkClipStack {
reed@google.com5c3d1472011-02-22 19:12:23 +000025public:
26 SkClipStack();
vandebo@chromium.org1e1c36f2011-05-03 16:26:09 +000027 SkClipStack(const SkClipStack& b);
robertphillips@google.comcc6493b2012-07-26 18:39:13 +000028 explicit SkClipStack(const SkRect& r);
robertphillips@google.com641f8b12012-07-31 19:15:58 +000029 explicit SkClipStack(const SkIRect& r);
vandebo@chromium.org610f7162012-03-14 18:34:15 +000030 ~SkClipStack();
reed@google.com5c3d1472011-02-22 19:12:23 +000031
vandebo@chromium.org1e1c36f2011-05-03 16:26:09 +000032 SkClipStack& operator=(const SkClipStack& b);
33 bool operator==(const SkClipStack& b) const;
34 bool operator!=(const SkClipStack& b) const { return !(*this == b); }
35
reed@google.com5c3d1472011-02-22 19:12:23 +000036 void reset();
37
38 int getSaveCount() const { return fSaveCount; }
39 void save();
40 void restore();
41
robertphillips@google.com607fe072012-07-24 13:54:00 +000042 enum BoundsType {
43 // The bounding box contains all the pixels that can be written to
44 kNormal_BoundsType,
45 // The bounding box contains all the pixels that cannot be written to.
46 // The real bound extends out to infinity and all the pixels outside
47 // of the bound can be written to. Note that some of the pixels inside
rmistry@google.comfbfcd562012-08-23 18:09:54 +000048 // the bound may also be writeable but all pixels that cannot be
robertphillips@google.com607fe072012-07-24 13:54:00 +000049 // written to are guaranteed to be inside.
50 kInsideOut_BoundsType
51 };
52
53 /**
54 * getBounds places the current finite bound in its first parameter. In its
rmistry@google.comfbfcd562012-08-23 18:09:54 +000055 * second, it indicates which kind of bound is being returned. If
robertphillips@google.com7b112892012-07-31 15:18:21 +000056 * 'canvFiniteBound' is a normal bounding box then it encloses all writeable
rmistry@google.comfbfcd562012-08-23 18:09:54 +000057 * pixels. If 'canvFiniteBound' is an inside out bounding box then it
robertphillips@google.com607fe072012-07-24 13:54:00 +000058 * encloses all the un-writeable pixels and the true/normal bound is the
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +000059 * infinite plane. isIntersectionOfRects is an optional parameter
robertphillips@google.com7b112892012-07-31 15:18:21 +000060 * that is true if 'canvFiniteBound' resulted from an intersection of rects.
robertphillips@google.com607fe072012-07-24 13:54:00 +000061 */
rmistry@google.comfbfcd562012-08-23 18:09:54 +000062 void getBounds(SkRect* canvFiniteBound,
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +000063 BoundsType* boundType,
64 bool* isIntersectionOfRects = NULL) const;
robertphillips@google.com607fe072012-07-24 13:54:00 +000065
bsalomon@google.com3ab43d52012-10-11 19:39:09 +000066 /**
67 * Takes an input rect in device space and conservatively clips it to the
68 * clip-stack. If false is returned then the rect does not intersect the
69 * clip and is unmodified.
70 */
71 bool intersectRectWithClip(SkRect* devRect) const;
72
reed@google.com115d9312012-05-16 18:50:40 +000073 void clipDevRect(const SkIRect& ir, SkRegion::Op op) {
reed@google.com5c3d1472011-02-22 19:12:23 +000074 SkRect r;
75 r.set(ir);
reed@google.com00177082011-10-12 14:34:30 +000076 this->clipDevRect(r, op, false);
reed@google.com5c3d1472011-02-22 19:12:23 +000077 }
reed@google.com00177082011-10-12 14:34:30 +000078 void clipDevRect(const SkRect&, SkRegion::Op, bool doAA);
79 void clipDevPath(const SkPath&, SkRegion::Op, bool doAA);
reed@google.com0557d9e2012-08-16 15:59:59 +000080 // An optimized version of clipDevRect(emptyRect, kIntersect, ...)
81 void clipEmpty();
reed@google.com5c3d1472011-02-22 19:12:23 +000082
robertphillips@google.comcc6493b2012-07-26 18:39:13 +000083 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +000084 * isWideOpen returns true if the clip state corresponds to the infinite
robertphillips@google.comcc6493b2012-07-26 18:39:13 +000085 * plane (i.e., draws are not limited at all)
86 */
87 bool isWideOpen() const;
88
robertphillips@google.com46f93502012-08-07 15:38:08 +000089 /**
90 * Add a callback function that will be called whenever a clip state
91 * is no longer viable. This will occur whenever restore
rmistry@google.comfbfcd562012-08-23 18:09:54 +000092 * is called or when a clipDevRect or clipDevPath call updates the
robertphillips@google.com46f93502012-08-07 15:38:08 +000093 * clip within an existing save/restore state. Each clip state is
94 * represented by a unique generation ID.
95 */
96 typedef void (*PFPurgeClipCB)(int genID, void* data);
97 void addPurgeClipCallback(PFPurgeClipCB callback, void* data) const;
98
99 /**
100 * Remove a callback added earlier via addPurgeClipCallback
101 */
102 void removePurgeClipCallback(PFPurgeClipCB callback, void* data) const;
103
104 /**
105 * The generation ID has three reserved values to indicate special
bsalomon@google.come8ca6c62012-11-07 21:19:10 +0000106 * (potentially ignorable) cases
robertphillips@google.com46f93502012-08-07 15:38:08 +0000107 */
108 static const int32_t kInvalidGenID = 0;
109 static const int32_t kEmptyGenID = 1; // no pixels writeable
110 static const int32_t kWideOpenGenID = 2; // all pixels writeable
111
robertphillips@google.com73e71022012-08-09 18:10:49 +0000112 int32_t getTopmostGenID() const;
113
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000114private:
115 struct Rec;
116
117public:
118 class Iter {
reed@google.com5c3d1472011-02-22 19:12:23 +0000119 public:
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000120 enum IterStart {
robertphillips@google.com80214e22012-07-20 15:33:18 +0000121 kBottom_IterStart = SkDeque::Iter::kFront_IterStart,
122 kTop_IterStart = SkDeque::Iter::kBack_IterStart
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000123 };
124
bsalomon@google.comd302f142011-03-03 13:54:13 +0000125 /**
126 * Creates an uninitialized iterator. Must be reset()
127 */
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000128 Iter();
bsalomon@google.comd302f142011-03-03 13:54:13 +0000129
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000130 Iter(const SkClipStack& stack, IterStart startLoc);
reed@google.com5c3d1472011-02-22 19:12:23 +0000131
132 struct Clip {
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000133 Clip() : fRect(NULL), fPath(NULL), fOp(SkRegion::kIntersect_Op),
robertphillips@google.comfa1d2912012-04-16 14:49:14 +0000134 fDoAA(false) {}
vandebo@chromium.org9fbdf872011-05-09 07:55:58 +0000135 friend bool operator==(const Clip& a, const Clip& b);
vandebo@chromium.org8887ede2011-05-25 01:27:52 +0000136 friend bool operator!=(const Clip& a, const Clip& b);
bsalomon@google.com7b7cdd12012-11-07 16:17:24 +0000137 /**
138 * Gets the bounds of the clip element, either the rect or path bounds.
139 */
140 const SkRect& getBounds() const;
141
reed@google.com5c3d1472011-02-22 19:12:23 +0000142 const SkRect* fRect; // if non-null, this is a rect clip
143 const SkPath* fPath; // if non-null, this is a path clip
144 SkRegion::Op fOp;
reed@google.com00177082011-10-12 14:34:30 +0000145 bool fDoAA;
robertphillips@google.com73e71022012-08-09 18:10:49 +0000146 int32_t fGenID;
reed@google.com5c3d1472011-02-22 19:12:23 +0000147 };
148
149 /**
150 * Return the clip for this element in the iterator. If next() returns
151 * NULL, then the iterator is done. The type of clip is determined by
152 * the pointers fRect and fPath:
153 *
154 * fRect==NULL fPath!=NULL path clip
155 * fRect!=NULL fPath==NULL rect clip
156 * fRect==NULL fPath==NULL empty clip
157 */
158 const Clip* next();
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000159 const Clip* prev();
reed@google.com5c3d1472011-02-22 19:12:23 +0000160
bsalomon@google.comd302f142011-03-03 13:54:13 +0000161 /**
bsalomon@google.come8ca6c62012-11-07 21:19:10 +0000162 * This is a variant of next() that greedily attempts to combine elements when possible.
163 * Currently it attempts to combine intersecting rectangles, though it may do more in the
164 * future. The returned Clip may not refer to a single element in the stack, so its
165 * generation ID may be invalid.
166 */
167 const Clip* nextCombined();
168
169 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000170 * Moves the iterator to the topmost clip with the specified RegionOp
171 * and returns that clip. If no clip with that op is found,
robertphillips@google.com5836b6d2012-07-18 12:06:15 +0000172 * returns NULL.
173 */
robertphillips@google.com80214e22012-07-20 15:33:18 +0000174 const Clip* skipToTopmost(SkRegion::Op op);
robertphillips@google.com5836b6d2012-07-18 12:06:15 +0000175
176 /**
bsalomon@google.comd302f142011-03-03 13:54:13 +0000177 * Restarts the iterator on a clip stack.
178 */
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000179 void reset(const SkClipStack& stack, IterStart startLoc);
bsalomon@google.comd302f142011-03-03 13:54:13 +0000180
reed@google.com5c3d1472011-02-22 19:12:23 +0000181 private:
robertphillips@google.com5836b6d2012-07-18 12:06:15 +0000182 const SkClipStack* fStack;
183 Clip fClip;
184 SkDeque::Iter fIter;
bsalomon@google.come8ca6c62012-11-07 21:19:10 +0000185 SkRect fCombinedRect; // used for nextCombined()
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000186 /**
187 * updateClip updates fClip to the current state of fIter. It unifies
188 * functionality needed by both next() and prev().
189 */
190 const Clip* updateClip(const SkClipStack::Rec* rec);
191 };
192
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000193 /**
robertphillips@google.com80214e22012-07-20 15:33:18 +0000194 * The B2TIter iterates from the bottom of the stack to the top.
195 * It inherits privately from Iter to prevent access to reverse iteration.
196 */
197 class B2TIter : private Iter {
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000198 public:
robertphillips@google.com80214e22012-07-20 15:33:18 +0000199 B2TIter() {}
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000200
201 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000202 * Wrap Iter's 2 parameter ctor to force initialization to the
robertphillips@google.com80214e22012-07-20 15:33:18 +0000203 * beginning of the deque/bottom of the stack
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000204 */
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000205 B2TIter(const SkClipStack& stack)
robertphillips@google.com80214e22012-07-20 15:33:18 +0000206 : INHERITED(stack, kBottom_IterStart) {
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000207 }
208
209 using Iter::Clip;
210 using Iter::next;
211
212 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000213 * Wrap Iter::reset to force initialization to the
robertphillips@google.com80214e22012-07-20 15:33:18 +0000214 * beginning of the deque/bottom of the stack
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000215 */
216 void reset(const SkClipStack& stack) {
robertphillips@google.com80214e22012-07-20 15:33:18 +0000217 this->INHERITED::reset(stack, kBottom_IterStart);
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000218 }
219
220 private:
221
222 typedef Iter INHERITED;
reed@google.com5c3d1472011-02-22 19:12:23 +0000223 };
224
robertphillips@google.com607fe072012-07-24 13:54:00 +0000225 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000226 * GetConservativeBounds returns a conservative bound of the current clip.
227 * Since this could be the infinite plane (if inverse fills were involved) the
228 * maxWidth and maxHeight parameters can be used to limit the returned bound
robertphillips@google.com607fe072012-07-24 13:54:00 +0000229 * to the expected drawing area. Similarly, the offsetX and offsetY parameters
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000230 * allow the caller to offset the returned bound to account for translated
robertphillips@google.com607fe072012-07-24 13:54:00 +0000231 * drawing areas (i.e., those resulting from a saveLayer). For finite bounds,
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000232 * the translation (+offsetX, +offsetY) is applied before the clamp to the
robertphillips@google.com607fe072012-07-24 13:54:00 +0000233 * maximum rectangle: [0,maxWidth) x [0,maxHeight).
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000234 * isIntersectionOfRects is an optional parameter that is true when
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000235 * 'devBounds' is the result of an intersection of rects. In this case
236 * 'devBounds' is the exact answer/clip.
robertphillips@google.com607fe072012-07-24 13:54:00 +0000237 */
238 void getConservativeBounds(int offsetX,
239 int offsetY,
240 int maxWidth,
241 int maxHeight,
robertphillips@google.com7b112892012-07-31 15:18:21 +0000242 SkRect* devBounds,
robertphillips@google.com4c2a2f72012-07-24 22:07:50 +0000243 bool* isIntersectionOfRects = NULL) const;
robertphillips@google.com607fe072012-07-24 13:54:00 +0000244
reed@google.com5c3d1472011-02-22 19:12:23 +0000245private:
robertphillips@google.com52cb2c72012-07-16 18:52:29 +0000246 friend class Iter;
reed@google.com5c3d1472011-02-22 19:12:23 +0000247
248 SkDeque fDeque;
249 int fSaveCount;
robertphillips@google.com46f93502012-08-07 15:38:08 +0000250
251 // Generation ID for the clip stack. This is incremented for each
252 // clipDevRect and clipDevPath call. 0 is reserved to indicate an
253 // invalid ID.
254 static int32_t gGenID;
255
256 struct ClipCallbackData {
257 PFPurgeClipCB fCallback;
258 void* fData;
259
260 friend bool operator==(const ClipCallbackData& a,
261 const ClipCallbackData& b) {
262 return a.fCallback == b.fCallback && a.fData == b.fData;
263 }
264 };
265
266 mutable SkTDArray<ClipCallbackData> fCallbackData;
267
268 /**
269 * Invoke all the purge callbacks passing in rec's generation ID.
270 */
271 void purgeClip(Rec* rec);
272
273 /**
274 * Return the next unique generation ID.
275 */
276 static int32_t GetNextGenID();
reed@google.com5c3d1472011-02-22 19:12:23 +0000277};
278
279#endif
280