Revert of Change device creation to see the (optional) layer-paint (patchset #9 id:160001 of https://codereview.chromium.org/988413003/)
Reason for revert: need to have chrome opt-in for the older API before this can land (in chrome) Original issue's description: > Change device creation to see the (optional) layer-paint > > Motivation: > > PDFDevice currently relies on 1) being told that the layer's paint has an imagefilter, and in the case, it creates a rasterdevice. It then relies on (2) canvas itself sniffing the layer's paint and offering to apply-the-imagefilter to call drawSprite instead of drawDevice. > > This subtle interchange is fragile, and also does not support other unsupported PDF features like colorfilters. This CL is a step toward making this use-raster-instead-of-native approach to layers more completely in the subclass' hands. > > Committed: https://skia.googlesource.com/skia/+/1182d9a96b80bd12183ee7c81325a979a51ee0c0 TBR=halcanary@google.com,senorblanco@google.com,robertphillips@google.com NOPRESUBMIT=true NOTREECHECKS=true NOTRY=true Review URL: https://codereview.chromium.org/1008863002
This commit is contained in:
parent
1182d9a96b
commit
0e040f7da2
@ -98,7 +98,11 @@ protected:
|
||||
const SkColor colors[], SkXfermode* xmode,
|
||||
const uint16_t indices[], int indexCount,
|
||||
const SkPaint& paint) SK_OVERRIDE;
|
||||
virtual void drawDevice(const SkDraw&, SkBaseDevice*, int x, int y, const SkPaint&) SK_OVERRIDE;
|
||||
/** The SkBaseDevice passed will be an SkBaseDevice which was returned by a call to
|
||||
onCreateCompatibleDevice on this device with kSaveLayer_Usage.
|
||||
*/
|
||||
virtual void drawDevice(const SkDraw&, SkBaseDevice*, int x, int y,
|
||||
const SkPaint&) SK_OVERRIDE;
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
|
||||
@ -141,7 +145,7 @@ private:
|
||||
// any clip information.
|
||||
void replaceBitmapBackendForRasterSurface(const SkBitmap&) SK_OVERRIDE;
|
||||
|
||||
SkBaseDevice* onCreateDevice(const CreateInfo&, const SkPaint*) SK_OVERRIDE;
|
||||
SkBaseDevice* onCreateCompatibleDevice(const CreateInfo&) SK_OVERRIDE;
|
||||
|
||||
SkSurface* newSurface(const SkImageInfo&, const SkSurfaceProps&) SK_OVERRIDE;
|
||||
const void* peekPixels(SkImageInfo*, size_t* rowBytes) SK_OVERRIDE;
|
||||
|
@ -1321,7 +1321,8 @@ private:
|
||||
void internalDrawBitmapNine(const SkBitmap& bitmap, const SkIRect& center,
|
||||
const SkRect& dst, const SkPaint* paint);
|
||||
void internalDrawPaint(const SkPaint& paint);
|
||||
void internalSaveLayer(const SkRect* bounds, const SkPaint*, SaveFlags, SaveLayerStrategy);
|
||||
void internalSaveLayer(const SkRect* bounds, const SkPaint* paint,
|
||||
SaveFlags, bool justForImageFilter, SaveLayerStrategy strategy);
|
||||
void internalDrawDevice(SkBaseDevice*, int x, int y, const SkPaint*);
|
||||
|
||||
// shared by save() and saveLayer()
|
||||
|
@ -125,9 +125,10 @@ public:
|
||||
};
|
||||
|
||||
protected:
|
||||
enum TileUsage {
|
||||
kPossible_TileUsage, //!< the created device may be drawn tiled
|
||||
kNever_TileUsage, //!< the created device will never be drawn tiled
|
||||
enum Usage {
|
||||
kGeneral_Usage,
|
||||
kSaveLayer_Usage, // <! internal use only
|
||||
kImageFilter_Usage // <! internal use only
|
||||
};
|
||||
|
||||
struct TextFlags {
|
||||
@ -230,7 +231,7 @@ protected:
|
||||
virtual void drawPatch(const SkDraw&, const SkPoint cubics[12], const SkColor colors[4],
|
||||
const SkPoint texCoords[4], SkXfermode* xmode, const SkPaint& paint);
|
||||
/** The SkDevice passed will be an SkDevice which was returned by a call to
|
||||
onCreateDevice on this device with kNeverTile_TileExpectation.
|
||||
onCreateCompatibleDevice on this device with kSaveLayer_Usage.
|
||||
*/
|
||||
virtual void drawDevice(const SkDraw&, SkBaseDevice*, int x, int y,
|
||||
const SkPaint&) = 0;
|
||||
@ -253,6 +254,14 @@ protected:
|
||||
virtual void lockPixels() {}
|
||||
virtual void unlockPixels() {}
|
||||
|
||||
/**
|
||||
* Returns true if the device allows processing of this imagefilter. If
|
||||
* false is returned, then the filter is ignored. This may happen for
|
||||
* some subclasses that do not support pixel manipulations after drawing
|
||||
* has occurred (e.g. printing). The default implementation returns true.
|
||||
*/
|
||||
virtual bool allowImageFilter(const SkImageFilter*) { return true; }
|
||||
|
||||
/**
|
||||
* Override and return true for filters that the device can handle
|
||||
* intrinsically. Doing so means that SkCanvas will pass-through this
|
||||
@ -327,39 +336,24 @@ protected:
|
||||
const SkPaint*);
|
||||
|
||||
struct CreateInfo {
|
||||
static SkPixelGeometry AdjustGeometry(const SkImageInfo&, TileUsage, SkPixelGeometry);
|
||||
static SkPixelGeometry AdjustGeometry(const SkImageInfo&, Usage, SkPixelGeometry geo);
|
||||
|
||||
// The constructor may change the pixel geometry based on other parameters.
|
||||
CreateInfo(const SkImageInfo& info, TileUsage tileUsage, SkPixelGeometry geo)
|
||||
// The construct may change the pixel geometry based on usage as needed.
|
||||
CreateInfo(const SkImageInfo& info, Usage usage, SkPixelGeometry geo)
|
||||
: fInfo(info)
|
||||
, fTileUsage(tileUsage)
|
||||
, fPixelGeometry(AdjustGeometry(info, tileUsage, geo))
|
||||
, fUsage(usage)
|
||||
, fPixelGeometry(AdjustGeometry(info, usage, geo))
|
||||
{}
|
||||
|
||||
const SkImageInfo fInfo;
|
||||
const TileUsage fTileUsage;
|
||||
const SkPixelGeometry fPixelGeometry;
|
||||
const SkImageInfo fInfo;
|
||||
const Usage fUsage;
|
||||
const SkPixelGeometry fPixelGeometry;
|
||||
};
|
||||
|
||||
#ifdef SK_SUPPORT_LEGACY_ONCREATECOMPATIBLEDEVICE
|
||||
// legacy method name -- please override onCreateDevice instead
|
||||
virtual SkBaseDevice* onCreateCompatibleDevice(const CreateInfo&) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
virtual SkBaseDevice* onCreateDevice(const CreateInfo& cinfo, const SkPaint* layerPaint) {
|
||||
return this->onCreateCompatibleDevice(cinfo);
|
||||
}
|
||||
#else
|
||||
/**
|
||||
* Create a new device based on CreateInfo. If the paint is not null, then it represents a
|
||||
* preview of how the new device will be composed with its creator device (this).
|
||||
*/
|
||||
virtual SkBaseDevice* onCreateDevice(const CreateInfo&, const SkPaint*) {
|
||||
return NULL;
|
||||
}
|
||||
#endif
|
||||
|
||||
virtual void initForRootLayer(SkPixelGeometry geo);
|
||||
|
||||
private:
|
||||
|
@ -135,6 +135,8 @@ protected:
|
||||
int x, int y,
|
||||
const SkPaint& paint) SK_OVERRIDE;
|
||||
|
||||
bool allowImageFilter(const SkImageFilter*) SK_OVERRIDE;
|
||||
|
||||
private:
|
||||
class TypefaceUse : ::SkNoncopyable {
|
||||
public:
|
||||
@ -309,7 +311,7 @@ private:
|
||||
const SkVector& ppuScale,
|
||||
IXpsOMPath* shadedPath);
|
||||
|
||||
SkBaseDevice* onCreateDevice(const CreateInfo&, const SkPaint*) SK_OVERRIDE;
|
||||
SkBaseDevice* onCreateCompatibleDevice(const CreateInfo&) SK_OVERRIDE;
|
||||
|
||||
// Disable the default copy and assign implementation.
|
||||
SkXPSDevice(const SkXPSDevice&);
|
||||
|
@ -110,7 +110,7 @@ void SkBitmapDevice::replaceBitmapBackendForRasterSurface(const SkBitmap& bm) {
|
||||
fBitmap.lockPixels();
|
||||
}
|
||||
|
||||
SkBaseDevice* SkBitmapDevice::onCreateDevice(const CreateInfo& cinfo, const SkPaint*) {
|
||||
SkBaseDevice* SkBitmapDevice::onCreateCompatibleDevice(const CreateInfo& cinfo) {
|
||||
SkDeviceProperties leaky(cinfo.fPixelGeometry);
|
||||
return SkBitmapDevice::Create(cinfo.fInfo, &leaky);
|
||||
}
|
||||
|
@ -299,7 +299,7 @@ public:
|
||||
SkPaint tmp;
|
||||
tmp.setImageFilter(fOrigPaint.getImageFilter());
|
||||
(void)canvas->internalSaveLayer(bounds, &tmp, SkCanvas::kARGB_ClipLayer_SaveFlag,
|
||||
SkCanvas::kFullLayer_SaveLayerStrategy);
|
||||
true, SkCanvas::kFullLayer_SaveLayerStrategy);
|
||||
// we'll clear the imageFilter for the actual draws in next(), so
|
||||
// it will only be applied during the restore().
|
||||
fDoClearImageFilter = true;
|
||||
@ -880,7 +880,7 @@ int SkCanvas::saveLayer(const SkRect* bounds, const SkPaint* paint) {
|
||||
}
|
||||
SaveLayerStrategy strategy = this->willSaveLayer(bounds, paint, kARGB_ClipLayer_SaveFlag);
|
||||
fSaveCount += 1;
|
||||
this->internalSaveLayer(bounds, paint, kARGB_ClipLayer_SaveFlag, strategy);
|
||||
this->internalSaveLayer(bounds, paint, kARGB_ClipLayer_SaveFlag, false, strategy);
|
||||
return this->getSaveCount() - 1;
|
||||
}
|
||||
|
||||
@ -890,12 +890,12 @@ int SkCanvas::saveLayer(const SkRect* bounds, const SkPaint* paint, SaveFlags fl
|
||||
}
|
||||
SaveLayerStrategy strategy = this->willSaveLayer(bounds, paint, flags);
|
||||
fSaveCount += 1;
|
||||
this->internalSaveLayer(bounds, paint, flags, strategy);
|
||||
this->internalSaveLayer(bounds, paint, flags, false, strategy);
|
||||
return this->getSaveCount() - 1;
|
||||
}
|
||||
|
||||
void SkCanvas::internalSaveLayer(const SkRect* bounds, const SkPaint* paint, SaveFlags flags,
|
||||
SaveLayerStrategy strategy) {
|
||||
bool justForImageFilter, SaveLayerStrategy strategy) {
|
||||
#ifndef SK_SUPPORT_LEGACY_CLIPTOLAYERFLAG
|
||||
flags |= kClipToLayer_SaveFlag;
|
||||
#endif
|
||||
@ -917,13 +917,21 @@ void SkCanvas::internalSaveLayer(const SkRect* bounds, const SkPaint* paint, Sav
|
||||
return;
|
||||
}
|
||||
|
||||
bool isOpaque = !SkToBool(flags & kHasAlphaLayer_SaveFlag);
|
||||
if (isOpaque && paint) {
|
||||
// TODO: perhaps add a query to filters so we might preserve opaqueness...
|
||||
if (paint->getImageFilter() || paint->getColorFilter()) {
|
||||
isOpaque = false;
|
||||
// Kill the imagefilter if our device doesn't allow it
|
||||
SkLazyPaint lazyP;
|
||||
if (paint && paint->getImageFilter()) {
|
||||
if (!this->getTopDevice()->allowImageFilter(paint->getImageFilter())) {
|
||||
if (justForImageFilter) {
|
||||
// early exit if the layer was just for the imageFilter
|
||||
return;
|
||||
}
|
||||
SkPaint* p = lazyP.set(*paint);
|
||||
p->setImageFilter(NULL);
|
||||
paint = p;
|
||||
}
|
||||
}
|
||||
|
||||
bool isOpaque = !SkToBool(flags & kHasAlphaLayer_SaveFlag);
|
||||
SkImageInfo info = SkImageInfo::MakeN32(ir.width(), ir.height(),
|
||||
isOpaque ? kOpaque_SkAlphaType : kPremul_SkAlphaType);
|
||||
|
||||
@ -933,17 +941,12 @@ void SkCanvas::internalSaveLayer(const SkRect* bounds, const SkPaint* paint, Sav
|
||||
return;
|
||||
}
|
||||
|
||||
SkBaseDevice::TileUsage usage = SkBaseDevice::kNever_TileUsage;
|
||||
#if 1
|
||||
// this seems needed for current GMs, but makes us draw slower on the GPU
|
||||
// Related to https://code.google.com/p/skia/issues/detail?id=3519 ?
|
||||
//
|
||||
SkBaseDevice::Usage usage = SkBaseDevice::kSaveLayer_Usage;
|
||||
if (paint && paint->getImageFilter()) {
|
||||
usage = SkBaseDevice::kPossible_TileUsage;
|
||||
usage = SkBaseDevice::kImageFilter_Usage;
|
||||
}
|
||||
#endif
|
||||
device = device->onCreateDevice(SkBaseDevice::CreateInfo(info, usage, fProps.pixelGeometry()),
|
||||
paint);
|
||||
device = device->onCreateCompatibleDevice(SkBaseDevice::CreateInfo(info, usage,
|
||||
fProps.pixelGeometry()));
|
||||
if (NULL == device) {
|
||||
SkErrorInternals::SetError( kInternalError_SkError,
|
||||
"Unable to create device for layer.");
|
||||
|
@ -64,16 +64,19 @@ const SkBitmap& SkBaseDevice::accessBitmap(bool changePixels) {
|
||||
}
|
||||
|
||||
SkPixelGeometry SkBaseDevice::CreateInfo::AdjustGeometry(const SkImageInfo& info,
|
||||
TileUsage tileUsage,
|
||||
Usage usage,
|
||||
SkPixelGeometry geo) {
|
||||
switch (tileUsage) {
|
||||
case kPossible_TileUsage:
|
||||
switch (usage) {
|
||||
case kGeneral_Usage:
|
||||
break;
|
||||
case kNever_TileUsage:
|
||||
case kSaveLayer_Usage:
|
||||
if (info.alphaType() != kOpaque_SkAlphaType) {
|
||||
geo = kUnknown_SkPixelGeometry;
|
||||
}
|
||||
break;
|
||||
case kImageFilter_Usage:
|
||||
geo = kUnknown_SkPixelGeometry;
|
||||
break;
|
||||
}
|
||||
return geo;
|
||||
}
|
||||
@ -83,7 +86,7 @@ void SkBaseDevice::initForRootLayer(SkPixelGeometry geo) {
|
||||
// anyway to document logically what is going on.
|
||||
//
|
||||
fLeakyProperties->setPixelGeometry(CreateInfo::AdjustGeometry(this->imageInfo(),
|
||||
kPossible_TileUsage,
|
||||
kGeneral_Usage,
|
||||
geo));
|
||||
}
|
||||
|
||||
|
@ -18,15 +18,15 @@ public:
|
||||
: fDevice(device)
|
||||
, fProps(props.flags(),
|
||||
SkBaseDevice::CreateInfo::AdjustGeometry(SkImageInfo(),
|
||||
SkBaseDevice::kPossible_TileUsage,
|
||||
SkBaseDevice::kImageFilter_Usage,
|
||||
props.pixelGeometry()))
|
||||
{}
|
||||
|
||||
SkBaseDevice* createDevice(int w, int h) SK_OVERRIDE {
|
||||
SkBaseDevice::CreateInfo cinfo(SkImageInfo::MakeN32Premul(w, h),
|
||||
SkBaseDevice::kPossible_TileUsage,
|
||||
SkBaseDevice::kImageFilter_Usage,
|
||||
kUnknown_SkPixelGeometry);
|
||||
return fDevice->onCreateDevice(cinfo, NULL);
|
||||
return fDevice->onCreateCompatibleDevice(cinfo);
|
||||
}
|
||||
bool canHandleImageFilter(const SkImageFilter* filter) SK_OVERRIDE {
|
||||
return fDevice->canHandleImageFilter(filter);
|
||||
|
@ -2251,7 +2251,7 @@ void SkXPSDevice::drawDevice(const SkDraw& d, SkBaseDevice* dev,
|
||||
"Could not add layer to current visuals.");
|
||||
}
|
||||
|
||||
SkBaseDevice* SkXPSDevice::onCreateDevice(const CreateInfo& info, const SkPaint*) {
|
||||
SkBaseDevice* SkXPSDevice::onCreateCompatibleDevice(const CreateInfo& info) {
|
||||
//Conditional for bug compatibility with PDF device.
|
||||
#if 0
|
||||
if (SkBaseDevice::kGeneral_Usage == info.fUsage) {
|
||||
@ -2282,3 +2282,6 @@ SkXPSDevice::SkXPSDevice(IXpsOMObjectFactory* xpsFactory)
|
||||
"Could not create canvas for layer.");
|
||||
}
|
||||
|
||||
bool SkXPSDevice::allowImageFilter(const SkImageFilter*) {
|
||||
return false;
|
||||
}
|
||||
|
@ -1862,7 +1862,7 @@ void SkGpuDevice::flush() {
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
SkBaseDevice* SkGpuDevice::onCreateDevice(const CreateInfo& cinfo, const SkPaint*) {
|
||||
SkBaseDevice* SkGpuDevice::onCreateCompatibleDevice(const CreateInfo& cinfo) {
|
||||
GrSurfaceDesc desc;
|
||||
desc.fConfig = fRenderTarget->config();
|
||||
desc.fFlags = kRenderTarget_GrSurfaceFlag;
|
||||
@ -1876,7 +1876,7 @@ SkBaseDevice* SkGpuDevice::onCreateDevice(const CreateInfo& cinfo, const SkPaint
|
||||
|
||||
// layers are never draw in repeat modes, so we can request an approx
|
||||
// match and ignore any padding.
|
||||
const GrContext::ScratchTexMatch match = (kNever_TileUsage == cinfo.fTileUsage) ?
|
||||
const GrContext::ScratchTexMatch match = (kSaveLayer_Usage == cinfo.fUsage) ?
|
||||
GrContext::kApprox_ScratchTexMatch :
|
||||
GrContext::kExact_ScratchTexMatch;
|
||||
texture.reset(fContext->refScratchTexture(desc, match));
|
||||
|
@ -51,9 +51,9 @@ public:
|
||||
virtual ~SkGpuDevice();
|
||||
|
||||
SkGpuDevice* cloneDevice(const SkSurfaceProps& props) {
|
||||
SkBaseDevice* dev = this->onCreateDevice(CreateInfo(this->imageInfo(), kPossible_TileUsage,
|
||||
props.pixelGeometry()),
|
||||
NULL);
|
||||
SkBaseDevice* dev = this->onCreateCompatibleDevice(CreateInfo(this->imageInfo(),
|
||||
kGeneral_Usage,
|
||||
props.pixelGeometry()));
|
||||
return static_cast<SkGpuDevice*>(dev);
|
||||
}
|
||||
|
||||
@ -147,7 +147,7 @@ private:
|
||||
|
||||
SkGpuDevice(GrRenderTarget*, const SkSurfaceProps*, unsigned flags);
|
||||
|
||||
SkBaseDevice* onCreateDevice(const CreateInfo&, const SkPaint*) SK_OVERRIDE;
|
||||
SkBaseDevice* onCreateCompatibleDevice(const CreateInfo&) SK_OVERRIDE;
|
||||
|
||||
SkSurface* newSurface(const SkImageInfo&, const SkSurfaceProps&) SK_OVERRIDE;
|
||||
|
||||
|
@ -566,19 +566,13 @@ void GraphicStackState::updateDrawingState(const GraphicStateEntry& state) {
|
||||
}
|
||||
}
|
||||
|
||||
static bool not_supported_for_layers(const SkPaint& layerPaint) {
|
||||
SkBaseDevice* SkPDFDevice::onCreateCompatibleDevice(const CreateInfo& cinfo) {
|
||||
// PDF does not support image filters, so render them on CPU.
|
||||
// Note that this rendering is done at "screen" resolution (100dpi), not
|
||||
// printer resolution.
|
||||
// FIXME: It may be possible to express some filters natively using PDF
|
||||
// to improve quality and file size (http://skbug.com/3043)
|
||||
|
||||
// TODO: should we return true if there is a colorfilter?
|
||||
return layerPaint.getImageFilter() != NULL;
|
||||
}
|
||||
|
||||
SkBaseDevice* SkPDFDevice::onCreateDevice(const CreateInfo& cinfo, const SkPaint* layerPaint) {
|
||||
if (layerPaint && not_supported_for_layers(*layerPaint)) {
|
||||
if (kImageFilter_Usage == cinfo.fUsage) {
|
||||
return SkBitmapDevice::Create(cinfo.fInfo);
|
||||
}
|
||||
SkISize size = SkISize::Make(cinfo.fInfo.width(), cinfo.fInfo.height());
|
||||
|
@ -235,7 +235,8 @@ private:
|
||||
ContentEntry* getLastContentEntry();
|
||||
void setLastContentEntry(ContentEntry* contentEntry);
|
||||
|
||||
SkBaseDevice* onCreateDevice(const CreateInfo&, const SkPaint*) SK_OVERRIDE;
|
||||
// override from SkBaseDevice
|
||||
SkBaseDevice* onCreateCompatibleDevice(const CreateInfo&) SK_OVERRIDE;
|
||||
|
||||
void init();
|
||||
void cleanUp(bool clearFontUsage);
|
||||
|
@ -162,7 +162,7 @@ public:
|
||||
|
||||
GrRenderTarget* accessRenderTarget() SK_OVERRIDE;
|
||||
|
||||
SkBaseDevice* onCreateDevice(const CreateInfo&, const SkPaint*) SK_OVERRIDE;
|
||||
SkBaseDevice* onCreateCompatibleDevice(const CreateInfo&) SK_OVERRIDE;
|
||||
|
||||
SkSurface* newSurface(const SkImageInfo&, const SkSurfaceProps&) SK_OVERRIDE;
|
||||
|
||||
@ -231,6 +231,9 @@ protected:
|
||||
void lockPixels() SK_OVERRIDE {}
|
||||
void unlockPixels() SK_OVERRIDE {}
|
||||
|
||||
bool allowImageFilter(const SkImageFilter*) SK_OVERRIDE {
|
||||
return false;
|
||||
}
|
||||
bool canHandleImageFilter(const SkImageFilter*) SK_OVERRIDE {
|
||||
return false;
|
||||
}
|
||||
@ -458,13 +461,16 @@ const SkBitmap& SkDeferredDevice::onAccessBitmap() {
|
||||
return immediateDevice()->accessBitmap(false);
|
||||
}
|
||||
|
||||
SkBaseDevice* SkDeferredDevice::onCreateDevice(const CreateInfo& cinfo, const SkPaint* layerPaint) {
|
||||
SkBaseDevice* SkDeferredDevice::onCreateCompatibleDevice(const CreateInfo& cinfo) {
|
||||
// Save layer usage not supported, and not required by SkDeferredCanvas.
|
||||
SkASSERT(cinfo.fUsage != kSaveLayer_Usage);
|
||||
|
||||
// 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 immediateDevice()->onCreateDevice(cinfo, layerPaint);
|
||||
return immediateDevice()->onCreateCompatibleDevice(cinfo);
|
||||
}
|
||||
|
||||
SkSurface* SkDeferredDevice::newSurface(const SkImageInfo& info, const SkSurfaceProps& props) {
|
||||
|
Loading…
Reference in New Issue
Block a user