08d1fccf6e
SkNVRefCnt is a variant of SkRefCnt that's Not Virtual, so weighs 4 bytes instead of 8 or 16. There's only benefit to doing this if the deriving class does not otherwise need a vtable, e.g. SkPicture. I've stripped out some cruft from SkPicture, rearranged fields to pack tightly, and added compile asserts for the sizes of SkPicture, SkRecord, and SkVarAlloc. BUG=skia:3144 Review URL: https://codereview.chromium.org/741793002
305 lines
12 KiB
C++
305 lines
12 KiB
C++
|
|
/*
|
|
* Copyright 2007 The Android Open Source Project
|
|
*
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*/
|
|
|
|
|
|
#ifndef SkPicture_DEFINED
|
|
#define SkPicture_DEFINED
|
|
|
|
#include "SkBitmap.h"
|
|
#include "SkDrawPictureCallback.h"
|
|
#include "SkImageDecoder.h"
|
|
#include "SkRefCnt.h"
|
|
#include "SkTDArray.h"
|
|
|
|
#if SK_SUPPORT_GPU
|
|
class GrContext;
|
|
#endif
|
|
|
|
class SkBBoxHierarchy;
|
|
class SkCanvas;
|
|
class SkData;
|
|
class SkPictureData;
|
|
class SkStream;
|
|
class SkWStream;
|
|
|
|
struct SkPictInfo;
|
|
|
|
class SkRecord;
|
|
|
|
namespace SkRecords {
|
|
class CollectLayers;
|
|
};
|
|
|
|
/** \class SkPicture
|
|
|
|
The SkPicture class records the drawing commands made to a canvas, to
|
|
be played back at a later time.
|
|
*/
|
|
class SK_API SkPicture : public SkNVRefCnt<SkPicture> {
|
|
public:
|
|
// AccelData provides a base class for device-specific acceleration
|
|
// data. It is added to the picture via EXPERIMENTAL_addAccelData.
|
|
class AccelData : public SkRefCnt {
|
|
public:
|
|
typedef uint8_t Domain;
|
|
typedef uint32_t Key;
|
|
|
|
AccelData(Key key) : fKey(key) { }
|
|
|
|
const Key& getKey() const { return fKey; }
|
|
|
|
// This entry point allows user's to get a unique domain prefix
|
|
// for their keys
|
|
static Domain GenerateDomain();
|
|
private:
|
|
Key fKey;
|
|
};
|
|
|
|
/** PRIVATE / EXPERIMENTAL -- do not call */
|
|
void EXPERIMENTAL_addAccelData(const AccelData*) const;
|
|
|
|
/** PRIVATE / EXPERIMENTAL -- do not call */
|
|
const AccelData* EXPERIMENTAL_getAccelData(AccelData::Key) const;
|
|
|
|
/**
|
|
* Function signature defining a function that sets up an SkBitmap from encoded data. On
|
|
* success, the SkBitmap should have its Config, width, height, rowBytes and pixelref set.
|
|
* If the installed pixelref has decoded the data into pixels, then the src buffer need not be
|
|
* copied. If the pixelref defers the actual decode until its lockPixels() is called, then it
|
|
* must make a copy of the src buffer.
|
|
* @param src Encoded data.
|
|
* @param length Size of the encoded data, in bytes.
|
|
* @param dst SkBitmap to install the pixel ref on.
|
|
* @param bool Whether or not a pixel ref was successfully installed.
|
|
*/
|
|
typedef bool (*InstallPixelRefProc)(const void* src, size_t length, SkBitmap* dst);
|
|
|
|
/**
|
|
* Recreate a picture that was serialized into a stream.
|
|
* @param SkStream Serialized picture data.
|
|
* @param proc Function pointer for installing pixelrefs on SkBitmaps representing the
|
|
* encoded bitmap data from the stream.
|
|
* @return A new SkPicture representing the serialized data, or NULL if the stream is
|
|
* invalid.
|
|
*/
|
|
static SkPicture* CreateFromStream(SkStream*,
|
|
InstallPixelRefProc proc = &SkImageDecoder::DecodeMemory);
|
|
|
|
/**
|
|
* Recreate a picture that was serialized into a buffer. If the creation requires bitmap
|
|
* decoding, the decoder must be set on the SkReadBuffer parameter by calling
|
|
* SkReadBuffer::setBitmapDecoder() before calling SkPicture::CreateFromBuffer().
|
|
* @param SkReadBuffer Serialized picture data.
|
|
* @return A new SkPicture representing the serialized data, or NULL if the buffer is
|
|
* invalid.
|
|
*/
|
|
static SkPicture* CreateFromBuffer(SkReadBuffer&);
|
|
|
|
~SkPicture();
|
|
|
|
/** Replays the drawing commands on the specified canvas. Note that
|
|
this has the effect of unfurling this picture into the destination
|
|
canvas. Using the SkCanvas::drawPicture entry point gives the destination
|
|
canvas the option of just taking a ref.
|
|
@param canvas the canvas receiving the drawing commands.
|
|
@param callback a callback that allows interruption of playback
|
|
*/
|
|
void playback(SkCanvas* canvas, SkDrawPictureCallback* = NULL) const;
|
|
|
|
#ifdef SK_LEGACY_PICTURE_DRAW_API
|
|
void draw(SkCanvas* canvas, SkDrawPictureCallback* callback = NULL) const {
|
|
this->playback(canvas, callback);
|
|
}
|
|
#endif
|
|
|
|
#ifdef SK_LEGACY_PICTURE_SIZE_API
|
|
int width() const { return fCullRect.roundOut().width(); }
|
|
int height() const { return fCullRect.roundOut().height(); }
|
|
#endif
|
|
|
|
/** Return the cull rect used when creating this picture: { 0, 0, cullWidth, cullHeight }.
|
|
It does not necessarily reflect the bounds of what has been recorded into the picture.
|
|
@return the cull rect used to create this picture
|
|
*/
|
|
SkRect cullRect() const { return fCullRect; }
|
|
|
|
/** Return a non-zero, unique value representing the picture. This call is
|
|
only valid when not recording. Between a beginRecording/endRecording
|
|
pair it will just return 0 (the invalid ID). Each beginRecording/
|
|
endRecording pair will cause a different generation ID to be returned.
|
|
*/
|
|
uint32_t uniqueID() const;
|
|
|
|
/**
|
|
* Function to encode an SkBitmap to an SkData. A function with this
|
|
* signature can be passed to serialize() and SkWriteBuffer.
|
|
* Returning NULL will tell the SkWriteBuffer to use
|
|
* SkBitmap::flatten() to store the bitmap.
|
|
*
|
|
* @param pixelRefOffset DEPRECATED -- caller assumes it will return 0.
|
|
* @return SkData If non-NULL, holds encoded data representing the passed
|
|
* in bitmap. The caller is responsible for calling unref().
|
|
*/
|
|
typedef SkData* (*EncodeBitmap)(size_t* pixelRefOffset, const SkBitmap& bm);
|
|
|
|
/**
|
|
* Serialize to a stream. If non NULL, encoder will be used to encode
|
|
* any bitmaps in the picture.
|
|
* encoder will never be called with a NULL pixelRefOffset.
|
|
*/
|
|
void serialize(SkWStream*, EncodeBitmap encoder = NULL) const;
|
|
|
|
/**
|
|
* Serialize to a buffer.
|
|
*/
|
|
void flatten(SkWriteBuffer&) const;
|
|
|
|
/**
|
|
* Returns true if any bitmaps may be produced when this SkPicture
|
|
* is replayed.
|
|
*/
|
|
bool willPlayBackBitmaps() const;
|
|
|
|
/** Return true if the SkStream/Buffer represents a serialized picture, and
|
|
fills out SkPictInfo. After this function returns, the data source is not
|
|
rewound so it will have to be manually reset before passing to
|
|
CreateFromStream or CreateFromBuffer. Note, CreateFromStream and
|
|
CreateFromBuffer perform this check internally so these entry points are
|
|
intended for stand alone tools.
|
|
If false is returned, SkPictInfo is unmodified.
|
|
*/
|
|
static bool InternalOnly_StreamIsSKP(SkStream*, SkPictInfo*);
|
|
static bool InternalOnly_BufferIsSKP(SkReadBuffer*, SkPictInfo*);
|
|
|
|
/** Return true if the picture is suitable for rendering on the GPU.
|
|
*/
|
|
|
|
#if SK_SUPPORT_GPU
|
|
bool suitableForGpuRasterization(GrContext*, const char ** = NULL) const;
|
|
#endif
|
|
|
|
class DeletionListener : public SkRefCnt {
|
|
public:
|
|
virtual void onDeletion(uint32_t pictureID) = 0;
|
|
};
|
|
|
|
// Takes ref on listener.
|
|
void addDeletionListener(DeletionListener* listener) const;
|
|
|
|
/** Return the approximate number of operations in this picture. This
|
|
* number may be greater or less than the number of SkCanvas calls
|
|
* recorded: some calls may be recorded as more than one operation, or some
|
|
* calls may be optimized away.
|
|
*/
|
|
int approximateOpCount() const;
|
|
|
|
/** Return true if this picture contains text.
|
|
*/
|
|
bool hasText() const;
|
|
|
|
private:
|
|
// V2 : adds SkPixelRef's generation ID.
|
|
// V3 : PictInfo tag at beginning, and EOF tag at the end
|
|
// V4 : move SkPictInfo to be the header
|
|
// V5 : don't read/write FunctionPtr on cross-process (we can detect that)
|
|
// V6 : added serialization of SkPath's bounds (and packed its flags tighter)
|
|
// V7 : changed drawBitmapRect(IRect) to drawBitmapRectToRect(Rect)
|
|
// V8 : Add an option for encoding bitmaps
|
|
// V9 : Allow the reader and writer of an SKP disagree on whether to support
|
|
// SK_SUPPORT_HINTING_SCALE_FACTOR
|
|
// V10: add drawRRect, drawOval, clipRRect
|
|
// V11: modify how readBitmap and writeBitmap store their info.
|
|
// V12: add conics to SkPath, use new SkPathRef flattening
|
|
// V13: add flag to drawBitmapRectToRect
|
|
// parameterize blurs by sigma rather than radius
|
|
// V14: Add flags word to PathRef serialization
|
|
// V15: Remove A1 bitmap config (and renumber remaining configs)
|
|
// V16: Move SkPath's isOval flag to SkPathRef
|
|
// V17: SkPixelRef now writes SkImageInfo
|
|
// V18: SkBitmap now records x,y for its pixelref origin, instead of offset.
|
|
// V19: encode matrices and regions into the ops stream
|
|
// V20: added bool to SkPictureImageFilter's serialization (to allow SkPicture serialization)
|
|
// V21: add pushCull, popCull
|
|
// V22: SK_PICT_FACTORY_TAG's size is now the chunk size in bytes
|
|
// V23: SkPaint::FilterLevel became a real enum
|
|
// V24: SkTwoPointConicalGradient now has fFlipped flag for gradient flipping
|
|
// V25: SkDashPathEffect now only writes phase and interval array when flattening
|
|
// V26: Removed boolean from SkColorShader for inheriting color from SkPaint.
|
|
// V27: Remove SkUnitMapper from gradients (and skia).
|
|
// V28: No longer call bitmap::flatten inside SkWriteBuffer::writeBitmap.
|
|
// V29: Removed SaveFlags parameter from save().
|
|
// V30: Remove redundant SkMatrix from SkLocalMatrixShader.
|
|
// V31: Add a serialized UniqueID to SkImageFilter.
|
|
// V32: Removed SkPaintOptionsAndroid from SkPaint
|
|
// V33: Serialize only public API of effects.
|
|
// V34: Add SkTextBlob serialization.
|
|
// V35: Store SkRect (rather then width & height) in header
|
|
// V36: Remove (obsolete) alphatype from SkColorTable
|
|
// V37: Added shadow only option to SkDropShadowImageFilter
|
|
|
|
// Note: If the picture version needs to be increased then please follow the
|
|
// steps to generate new SKPs in (only accessible to Googlers): http://goo.gl/qATVcw
|
|
|
|
// Only SKPs within the min/current picture version range (inclusive) can be read.
|
|
static const uint32_t MIN_PICTURE_VERSION = 19;
|
|
static const uint32_t CURRENT_PICTURE_VERSION = 37;
|
|
|
|
void needsNewGenID() { fUniqueID = SK_InvalidGenID; }
|
|
void callDeletionListeners();
|
|
|
|
void createHeader(SkPictInfo* info) const;
|
|
static bool IsValidPictInfo(const SkPictInfo& info);
|
|
|
|
// Takes ownership of the SkRecord, refs the (optional) drawablePicts and BBH.
|
|
SkPicture(const SkRect& cullRect, SkRecord*, SkData* drawablePicts,
|
|
SkBBoxHierarchy*);
|
|
|
|
static SkPicture* Forwardport(const SkPictInfo&, const SkPictureData*);
|
|
static SkPictureData* Backport(const SkRecord&, const SkPictInfo&,
|
|
SkPicture const* const drawablePics[], int drawableCount);
|
|
|
|
// uint32_t fRefCnt; from SkNVRefCnt<SkPicture>
|
|
mutable uint32_t fUniqueID;
|
|
const SkRect fCullRect;
|
|
mutable SkAutoTUnref<const AccelData> fAccelData;
|
|
mutable SkTDArray<DeletionListener*> fDeletionListeners; // pointers are refed
|
|
SkAutoTDelete<SkRecord> fRecord;
|
|
SkAutoTUnref<SkBBoxHierarchy> fBBH;
|
|
SkAutoTUnref<SkData> fDrawablePicts;
|
|
|
|
// helpers for fDrawablePicts
|
|
int drawableCount() const;
|
|
// will return NULL if drawableCount() returns 0
|
|
SkPicture const* const* drawablePicts() const;
|
|
|
|
struct PathCounter;
|
|
|
|
struct Analysis {
|
|
Analysis() {} // Only used by SkPictureData codepath.
|
|
explicit Analysis(const SkRecord&);
|
|
|
|
bool suitableForGpuRasterization(const char** reason, int sampleCount) const;
|
|
|
|
bool fWillPlaybackBitmaps;
|
|
bool fHasText;
|
|
int fNumPaintWithPathEffectUses;
|
|
int fNumFastPathDashEffects;
|
|
int fNumAAConcavePaths;
|
|
int fNumAAHairlineConcavePaths;
|
|
int fNumAADFEligibleConcavePaths;
|
|
} fAnalysis;
|
|
|
|
friend class SkPictureRecorder; // SkRecord-based constructor.
|
|
friend class GrLayerHoister; // access to fRecord
|
|
friend class ReplaceDraw;
|
|
friend class SkPictureUtils;
|
|
};
|
|
SK_COMPILE_ASSERT(sizeof(SkPicture) <= 96, SkPictureSize);
|
|
|
|
#endif
|