blob: 53bea82d3ff2524703a078cd70fdade8f47eb864 [file] [log] [blame]
junov@google.com4370aed2012-01-18 16:21:08 +00001/*
junov@chromium.org9ed02b92012-08-14 13:36:26 +00002 * Copyright 2012 Google Inc.
junov@google.com4370aed2012-01-18 16:21:08 +00003 *
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 SkDeferredCanvas_DEFINED
9#define SkDeferredCanvas_DEFINED
10
11#include "SkCanvas.h"
junov@google.com4370aed2012-01-18 16:21:08 +000012#include "SkPixelRef.h"
13
junov@chromium.org88e29142012-08-07 16:48:22 +000014class DeferredDevice;
junov@chromium.org77eec242012-07-18 17:54:45 +000015
junov@google.com4370aed2012-01-18 16:21:08 +000016/** \class SkDeferredCanvas
junov@chromium.org77eec242012-07-18 17:54:45 +000017 Subclass of SkCanvas that encapsulates an SkPicture or SkGPipe for deferred
18 drawing. The main difference between this class and SkPictureRecord (the
19 canvas provided by SkPicture) is that this is a full drop-in replacement
20 for SkCanvas, while SkPictureRecord only supports draw operations.
junov@google.com4370aed2012-01-18 16:21:08 +000021 SkDeferredCanvas will transparently trigger the flushing of deferred
vandebo@chromium.org74b46192012-01-28 01:45:11 +000022 draw operations when an attempt is made to access the pixel data.
junov@google.com4370aed2012-01-18 16:21:08 +000023*/
24class SK_API SkDeferredCanvas : public SkCanvas {
25public:
junov@chromium.org9ed02b92012-08-14 13:36:26 +000026 class NotificationClient;
junov@google.com4370aed2012-01-18 16:21:08 +000027
28 SkDeferredCanvas();
29
30 /** Construct a canvas with the specified device to draw into.
31 Equivalent to calling default constructor, then setDevice.
32 @param device Specifies a device for the canvas to draw into.
33 */
34 explicit SkDeferredCanvas(SkDevice* device);
35
junov@google.com4370aed2012-01-18 16:21:08 +000036 virtual ~SkDeferredCanvas();
37
38 /**
39 * Specify a device to be used by this canvas. Calling setDevice will
junov@chromium.org9ed02b92012-08-14 13:36:26 +000040 * release the previously set device, if any. Takes a reference on the
41 * device.
junov@google.com4370aed2012-01-18 16:21:08 +000042 *
43 * @param device The device that the canvas will raw into
44 * @return The device argument, for convenience.
45 */
46 virtual SkDevice* setDevice(SkDevice* device);
47
48 /**
junov@chromium.org9ed02b92012-08-14 13:36:26 +000049 * Specify a NotificationClient to be used by this canvas. Calling
rmistry@google.comfbfcd562012-08-23 18:09:54 +000050 * setNotificationClient will release the previously set
junov@chromium.org52805482012-08-20 14:25:04 +000051 * NotificationClient, if any. SkDeferredCanvas does not take ownership
52 * of the notification client. Therefore user code is resposible
53 * for its destruction. The notification client must be unregistered
54 * by calling setNotificationClient(NULL) if it is destroyed before
55 * this canvas.
junov@google.com4370aed2012-01-18 16:21:08 +000056 * Note: Must be called after the device is set with setDevice.
57 *
junov@chromium.org9ed02b92012-08-14 13:36:26 +000058 * @param notificationClient interface for dispatching notifications
59 * @return The notificationClient argument, for convenience.
junov@google.com4370aed2012-01-18 16:21:08 +000060 */
junov@chromium.org9ed02b92012-08-14 13:36:26 +000061 NotificationClient* setNotificationClient(NotificationClient* notificationClient);
junov@google.com4370aed2012-01-18 16:21:08 +000062
63 /**
64 * Enable or disable deferred drawing. When deferral is disabled,
65 * pending draw operations are immediately flushed and from then on,
66 * the SkDeferredCanvas behaves just like a regular SkCanvas.
67 * This method must not be called while the save/restore stack is in use.
68 * @param deferred true/false
69 */
70 void setDeferredDrawing(bool deferred);
71
junov@chromium.orgbfeddae2012-07-23 13:35:14 +000072 /**
junov@chromium.orgb10a6bd2012-07-25 17:27:13 +000073 * Returns true if deferred drawing is currenlty enabled.
74 */
junov@chromium.org88e29142012-08-07 16:48:22 +000075 bool isDeferredDrawing() const;
76
77 /**
rmistry@google.comfbfcd562012-08-23 18:09:54 +000078 * Returns true if the canvas contains a fresh frame. A frame is
junov@chromium.org88e29142012-08-07 16:48:22 +000079 * considered fresh when its content do not depend on the contents
80 * of the previous frame. For example, if a canvas is cleared before
81 * drawing each frame, the frames will all be considered fresh.
82 * A frame is defined as the graphics image produced by as a result
83 * of all the canvas draws operation executed between two successive
84 * calls to isFreshFrame. The result of isFreshFrame is computed
85 * conservatively, so it may report false negatives.
86 */
87 bool isFreshFrame() const;
junov@chromium.orgb10a6bd2012-07-25 17:27:13 +000088
89 /**
junov@chromium.orga38dfb62012-09-20 22:10:33 +000090 * Returns true if the canvas has recorded draw commands that have
91 * not yet been played back.
92 */
93 bool hasPendingCommands() const;
94
95 /**
junov@chromium.orgbfeddae2012-07-23 13:35:14 +000096 * Specify the maximum number of bytes to be allocated for the purpose
97 * of recording draw commands to this canvas. The default limit, is
98 * 64MB.
99 * @param maxStorage The maximum number of bytes to be allocated.
100 */
101 void setMaxRecordingStorage(size_t maxStorage);
102
junov@chromium.org2e14ba82012-08-07 14:26:57 +0000103 /**
104 * Returns the number of bytes currently allocated for the purpose of
105 * recording draw commands.
106 */
107 size_t storageAllocatedForRecording() const;
108
109 /**
110 * Attempt to reduce the storage allocated for recording by evicting
111 * cache resources.
112 * @param bytesToFree minimum number of bytes that should be attempted to
113 * be freed.
114 * @return number of bytes actually freed.
115 */
116 size_t freeMemoryIfPossible(size_t bytesToFree);
117
junov@chromium.orgfb103892012-09-20 19:35:43 +0000118 /**
sugoi@google.com7775fd52012-11-21 15:47:04 +0000119 * Specifies the maximum size (in bytes) allowed for a given image to be
120 * rendered using the deferred canvas.
121 */
122 void setBitmapSizeThreshold(size_t sizeThreshold);
123
124 /**
junov@chromium.orgfb103892012-09-20 19:35:43 +0000125 * Executes all pending commands without drawing
126 */
127 void silentFlush();
128
junov@google.com4370aed2012-01-18 16:21:08 +0000129 // Overrides of the SkCanvas interface
130 virtual int save(SaveFlags flags) SK_OVERRIDE;
131 virtual int saveLayer(const SkRect* bounds, const SkPaint* paint,
132 SaveFlags flags) SK_OVERRIDE;
133 virtual void restore() SK_OVERRIDE;
junov@chromium.orga907ac32012-02-24 21:54:07 +0000134 virtual bool isDrawingToLayer() const SK_OVERRIDE;
junov@google.com4370aed2012-01-18 16:21:08 +0000135 virtual bool translate(SkScalar dx, SkScalar dy) SK_OVERRIDE;
136 virtual bool scale(SkScalar sx, SkScalar sy) SK_OVERRIDE;
137 virtual bool rotate(SkScalar degrees) SK_OVERRIDE;
138 virtual bool skew(SkScalar sx, SkScalar sy) SK_OVERRIDE;
139 virtual bool concat(const SkMatrix& matrix) SK_OVERRIDE;
140 virtual void setMatrix(const SkMatrix& matrix) SK_OVERRIDE;
junov@google.com4370aed2012-01-18 16:21:08 +0000141 virtual bool clipRect(const SkRect& rect, SkRegion::Op op,
142 bool doAntiAlias) SK_OVERRIDE;
reed@google.com4ed0fb72012-12-12 20:48:18 +0000143 virtual bool clipRRect(const SkRRect& rect, SkRegion::Op op,
144 bool doAntiAlias) SK_OVERRIDE;
junov@google.com4370aed2012-01-18 16:21:08 +0000145 virtual bool clipPath(const SkPath& path, SkRegion::Op op,
146 bool doAntiAlias) SK_OVERRIDE;
147 virtual bool clipRegion(const SkRegion& deviceRgn,
148 SkRegion::Op op) SK_OVERRIDE;
149 virtual void clear(SkColor) SK_OVERRIDE;
150 virtual void drawPaint(const SkPaint& paint) SK_OVERRIDE;
151 virtual void drawPoints(PointMode mode, size_t count, const SkPoint pts[],
152 const SkPaint& paint) SK_OVERRIDE;
reed@google.com4ed0fb72012-12-12 20:48:18 +0000153 virtual void drawOval(const SkRect&, const SkPaint& paint) SK_OVERRIDE;
154 virtual void drawRect(const SkRect& rect, const SkPaint& paint) SK_OVERRIDE;
155 virtual void drawRRect(const SkRRect&, const SkPaint& paint) SK_OVERRIDE;
junov@google.com4370aed2012-01-18 16:21:08 +0000156 virtual void drawPath(const SkPath& path, const SkPaint& paint)
157 SK_OVERRIDE;
vandebo@chromium.org74b46192012-01-28 01:45:11 +0000158 virtual void drawBitmap(const SkBitmap& bitmap, SkScalar left,
junov@google.com4370aed2012-01-18 16:21:08 +0000159 SkScalar top, const SkPaint* paint)
160 SK_OVERRIDE;
reed@google.com71121732012-09-18 15:14:33 +0000161 virtual void drawBitmapRectToRect(const SkBitmap& bitmap, const SkRect* src,
junov@google.com4370aed2012-01-18 16:21:08 +0000162 const SkRect& dst, const SkPaint* paint)
163 SK_OVERRIDE;
164
165 virtual void drawBitmapMatrix(const SkBitmap& bitmap, const SkMatrix& m,
166 const SkPaint* paint) SK_OVERRIDE;
167 virtual void drawBitmapNine(const SkBitmap& bitmap, const SkIRect& center,
168 const SkRect& dst, const SkPaint* paint)
169 SK_OVERRIDE;
170 virtual void drawSprite(const SkBitmap& bitmap, int left, int top,
171 const SkPaint* paint) SK_OVERRIDE;
172 virtual void drawText(const void* text, size_t byteLength, SkScalar x,
173 SkScalar y, const SkPaint& paint) SK_OVERRIDE;
174 virtual void drawPosText(const void* text, size_t byteLength,
vandebo@chromium.org74b46192012-01-28 01:45:11 +0000175 const SkPoint pos[], const SkPaint& paint)
junov@google.com4370aed2012-01-18 16:21:08 +0000176 SK_OVERRIDE;
177 virtual void drawPosTextH(const void* text, size_t byteLength,
178 const SkScalar xpos[], SkScalar constY,
179 const SkPaint& paint) SK_OVERRIDE;
180 virtual void drawTextOnPath(const void* text, size_t byteLength,
181 const SkPath& path, const SkMatrix* matrix,
182 const SkPaint& paint) SK_OVERRIDE;
183 virtual void drawPicture(SkPicture& picture) SK_OVERRIDE;
184 virtual void drawVertices(VertexMode vmode, int vertexCount,
185 const SkPoint vertices[], const SkPoint texs[],
186 const SkColor colors[], SkXfermode* xmode,
187 const uint16_t indices[], int indexCount,
188 const SkPaint& paint) SK_OVERRIDE;
189 virtual SkBounder* setBounder(SkBounder* bounder) SK_OVERRIDE;
190 virtual SkDrawFilter* setDrawFilter(SkDrawFilter* filter) SK_OVERRIDE;
191
junov@google.com4370aed2012-01-18 16:21:08 +0000192public:
junov@chromium.org52805482012-08-20 14:25:04 +0000193 class NotificationClient {
junov@google.com4370aed2012-01-18 16:21:08 +0000194 public:
robertphillips@google.com59903972013-02-07 21:02:23 +0000195 virtual ~NotificationClient() {}
196
junov@chromium.org9ed02b92012-08-14 13:36:26 +0000197 /**
198 * Called before executing one or several draw commands, which means
199 * once per flush when deferred rendering is enabled.
200 */
junov@google.com4370aed2012-01-18 16:21:08 +0000201 virtual void prepareForDraw() {}
junov@chromium.org9ed02b92012-08-14 13:36:26 +0000202
203 /**
204 * Called after a recording a draw command if additional memory
205 * had to be allocated for recording.
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000206 * @param newAllocatedStorage same value as would be returned by
junov@chromium.org9ed02b92012-08-14 13:36:26 +0000207 * storageAllocatedForRecording(), for convenience.
208 */
209 virtual void storageAllocatedForRecordingChanged(
210 size_t newAllocatedStorage) {}
211
212 /**
213 * Called after pending draw commands have been flushed
214 */
215 virtual void flushedDrawCommands() {}
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000216
junov@google.com52a00ca2012-10-01 15:27:14 +0000217 /**
218 * Called after pending draw commands have been skipped, meaning
219 * that they were optimized-out because the canvas is cleared
220 * or completely overwritten by the command currently being recorded.
221 */
222 virtual void skippedPendingDrawCommands() {}
junov@google.com4370aed2012-01-18 16:21:08 +0000223 };
224
junov@chromium.org77eec242012-07-18 17:54:45 +0000225protected:
junov@google.com4370aed2012-01-18 16:21:08 +0000226 virtual SkCanvas* canvasForDrawIter();
junov@chromium.org88e29142012-08-07 16:48:22 +0000227 DeferredDevice* getDeferredDevice() const;
junov@google.com4370aed2012-01-18 16:21:08 +0000228
229private:
junov@chromium.org9ed02b92012-08-14 13:36:26 +0000230 void recordedDrawCommand();
junov@chromium.org5e5a0952012-02-28 15:27:59 +0000231 SkCanvas* drawingCanvas() const;
junov@chromium.org88e29142012-08-07 16:48:22 +0000232 SkCanvas* immediateCanvas() const;
junov@google.com4370aed2012-01-18 16:21:08 +0000233 bool isFullFrame(const SkRect*, const SkPaint*) const;
234 void validate() const;
235 void init();
junov@chromium.org5e5a0952012-02-28 15:27:59 +0000236 bool fDeferredDrawing;
junov@google.com4370aed2012-01-18 16:21:08 +0000237
junov@chromium.org88e29142012-08-07 16:48:22 +0000238 friend class SkDeferredCanvasTester; // for unit testing
junov@google.com4370aed2012-01-18 16:21:08 +0000239 typedef SkCanvas INHERITED;
240};
241
242
243#endif