Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(322)

Unified Diff: src/utils/SkDeferredCanvas.cpp

Issue 1269093002: remove SkDeferredCanvas (Closed) Base URL: https://skia.googlesource.com/skia.git@master
Patch Set: Created 5 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « src/core/SkPictureRecord.cpp ('k') | tests/CanvasTest.cpp » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: src/utils/SkDeferredCanvas.cpp
diff --git a/src/utils/SkDeferredCanvas.cpp b/src/utils/SkDeferredCanvas.cpp
deleted file mode 100644
index 6ccfb5e4b3c6711fe402e1726c697de949f512fb..0000000000000000000000000000000000000000
--- a/src/utils/SkDeferredCanvas.cpp
+++ /dev/null
@@ -1,1027 +0,0 @@
-
-/*
- * Copyright 2013 Google Inc.
- *
- * Use of this source code is governed by a BSD-style license that can be
- * found in the LICENSE file.
- */
-
-#include "SkDeferredCanvas.h"
-
-#include "SkChunkAlloc.h"
-#include "SkClipStack.h"
-#include "SkColorFilter.h"
-#include "SkDevice.h"
-#include "SkDrawFilter.h"
-#include "SkGPipe.h"
-#include "SkImage_Base.h"
-#include "SkPaint.h"
-#include "SkPaintPriv.h"
-#include "SkRRect.h"
-#include "SkShader.h"
-#include "SkSurface.h"
-
-enum {
- // Deferred canvas will auto-flush when recording reaches this limit
- kDefaultMaxRecordingStorageBytes = 64*1024*1024,
- kDeferredCanvasBitmapSizeThreshold = ~0U, // Disables this feature
-
- kNoSaveLayerIndex = -1,
-};
-
-enum PlaybackMode {
- kNormal_PlaybackMode,
- kSilent_PlaybackMode,
-};
-
-static uint64_t image_area(const SkImage* image) {
- return sk_64_mul(image->width(), image->height());
-}
-
-// HACK -- see crbug.com/485243
-//
-// Work around case where Blink gives us an image, but will "mutate" it (by changing its contents
-// directly using webgl). Until that is fixed at the call-site, we treat gpu-backed-images as
-// mutable for now (at least for the purposes of deferred canvas)
-//
-static bool should_draw_gpu_image_immediately(const SkImage* image) {
- return as_IB(image)->getTexture() != NULL;
-}
-
-static bool should_draw_immediately(const SkBitmap* bitmap, const SkImage* image,
- const SkPaint* paint, size_t bitmapSizeThreshold) {
- if (bitmap && ((bitmap->getTexture() && !bitmap->isImmutable()) ||
- (bitmap->getSize() > bitmapSizeThreshold))) {
- return true;
- }
- if (image) {
- if (should_draw_gpu_image_immediately(image) || image_area(image) > bitmapSizeThreshold) {
- return true;
- }
- }
- if (paint) {
- SkShader* shader = paint->getShader();
- // Here we detect the case where the shader is an SkBitmapProcShader
- // with a gpu texture attached. Checking this without RTTI
- // requires making the assumption that only gradient shaders
- // and SkBitmapProcShader implement asABitmap(). The following
- // code may need to be revised if that assumption is ever broken.
- if (shader && !shader->asAGradient(NULL)) {
- SkBitmap bm;
- if (shader->asABitmap(&bm, NULL, NULL) &&
- bm.getTexture()) {
- return true;
- }
- }
- }
- return false;
-}
-
-//-----------------------------------------------------------------------------
-// DeferredPipeController
-//-----------------------------------------------------------------------------
-
-class DeferredPipeController : public SkGPipeController {
-public:
- DeferredPipeController();
- void setPlaybackCanvas(SkCanvas*);
- virtual ~DeferredPipeController();
- void* requestBlock(size_t minRequest, size_t* actual) override;
- void notifyWritten(size_t bytes) override;
- void playback(bool silent);
- bool hasPendingCommands() const { return fAllocator.totalUsed() != 0; }
- size_t storageAllocatedForRecording() const { return fAllocator.totalCapacity(); }
-private:
- enum {
- kMinBlockSize = 4096
- };
- struct PipeBlock {
- PipeBlock(void* block, size_t size) { fBlock = block, fSize = size; }
- void* fBlock;
- size_t fSize;
- };
- void* fBlock;
- size_t fBytesWritten;
- SkChunkAlloc fAllocator;
- SkTDArray<PipeBlock> fBlockList;
- SkGPipeReader fReader;
-};
-
-DeferredPipeController::DeferredPipeController() :
- fAllocator(kMinBlockSize) {
- fBlock = NULL;
- fBytesWritten = 0;
-}
-
-DeferredPipeController::~DeferredPipeController() {
- fAllocator.reset();
-}
-
-void DeferredPipeController::setPlaybackCanvas(SkCanvas* canvas) {
- fReader.setCanvas(canvas);
-}
-
-void* DeferredPipeController::requestBlock(size_t minRequest, size_t *actual) {
- if (fBlock) {
- // Save the previous block for later
- PipeBlock previousBloc(fBlock, fBytesWritten);
- fBlockList.push(previousBloc);
- }
- size_t blockSize = SkTMax<size_t>(minRequest, kMinBlockSize);
- fBlock = fAllocator.allocThrow(blockSize);
- fBytesWritten = 0;
- *actual = blockSize;
- return fBlock;
-}
-
-void DeferredPipeController::notifyWritten(size_t bytes) {
- fBytesWritten += bytes;
-}
-
-void DeferredPipeController::playback(bool silent) {
- uint32_t flags = silent ? SkGPipeReader::kSilent_PlaybackFlag : 0;
- for (int currentBlock = 0; currentBlock < fBlockList.count(); currentBlock++ ) {
- fReader.playback(fBlockList[currentBlock].fBlock, fBlockList[currentBlock].fSize,
- flags);
- }
- fBlockList.reset();
-
- if (fBlock) {
- fReader.playback(fBlock, fBytesWritten, flags);
- fBlock = NULL;
- }
-
- // Release all allocated blocks
- fAllocator.reset();
-
- this->purgeCaches();
-}
-
-//-----------------------------------------------------------------------------
-// SkDeferredDevice
-//-----------------------------------------------------------------------------
-class SkDeferredDevice : public SkBaseDevice {
-public:
- explicit SkDeferredDevice(SkSurface* surface);
- ~SkDeferredDevice();
-
- void setNotificationClient(SkDeferredCanvas::NotificationClient* notificationClient);
- SkCanvas* recordingCanvas();
- SkCanvas* immediateCanvas() const {return fImmediateCanvas;}
- SkBaseDevice* immediateDevice() const {return fImmediateCanvas->getTopDevice();}
- SkImage* newImageSnapshot();
- void setSurface(SkSurface* surface);
- bool isFreshFrame();
- bool hasPendingCommands();
- size_t storageAllocatedForRecording() const;
- size_t freeMemoryIfPossible(size_t bytesToFree);
- void flushPendingCommands(PlaybackMode);
- void skipPendingCommands();
- void setMaxRecordingStorage(size_t);
- void recordedDrawCommand();
- void setIsDrawingToLayer(bool value) {fIsDrawingToLayer = value;}
-
- SkImageInfo imageInfo() const override;
-
- GrRenderTarget* accessRenderTarget() override;
-
- SkBaseDevice* onCreateDevice(const CreateInfo&, const SkPaint*) override;
-
- SkSurface* newSurface(const SkImageInfo&, const SkSurfaceProps&) override;
-
-protected:
- const SkBitmap& onAccessBitmap() override;
- bool onReadPixels(const SkImageInfo&, void*, size_t, int x, int y) override;
- bool onWritePixels(const SkImageInfo&, const void*, size_t, int x, int y) override;
-
- // None of the following drawing methods should ever get called on the
- // deferred device
- void drawPaint(const SkDraw&, const SkPaint& paint) override
- {SkASSERT(0);}
- void drawPoints(const SkDraw&, SkCanvas::PointMode mode,
- size_t count, const SkPoint[],
- const SkPaint& paint) override
- {SkASSERT(0);}
- void drawRect(const SkDraw&, const SkRect& r,
- const SkPaint& paint) override
- {SkASSERT(0);}
- void drawOval(const SkDraw&, const SkRect&, const SkPaint&) override
- {SkASSERT(0);}
- void drawRRect(const SkDraw&, const SkRRect& rr,
- const SkPaint& paint) override
- {SkASSERT(0);}
- void drawPath(const SkDraw&, const SkPath& path,
- const SkPaint& paint,
- const SkMatrix* prePathMatrix = NULL,
- bool pathIsMutable = false) override
- {SkASSERT(0);}
- void drawBitmap(const SkDraw&, const SkBitmap& bitmap,
- const SkMatrix& matrix, const SkPaint& paint) override
- {SkASSERT(0);}
- void drawBitmapRect(const SkDraw&, const SkBitmap&, const SkRect*,
- const SkRect&, const SkPaint&, SkCanvas::SrcRectConstraint) override
- {SkASSERT(0);}
- void drawSprite(const SkDraw&, const SkBitmap& bitmap,
- int x, int y, const SkPaint& paint) override
- {SkASSERT(0);}
- void drawImage(const SkDraw&, const SkImage*, SkScalar, SkScalar, const SkPaint&) override
- {SkASSERT(0);}
- void drawImageRect(const SkDraw&, const SkImage*, const SkRect*, const SkRect&,
- const SkPaint&, SkCanvas::SrcRectConstraint) override
- {SkASSERT(0);}
- void drawImageNine(const SkDraw&, const SkImage*, const SkIRect&, const SkRect&,
- const SkPaint&) override
- {SkASSERT(0);}
- void drawText(const SkDraw&, const void* text, size_t len,
- SkScalar x, SkScalar y, const SkPaint& paint) override
- {SkASSERT(0);}
- void drawPosText(const SkDraw&, const void* text, size_t len,
- const SkScalar pos[], int scalarsPerPos,
- const SkPoint& offset, const SkPaint& paint) override
- {SkASSERT(0);}
- void drawTextOnPath(const SkDraw&, const void* text,
- size_t len, const SkPath& path,
- const SkMatrix* matrix,
- const SkPaint& paint) override
- {SkASSERT(0);}
- void drawVertices(const SkDraw&, SkCanvas::VertexMode,
- int vertexCount, const SkPoint verts[],
- const SkPoint texs[], const SkColor colors[],
- SkXfermode* xmode, const uint16_t indices[],
- int indexCount, const SkPaint& paint) override
- {SkASSERT(0);}
- void drawPatch(const SkDraw&, const SkPoint cubics[12], const SkColor colors[4],
- const SkPoint texCoords[4], SkXfermode* xmode,
- const SkPaint& paint) override
- {SkASSERT(0);}
- void drawAtlas(const SkDraw&, const SkImage* atlas, const SkRSXform[], const SkRect[],
- const SkColor[], int count, SkXfermode::Mode, const SkPaint&) override
- {SkASSERT(0);}
-
- void drawDevice(const SkDraw&, SkBaseDevice*, int x, int y,
- const SkPaint&) override
- {SkASSERT(0);}
-
- bool canHandleImageFilter(const SkImageFilter*) override {
- return false;
- }
- bool filterImage(const SkImageFilter*, const SkBitmap&,
- const SkImageFilter::Context&, SkBitmap*, SkIPoint*) override {
- return false;
- }
-
-private:
- void flush() override;
- void replaceBitmapBackendForRasterSurface(const SkBitmap&) override {}
-
- void beginRecording();
- void init();
- void aboutToDraw();
- void prepareForImmediatePixelWrite();
-
- DeferredPipeController fPipeController;
- SkGPipeWriter fPipeWriter;
- SkCanvas* fImmediateCanvas;
- SkCanvas* fRecordingCanvas;
- SkSurface* fSurface;
- SkDeferredCanvas::NotificationClient* fNotificationClient;
- bool fFreshFrame;
- bool fCanDiscardCanvasContents;
- bool fIsDrawingToLayer;
- size_t fMaxRecordingStorageBytes;
- size_t fPreviousStorageAllocated;
-
- typedef SkBaseDevice INHERITED;
-};
-
-SkDeferredDevice::SkDeferredDevice(SkSurface* surface)
- : INHERITED(surface->props()) {
- fMaxRecordingStorageBytes = kDefaultMaxRecordingStorageBytes;
- fNotificationClient = NULL;
- fImmediateCanvas = NULL;
- fSurface = NULL;
- this->setSurface(surface);
- this->init();
-}
-
-void SkDeferredDevice::setSurface(SkSurface* surface) {
- SkRefCnt_SafeAssign(fImmediateCanvas, surface->getCanvas());
- SkRefCnt_SafeAssign(fSurface, surface);
- fPipeController.setPlaybackCanvas(fImmediateCanvas);
-}
-
-void SkDeferredDevice::init() {
- fRecordingCanvas = NULL;
- fFreshFrame = true;
- fIsDrawingToLayer = false;
- fCanDiscardCanvasContents = false;
- fPreviousStorageAllocated = 0;
- fMaxRecordingStorageBytes = kDefaultMaxRecordingStorageBytes;
- fNotificationClient = NULL;
- this->beginRecording();
-}
-
-SkDeferredDevice::~SkDeferredDevice() {
- this->flushPendingCommands(kSilent_PlaybackMode);
- SkSafeUnref(fImmediateCanvas);
- SkSafeUnref(fSurface);
-}
-
-void SkDeferredDevice::setMaxRecordingStorage(size_t maxStorage) {
- fMaxRecordingStorageBytes = maxStorage;
- this->recordingCanvas(); // Accessing the recording canvas applies the new limit.
-}
-
-void SkDeferredDevice::beginRecording() {
- SkASSERT(NULL == fRecordingCanvas);
- fRecordingCanvas = fPipeWriter.startRecording(&fPipeController, 0,
- immediateDevice()->width(), immediateDevice()->height());
-}
-
-void SkDeferredDevice::setNotificationClient(
- SkDeferredCanvas::NotificationClient* notificationClient) {
- fNotificationClient = notificationClient;
-}
-
-void SkDeferredDevice::skipPendingCommands() {
- if (!fIsDrawingToLayer) {
- fCanDiscardCanvasContents = true;
- if (fPipeController.hasPendingCommands()) {
- fFreshFrame = true;
- flushPendingCommands(kSilent_PlaybackMode);
- }
- }
-}
-
-bool SkDeferredDevice::isFreshFrame() {
- bool ret = fFreshFrame;
- fFreshFrame = false;
- return ret;
-}
-
-bool SkDeferredDevice::hasPendingCommands() {
- return fPipeController.hasPendingCommands();
-}
-
-void SkDeferredDevice::aboutToDraw() {
- if (fNotificationClient) {
- fNotificationClient->prepareForDraw();
- }
- if (fCanDiscardCanvasContents) {
- if (fSurface) {
- fSurface->notifyContentWillChange(SkSurface::kDiscard_ContentChangeMode);
- }
- fCanDiscardCanvasContents = false;
- }
-}
-
-void SkDeferredDevice::flushPendingCommands(PlaybackMode playbackMode) {
- if (!fPipeController.hasPendingCommands()) {
- return;
- }
- if (playbackMode == kNormal_PlaybackMode) {
- aboutToDraw();
- }
- fPipeWriter.flushRecording(true);
- fPipeController.playback(kSilent_PlaybackMode == playbackMode);
- if (fNotificationClient) {
- if (playbackMode == kSilent_PlaybackMode) {
- fNotificationClient->skippedPendingDrawCommands();
- } else {
- fNotificationClient->flushedDrawCommands();
- }
- }
-
- fPreviousStorageAllocated = storageAllocatedForRecording();
-}
-
-void SkDeferredDevice::flush() {
- this->flushPendingCommands(kNormal_PlaybackMode);
- fImmediateCanvas->flush();
-}
-
-size_t SkDeferredDevice::freeMemoryIfPossible(size_t bytesToFree) {
- size_t val = fPipeWriter.freeMemoryIfPossible(bytesToFree);
- fPreviousStorageAllocated = storageAllocatedForRecording();
- return val;
-}
-
-size_t SkDeferredDevice::storageAllocatedForRecording() const {
- return (fPipeController.storageAllocatedForRecording()
- + fPipeWriter.storageAllocatedForRecording());
-}
-
-void SkDeferredDevice::recordedDrawCommand() {
- size_t storageAllocated = this->storageAllocatedForRecording();
-
- if (storageAllocated > fMaxRecordingStorageBytes) {
- // First, attempt to reduce cache without flushing
- size_t tryFree = storageAllocated - fMaxRecordingStorageBytes;
- if (this->freeMemoryIfPossible(tryFree) < tryFree) {
- // Flush is necessary to free more space.
- this->flushPendingCommands(kNormal_PlaybackMode);
- // Free as much as possible to avoid oscillating around fMaxRecordingStorageBytes
- // which could cause a high flushing frequency.
- this->freeMemoryIfPossible(~0U);
- }
- storageAllocated = this->storageAllocatedForRecording();
- }
-
- if (fNotificationClient &&
- storageAllocated != fPreviousStorageAllocated) {
- fPreviousStorageAllocated = storageAllocated;
- fNotificationClient->storageAllocatedForRecordingChanged(storageAllocated);
- }
-}
-
-SkCanvas* SkDeferredDevice::recordingCanvas() {
- return fRecordingCanvas;
-}
-
-SkImage* SkDeferredDevice::newImageSnapshot() {
- this->flush();
- return fSurface ? fSurface->newImageSnapshot() : NULL;
-}
-
-SkImageInfo SkDeferredDevice::imageInfo() const {
- return immediateDevice()->imageInfo();
-}
-
-GrRenderTarget* SkDeferredDevice::accessRenderTarget() {
- this->flushPendingCommands(kNormal_PlaybackMode);
- return immediateDevice()->accessRenderTarget();
-}
-
-void SkDeferredDevice::prepareForImmediatePixelWrite() {
- // The purpose of the following code is to make sure commands are flushed, that
- // aboutToDraw() is called and that notifyContentWillChange is called, without
- // calling anything redundantly.
- if (fPipeController.hasPendingCommands()) {
- this->flushPendingCommands(kNormal_PlaybackMode);
- } else {
- bool mustNotifyDirectly = !fCanDiscardCanvasContents;
- this->aboutToDraw();
- if (mustNotifyDirectly) {
- fSurface->notifyContentWillChange(SkSurface::kRetain_ContentChangeMode);
- }
- }
-
- fImmediateCanvas->flush();
-}
-
-bool SkDeferredDevice::onWritePixels(const SkImageInfo& info, const void* pixels, size_t rowBytes,
- int x, int y) {
- SkASSERT(x >= 0 && y >= 0);
- SkASSERT(x + info.width() <= width());
- SkASSERT(y + info.height() <= height());
-
- const SkImageInfo deviceInfo = this->imageInfo();
- if (info.width() == deviceInfo.width() && info.height() == deviceInfo.height()) {
- this->skipPendingCommands();
- } else {
- this->flushPendingCommands(kNormal_PlaybackMode);
- }
-
- this->prepareForImmediatePixelWrite();
- return immediateDevice()->onWritePixels(info, pixels, rowBytes, x, y);
-}
-
-const SkBitmap& SkDeferredDevice::onAccessBitmap() {
- this->flushPendingCommands(kNormal_PlaybackMode);
- return immediateDevice()->accessBitmap(false);
-}
-
-SkBaseDevice* SkDeferredDevice::onCreateDevice(const CreateInfo& cinfo, const SkPaint* layerPaint) {
- // Create a compatible non-deferred device.
- // We do not create a deferred device because we know the new device
- // will not be used with a deferred canvas (there is no API for that).
- // And connecting a SkDeferredDevice to non-deferred canvas can result
- // in unpredictable behavior.
- return this->immediateDevice()->onCreateDevice(cinfo, layerPaint);
-}
-
-SkSurface* SkDeferredDevice::newSurface(const SkImageInfo& info, const SkSurfaceProps& props) {
- return this->immediateDevice()->newSurface(info, props);
-}
-
-bool SkDeferredDevice::onReadPixels(const SkImageInfo& info, void* pixels, size_t rowBytes,
- int x, int y) {
- this->flushPendingCommands(kNormal_PlaybackMode);
- return fImmediateCanvas->readPixels(info, pixels, rowBytes, x, y);
-}
-
-class AutoImmediateDrawIfNeeded {
-public:
- AutoImmediateDrawIfNeeded(SkDeferredCanvas& canvas, const SkBitmap* bitmap,
- const SkPaint* paint) {
- this->init(canvas, bitmap, NULL, paint);
- }
- AutoImmediateDrawIfNeeded(SkDeferredCanvas& canvas, const SkImage* image,
- const SkPaint* paint) {
- this->init(canvas, NULL, image, paint);
- }
-
- AutoImmediateDrawIfNeeded(SkDeferredCanvas& canvas, const SkPaint* paint) {
- this->init(canvas, NULL, NULL, paint);
- }
-
- ~AutoImmediateDrawIfNeeded() {
- if (fCanvas) {
- fCanvas->setDeferredDrawing(true);
- }
- }
-private:
- void init(SkDeferredCanvas& canvas, const SkBitmap* bitmap, const SkImage* image,
- const SkPaint* paint) {
- if (canvas.isDeferredDrawing() &&
- should_draw_immediately(bitmap, image, paint, canvas.getBitmapSizeThreshold())) {
- canvas.setDeferredDrawing(false);
- fCanvas = &canvas;
- } else {
- fCanvas = NULL;
- }
- }
-
- SkDeferredCanvas* fCanvas;
-};
-
-SkDeferredCanvas* SkDeferredCanvas::Create(SkSurface* surface) {
- if (!surface) {
- return NULL;
- }
-
- SkAutoTUnref<SkDeferredDevice> deferredDevice(SkNEW_ARGS(SkDeferredDevice, (surface)));
- return SkNEW_ARGS(SkDeferredCanvas, (deferredDevice));
-}
-
-SkDeferredCanvas::SkDeferredCanvas(SkDeferredDevice* device) : SkCanvas (device) {
- this->init();
-}
-
-void SkDeferredCanvas::init() {
- fBitmapSizeThreshold = kDeferredCanvasBitmapSizeThreshold;
- fDeferredDrawing = true; // On by default
- fCachedCanvasSize.setEmpty();
- fCachedCanvasSizeDirty = true;
- fSaveLevel = 0;
- fFirstSaveLayerIndex = kNoSaveLayerIndex;
-}
-
-void SkDeferredCanvas::setMaxRecordingStorage(size_t maxStorage) {
- this->validate();
- this->getDeferredDevice()->setMaxRecordingStorage(maxStorage);
-}
-
-size_t SkDeferredCanvas::storageAllocatedForRecording() const {
- return this->getDeferredDevice()->storageAllocatedForRecording();
-}
-
-size_t SkDeferredCanvas::freeMemoryIfPossible(size_t bytesToFree) {
- return this->getDeferredDevice()->freeMemoryIfPossible(bytesToFree);
-}
-
-void SkDeferredCanvas::setBitmapSizeThreshold(size_t sizeThreshold) {
- fBitmapSizeThreshold = sizeThreshold;
-}
-
-void SkDeferredCanvas::recordedDrawCommand() {
- if (fDeferredDrawing) {
- this->getDeferredDevice()->recordedDrawCommand();
- }
-}
-
-void SkDeferredCanvas::validate() const {
- SkASSERT(this->getDevice());
-}
-
-SkCanvas* SkDeferredCanvas::drawingCanvas() const {
- this->validate();
- return fDeferredDrawing ? this->getDeferredDevice()->recordingCanvas() :
- this->getDeferredDevice()->immediateCanvas();
-}
-
-SkCanvas* SkDeferredCanvas::immediateCanvas() const {
- this->validate();
- return this->getDeferredDevice()->immediateCanvas();
-}
-
-SkDeferredDevice* SkDeferredCanvas::getDeferredDevice() const {
- return static_cast<SkDeferredDevice*>(this->getDevice());
-}
-
-void SkDeferredCanvas::setDeferredDrawing(bool val) {
- this->validate(); // Must set device before calling this method
- if (val != fDeferredDrawing) {
- if (fDeferredDrawing) {
- // Going live.
- this->getDeferredDevice()->flushPendingCommands(kNormal_PlaybackMode);
- }
- fDeferredDrawing = val;
- }
-}
-
-bool SkDeferredCanvas::isDeferredDrawing() const {
- return fDeferredDrawing;
-}
-
-bool SkDeferredCanvas::isFreshFrame() const {
- return this->getDeferredDevice()->isFreshFrame();
-}
-
-SkISize SkDeferredCanvas::getCanvasSize() const {
- if (fCachedCanvasSizeDirty) {
- fCachedCanvasSize = this->getBaseLayerSize();
- fCachedCanvasSizeDirty = false;
- }
- return fCachedCanvasSize;
-}
-
-bool SkDeferredCanvas::hasPendingCommands() const {
- return this->getDeferredDevice()->hasPendingCommands();
-}
-
-void SkDeferredCanvas::silentFlush() {
- if (fDeferredDrawing) {
- this->getDeferredDevice()->flushPendingCommands(kSilent_PlaybackMode);
- }
-}
-
-SkDeferredCanvas::~SkDeferredCanvas() {
-}
-
-SkSurface* SkDeferredCanvas::setSurface(SkSurface* surface) {
- SkDeferredDevice* deferredDevice = this->getDeferredDevice();
- SkASSERT(deferredDevice);
- // By swapping the surface into the existing device, we preserve
- // all pending commands, which can help to seamlessly recover from
- // a lost accelerated graphics context.
- deferredDevice->setSurface(surface);
- fCachedCanvasSizeDirty = true;
- return surface;
-}
-
-SkDeferredCanvas::NotificationClient* SkDeferredCanvas::setNotificationClient(
- NotificationClient* notificationClient) {
-
- SkDeferredDevice* deferredDevice = this->getDeferredDevice();
- SkASSERT(deferredDevice);
- if (deferredDevice) {
- deferredDevice->setNotificationClient(notificationClient);
- }
- return notificationClient;
-}
-
-SkImage* SkDeferredCanvas::newImageSnapshot() {
- SkDeferredDevice* deferredDevice = this->getDeferredDevice();
- SkASSERT(deferredDevice);
- return deferredDevice ? deferredDevice->newImageSnapshot() : NULL;
-}
-
-bool SkDeferredCanvas::isFullFrame(const SkRect* rect,
- const SkPaint* paint) const {
- SkCanvas* canvas = this->drawingCanvas();
- SkISize canvasSize = this->getCanvasSize();
- if (rect) {
- if (!canvas->getTotalMatrix().rectStaysRect()) {
- return false; // conservative
- }
-
- SkRect transformedRect;
- canvas->getTotalMatrix().mapRect(&transformedRect, *rect);
-
- if (paint) {
- SkPaint::Style paintStyle = paint->getStyle();
- if (!(paintStyle == SkPaint::kFill_Style ||
- paintStyle == SkPaint::kStrokeAndFill_Style)) {
- return false;
- }
- if (paint->getMaskFilter() || paint->getLooper()
- || paint->getPathEffect() || paint->getImageFilter()) {
- return false; // conservative
- }
- }
-
- // The following test holds with AA enabled, and is conservative
- // by a 0.5 pixel margin with AA disabled
- if (transformedRect.fLeft > SkIntToScalar(0) ||
- transformedRect.fTop > SkIntToScalar(0) ||
- transformedRect.fRight < SkIntToScalar(canvasSize.fWidth) ||
- transformedRect.fBottom < SkIntToScalar(canvasSize.fHeight)) {
- return false;
- }
- }
-
- return this->getClipStack()->quickContains(SkRect::MakeXYWH(0, 0,
- SkIntToScalar(canvasSize.fWidth), SkIntToScalar(canvasSize.fHeight)));
-}
-
-void SkDeferredCanvas::willSave() {
- fSaveLevel++;
- this->drawingCanvas()->save();
- this->recordedDrawCommand();
- this->INHERITED::willSave();
-}
-
-SkCanvas::SaveLayerStrategy SkDeferredCanvas::willSaveLayer(const SkRect* bounds,
- const SkPaint* paint, SaveFlags flags) {
- fSaveLevel++;
- if (fFirstSaveLayerIndex == kNoSaveLayerIndex) {
- fFirstSaveLayerIndex = fSaveLevel;
- this->getDeferredDevice()->setIsDrawingToLayer(true);
- }
- this->drawingCanvas()->saveLayer(bounds, paint, flags);
- this->recordedDrawCommand();
- this->INHERITED::willSaveLayer(bounds, paint, flags);
- // No need for a full layer.
- return kNoLayer_SaveLayerStrategy;
-}
-
-void SkDeferredCanvas::willRestore() {
- SkASSERT(fFirstSaveLayerIndex == kNoSaveLayerIndex || fFirstSaveLayerIndex <= fSaveLevel);
- if (fFirstSaveLayerIndex == fSaveLevel) {
- fFirstSaveLayerIndex = kNoSaveLayerIndex;
- this->getDeferredDevice()->setIsDrawingToLayer(false);
- }
- fSaveLevel--;
- this->drawingCanvas()->restore();
- this->recordedDrawCommand();
- this->INHERITED::willRestore();
-}
-
-void SkDeferredCanvas::didConcat(const SkMatrix& matrix) {
- this->drawingCanvas()->concat(matrix);
- this->recordedDrawCommand();
- this->INHERITED::didConcat(matrix);
-}
-
-void SkDeferredCanvas::didSetMatrix(const SkMatrix& matrix) {
- this->drawingCanvas()->setMatrix(matrix);
- this->recordedDrawCommand();
- this->INHERITED::didSetMatrix(matrix);
-}
-
-void SkDeferredCanvas::onClipRect(const SkRect& rect,
- SkRegion::Op op,
- ClipEdgeStyle edgeStyle) {
- this->drawingCanvas()->clipRect(rect, op, kSoft_ClipEdgeStyle == edgeStyle);
- this->INHERITED::onClipRect(rect, op, edgeStyle);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onClipRRect(const SkRRect& rrect,
- SkRegion::Op op,
- ClipEdgeStyle edgeStyle) {
- this->drawingCanvas()->clipRRect(rrect, op, kSoft_ClipEdgeStyle == edgeStyle);
- this->INHERITED::onClipRRect(rrect, op, edgeStyle);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onClipPath(const SkPath& path,
- SkRegion::Op op,
- ClipEdgeStyle edgeStyle) {
- this->drawingCanvas()->clipPath(path, op, kSoft_ClipEdgeStyle == edgeStyle);
- this->INHERITED::onClipPath(path, op, edgeStyle);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onClipRegion(const SkRegion& deviceRgn, SkRegion::Op op) {
- this->drawingCanvas()->clipRegion(deviceRgn, op);
- this->INHERITED::onClipRegion(deviceRgn, op);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawPaint(const SkPaint& paint) {
- if (fDeferredDrawing && this->isFullFrame(NULL, &paint) && SkPaintPriv::Overwrites(paint)) {
- this->getDeferredDevice()->skipPendingCommands();
- }
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawPaint(paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawPoints(PointMode mode, size_t count,
- const SkPoint pts[], const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawPoints(mode, count, pts, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawOval(const SkRect& rect, const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawOval(rect, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawRect(const SkRect& rect, const SkPaint& paint) {
- if (fDeferredDrawing && this->isFullFrame(&rect, &paint) && SkPaintPriv::Overwrites(paint)) {
- this->getDeferredDevice()->skipPendingCommands();
- }
-
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawRect(rect, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawRRect(const SkRRect& rrect, const SkPaint& paint) {
- if (rrect.isRect()) {
- this->SkDeferredCanvas::drawRect(rrect.getBounds(), paint);
- } else if (rrect.isOval()) {
- this->SkDeferredCanvas::drawOval(rrect.getBounds(), paint);
- } else {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawRRect(rrect, paint);
- this->recordedDrawCommand();
- }
-}
-
-void SkDeferredCanvas::onDrawDRRect(const SkRRect& outer, const SkRRect& inner,
- const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawDRRect(outer, inner, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawPath(const SkPath& path, const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawPath(path, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawBitmap(const SkBitmap& bitmap, SkScalar left,
- SkScalar top, const SkPaint* paint) {
- SkRect bitmapRect = SkRect::MakeXYWH(left, top,
- SkIntToScalar(bitmap.width()), SkIntToScalar(bitmap.height()));
- if (fDeferredDrawing &&
- this->isFullFrame(&bitmapRect, paint) &&
- SkPaintPriv::Overwrites(bitmap, paint)) {
- this->getDeferredDevice()->skipPendingCommands();
- }
-
- AutoImmediateDrawIfNeeded autoDraw(*this, &bitmap, paint);
- this->drawingCanvas()->drawBitmap(bitmap, left, top, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawBitmapRect(const SkBitmap& bitmap, const SkRect* src,
- const SkRect& dst, const SkPaint* paint,
- SrcRectConstraint constraint) {
- if (fDeferredDrawing &&
- this->isFullFrame(&dst, paint) &&
- SkPaintPriv::Overwrites(bitmap, paint)) {
- this->getDeferredDevice()->skipPendingCommands();
- }
-
- AutoImmediateDrawIfNeeded autoDraw(*this, &bitmap, paint);
- this->drawingCanvas()->legacy_drawBitmapRect(bitmap, src, dst, paint, (SrcRectConstraint)constraint);
- this->recordedDrawCommand();
-}
-
-
-void SkDeferredCanvas::onDrawImage(const SkImage* image, SkScalar x, SkScalar y,
- const SkPaint* paint) {
- SkRect bounds = SkRect::MakeXYWH(x, y,
- SkIntToScalar(image->width()), SkIntToScalar(image->height()));
- if (fDeferredDrawing &&
- this->isFullFrame(&bounds, paint) &&
- SkPaintPriv::Overwrites(image, paint)) {
- this->getDeferredDevice()->skipPendingCommands();
- }
-
- AutoImmediateDrawIfNeeded autoDraw(*this, image, paint);
- this->drawingCanvas()->drawImage(image, x, y, paint);
- this->recordedDrawCommand();
-}
-void SkDeferredCanvas::onDrawImageRect(const SkImage* image, const SkRect* src, const SkRect& dst,
- const SkPaint* paint, SrcRectConstraint constraint) {
- if (fDeferredDrawing &&
- this->isFullFrame(&dst, paint) &&
- SkPaintPriv::Overwrites(image, paint)) {
- this->getDeferredDevice()->skipPendingCommands();
- }
-
- AutoImmediateDrawIfNeeded autoDraw(*this, image, paint);
- this->drawingCanvas()->legacy_drawImageRect(image, src, dst, paint, constraint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawImageNine(const SkImage* image, const SkIRect& center,
- const SkRect& dst, const SkPaint* paint) {
- if (fDeferredDrawing &&
- this->isFullFrame(&dst, paint) &&
- SkPaintPriv::Overwrites(image, paint)) {
- this->getDeferredDevice()->skipPendingCommands();
- }
-
- AutoImmediateDrawIfNeeded autoDraw(*this, image, paint);
- this->drawingCanvas()->drawImageNine(image, center, dst, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawBitmapNine(const SkBitmap& bitmap,
- const SkIRect& center, const SkRect& dst,
- const SkPaint* paint) {
- // TODO: reset recording canvas if paint+bitmap is opaque and clip rect
- // covers canvas entirely and dst covers canvas entirely
- AutoImmediateDrawIfNeeded autoDraw(*this, &bitmap, paint);
- this->drawingCanvas()->drawBitmapNine(bitmap, center, dst, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawSprite(const SkBitmap& bitmap, int left, int top,
- const SkPaint* paint) {
- SkRect bitmapRect = SkRect::MakeXYWH(
- SkIntToScalar(left),
- SkIntToScalar(top),
- SkIntToScalar(bitmap.width()),
- SkIntToScalar(bitmap.height()));
- if (fDeferredDrawing &&
- this->isFullFrame(&bitmapRect, paint) &&
- SkPaintPriv::Overwrites(bitmap, paint)) {
- this->getDeferredDevice()->skipPendingCommands();
- }
-
- AutoImmediateDrawIfNeeded autoDraw(*this, &bitmap, paint);
- this->drawingCanvas()->drawSprite(bitmap, left, top, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawText(const void* text, size_t byteLength, SkScalar x, SkScalar y,
- const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawText(text, byteLength, x, y, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawPosText(const void* text, size_t byteLength, const SkPoint pos[],
- const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawPosText(text, byteLength, pos, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawPosTextH(const void* text, size_t byteLength, const SkScalar xpos[],
- SkScalar constY, const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawPosTextH(text, byteLength, xpos, constY, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawTextOnPath(const void* text, size_t byteLength, const SkPath& path,
- const SkMatrix* matrix, const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawTextOnPath(text, byteLength, path, matrix, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawTextBlob(const SkTextBlob* blob, SkScalar x, SkScalar y,
- const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawTextBlob(blob, x, y, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawPicture(const SkPicture* picture, const SkMatrix* matrix,
- const SkPaint* paint) {
- this->drawingCanvas()->drawPicture(picture, matrix, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawVertices(VertexMode vmode, int vertexCount,
- const SkPoint vertices[],
- const SkPoint texs[],
- const SkColor colors[], SkXfermode* xmode,
- const uint16_t indices[], int indexCount,
- const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawVertices(vmode, vertexCount, vertices, texs, colors, xmode,
- indices, indexCount, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawPatch(const SkPoint cubics[12], const SkColor colors[4],
- const SkPoint texCoords[4], SkXfermode* xmode,
- const SkPaint& paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, &paint);
- this->drawingCanvas()->drawPatch(cubics, colors, texCoords, xmode, paint);
- this->recordedDrawCommand();
-}
-
-void SkDeferredCanvas::onDrawAtlas(const SkImage* atlas, const SkRSXform xform[],
- const SkRect tex[], const SkColor colors[], int count,
- SkXfermode::Mode mode, const SkRect* cullRect,
- const SkPaint* paint) {
- AutoImmediateDrawIfNeeded autoDraw(*this, paint);
- this->drawingCanvas()->drawAtlas(atlas, xform, tex, colors, count, mode, cullRect, paint);
- this->recordedDrawCommand();
-}
-
-SkDrawFilter* SkDeferredCanvas::setDrawFilter(SkDrawFilter* filter) {
- this->drawingCanvas()->setDrawFilter(filter);
- this->INHERITED::setDrawFilter(filter);
- this->recordedDrawCommand();
- return filter;
-}
-
-SkCanvas* SkDeferredCanvas::canvasForDrawIter() {
- return this->drawingCanvas();
-}
« no previous file with comments | « src/core/SkPictureRecord.cpp ('k') | tests/CanvasTest.cpp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698