2012-06-29 14:21:22 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright 2012 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2013-02-06 20:13:54 +00:00
|
|
|
#include "SkColorPriv.h"
|
2012-06-29 14:21:22 +00:00
|
|
|
#include "SkDebugCanvas.h"
|
|
|
|
#include "SkDrawCommand.h"
|
2013-02-06 20:13:54 +00:00
|
|
|
#include "SkDrawFilter.h"
|
2012-11-21 17:11:02 +00:00
|
|
|
#include "SkDevice.h"
|
2013-02-06 20:13:54 +00:00
|
|
|
#include "SkXfermode.h"
|
2012-06-29 14:21:22 +00:00
|
|
|
|
2014-08-29 15:03:56 +00:00
|
|
|
SkDebugCanvas::SkDebugCanvas(int windowWidth, int windowHeight)
|
|
|
|
: INHERITED(windowWidth, windowHeight)
|
2014-03-25 23:31:33 +00:00
|
|
|
, fPicture(NULL)
|
2014-08-29 15:03:56 +00:00
|
|
|
, fWindowSize(SkISize::Make(windowWidth, windowHeight))
|
2013-12-04 13:42:46 +00:00
|
|
|
, fFilter(false)
|
2014-03-03 16:32:17 +00:00
|
|
|
, fMegaVizMode(false)
|
2013-12-04 13:42:46 +00:00
|
|
|
, fIndex(0)
|
2013-02-06 20:13:54 +00:00
|
|
|
, fOverdrawViz(false)
|
2013-02-08 21:16:19 +00:00
|
|
|
, fOverdrawFilter(NULL)
|
2013-10-17 17:56:10 +00:00
|
|
|
, fOverrideTexFiltering(false)
|
|
|
|
, fTexOverrideFilter(NULL)
|
2013-02-08 21:16:19 +00:00
|
|
|
, fOutstandingSaveCount(0) {
|
2013-01-17 16:30:56 +00:00
|
|
|
fUserMatrix.reset();
|
2014-10-16 21:28:28 +00:00
|
|
|
fDrawNeedsReset = false;
|
2013-11-07 22:20:31 +00:00
|
|
|
|
|
|
|
// SkPicturePlayback uses the base-class' quickReject calls to cull clipped
|
|
|
|
// operations. This can lead to problems in the debugger which expects all
|
|
|
|
// the operations in the captured skp to appear in the debug canvas. To
|
|
|
|
// circumvent this we create a wide open clip here (an empty clip rect
|
|
|
|
// is not sufficient).
|
|
|
|
// Internally, the SkRect passed to clipRect is converted to an SkIRect and
|
|
|
|
// rounded out. The following code creates a nearly maximal rect that will
|
|
|
|
// not get collapsed by the coming conversions (Due to precision loss the
|
|
|
|
// inset has to be surprisingly large).
|
|
|
|
SkIRect largeIRect = SkIRect::MakeLargest();
|
|
|
|
largeIRect.inset(1024, 1024);
|
2013-11-10 15:08:45 +00:00
|
|
|
SkRect large = SkRect::Make(largeIRect);
|
2013-11-07 22:20:31 +00:00
|
|
|
#ifdef SK_DEBUG
|
2014-11-19 13:03:18 +00:00
|
|
|
SkASSERT(!large.roundOut().isEmpty());
|
2013-11-07 22:20:31 +00:00
|
|
|
#endif
|
2014-02-28 18:19:39 +00:00
|
|
|
// call the base class' version to avoid adding a draw command
|
|
|
|
this->INHERITED::onClipRect(large, SkRegion::kReplace_Op, kHard_ClipEdgeStyle);
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2012-08-07 15:08:33 +00:00
|
|
|
SkDebugCanvas::~SkDebugCanvas() {
|
2013-01-02 20:20:31 +00:00
|
|
|
fCommandVector.deleteAll();
|
2013-02-06 20:13:54 +00:00
|
|
|
SkSafeUnref(fOverdrawFilter);
|
2013-12-04 13:42:46 +00:00
|
|
|
SkSafeUnref(fTexOverrideFilter);
|
2012-08-07 15:08:33 +00:00
|
|
|
}
|
2012-06-29 14:21:22 +00:00
|
|
|
|
|
|
|
void SkDebugCanvas::addDrawCommand(SkDrawCommand* command) {
|
2014-03-25 23:31:33 +00:00
|
|
|
command->setOffset(this->getOpID());
|
2013-01-02 20:20:31 +00:00
|
|
|
fCommandVector.push(command);
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SkDebugCanvas::draw(SkCanvas* canvas) {
|
2014-10-21 17:31:38 +00:00
|
|
|
fDrawNeedsReset = true;
|
|
|
|
|
2013-12-04 13:42:46 +00:00
|
|
|
if (!fCommandVector.isEmpty()) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->drawTo(canvas, fCommandVector.count() - 1);
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-01 15:57:52 +00:00
|
|
|
void SkDebugCanvas::applyUserTransform(SkCanvas* canvas) {
|
2013-01-17 16:30:56 +00:00
|
|
|
canvas->concat(fUserMatrix);
|
2012-08-01 15:57:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int SkDebugCanvas::getCommandAtPoint(int x, int y, int index) {
|
2012-07-31 19:55:32 +00:00
|
|
|
SkBitmap bitmap;
|
2014-01-24 18:53:42 +00:00
|
|
|
bitmap.allocPixels(SkImageInfo::MakeN32Premul(1, 1));
|
2012-07-31 19:55:32 +00:00
|
|
|
|
|
|
|
SkCanvas canvas(bitmap);
|
2012-09-06 18:43:21 +00:00
|
|
|
canvas.translate(SkIntToScalar(-x), SkIntToScalar(-y));
|
2014-08-29 15:03:56 +00:00
|
|
|
this->applyUserTransform(&canvas);
|
2012-07-31 19:55:32 +00:00
|
|
|
|
|
|
|
int layer = 0;
|
2012-07-31 20:07:42 +00:00
|
|
|
SkColor prev = bitmap.getColor(0,0);
|
2012-07-31 19:55:32 +00:00
|
|
|
for (int i = 0; i < index; i++) {
|
2013-01-02 20:20:31 +00:00
|
|
|
if (fCommandVector[i]->isVisible()) {
|
2014-10-16 21:28:28 +00:00
|
|
|
fCommandVector[i]->setUserMatrix(fUserMatrix);
|
2013-01-02 20:20:31 +00:00
|
|
|
fCommandVector[i]->execute(&canvas);
|
2012-07-31 19:55:32 +00:00
|
|
|
}
|
|
|
|
if (prev != bitmap.getColor(0,0)) {
|
|
|
|
layer = i;
|
|
|
|
}
|
|
|
|
prev = bitmap.getColor(0,0);
|
|
|
|
}
|
|
|
|
return layer;
|
|
|
|
}
|
|
|
|
|
2014-04-17 15:19:32 +00:00
|
|
|
class OverdrawXfermode : public SkXfermode {
|
|
|
|
public:
|
|
|
|
virtual SkPMColor xferColor(SkPMColor src, SkPMColor dst) const SK_OVERRIDE {
|
|
|
|
// This table encodes the color progression of the overdraw visualization
|
|
|
|
static const SkPMColor gTable[] = {
|
|
|
|
SkPackARGB32(0x00, 0x00, 0x00, 0x00),
|
|
|
|
SkPackARGB32(0xFF, 128, 158, 255),
|
|
|
|
SkPackARGB32(0xFF, 170, 185, 212),
|
|
|
|
SkPackARGB32(0xFF, 213, 195, 170),
|
|
|
|
SkPackARGB32(0xFF, 255, 192, 127),
|
|
|
|
SkPackARGB32(0xFF, 255, 185, 85),
|
|
|
|
SkPackARGB32(0xFF, 255, 165, 42),
|
|
|
|
SkPackARGB32(0xFF, 255, 135, 0),
|
|
|
|
SkPackARGB32(0xFF, 255, 95, 0),
|
|
|
|
SkPackARGB32(0xFF, 255, 50, 0),
|
|
|
|
SkPackARGB32(0xFF, 255, 0, 0)
|
|
|
|
};
|
2014-04-18 03:03:54 +00:00
|
|
|
|
2014-04-17 15:19:32 +00:00
|
|
|
for (size_t i = 0; i < SK_ARRAY_COUNT(gTable)-1; ++i) {
|
|
|
|
if (gTable[i] == dst) {
|
|
|
|
return gTable[i+1];
|
|
|
|
}
|
2013-02-06 20:13:54 +00:00
|
|
|
}
|
2014-04-18 03:03:54 +00:00
|
|
|
|
2014-04-17 15:19:32 +00:00
|
|
|
return gTable[SK_ARRAY_COUNT(gTable)-1];
|
2013-02-06 20:13:54 +00:00
|
|
|
}
|
|
|
|
|
2014-04-17 15:19:32 +00:00
|
|
|
virtual Factory getFactory() const SK_OVERRIDE { return NULL; }
|
|
|
|
#ifndef SK_IGNORE_TO_STRING
|
|
|
|
virtual void toString(SkString* str) const { str->set("OverdrawXfermode"); }
|
|
|
|
#endif
|
|
|
|
};
|
2013-02-06 20:13:54 +00:00
|
|
|
|
2013-10-17 17:56:10 +00:00
|
|
|
class SkOverdrawFilter : public SkDrawFilter {
|
2013-02-06 20:13:54 +00:00
|
|
|
public:
|
2013-10-17 17:56:10 +00:00
|
|
|
SkOverdrawFilter() {
|
2014-04-17 15:19:32 +00:00
|
|
|
fXferMode = SkNEW(OverdrawXfermode);
|
2013-02-06 20:13:54 +00:00
|
|
|
}
|
|
|
|
|
2013-10-17 17:56:10 +00:00
|
|
|
virtual ~SkOverdrawFilter() {
|
2013-02-06 20:13:54 +00:00
|
|
|
delete fXferMode;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool filter(SkPaint* p, Type) SK_OVERRIDE {
|
|
|
|
p->setXfermode(fXferMode);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
SkXfermode* fXferMode;
|
|
|
|
|
|
|
|
private:
|
|
|
|
typedef SkDrawFilter INHERITED;
|
|
|
|
};
|
|
|
|
|
2013-10-18 07:01:59 +00:00
|
|
|
// SkTexOverrideFilter modifies every paint to use the specified
|
2013-10-17 17:56:10 +00:00
|
|
|
// texture filtering mode
|
|
|
|
class SkTexOverrideFilter : public SkDrawFilter {
|
|
|
|
public:
|
|
|
|
SkTexOverrideFilter() : fFilterLevel(SkPaint::kNone_FilterLevel) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void setFilterLevel(SkPaint::FilterLevel filterLevel) {
|
|
|
|
fFilterLevel = filterLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool filter(SkPaint* p, Type) SK_OVERRIDE {
|
|
|
|
p->setFilterLevel(fFilterLevel);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
SkPaint::FilterLevel fFilterLevel;
|
|
|
|
|
|
|
|
private:
|
|
|
|
typedef SkDrawFilter INHERITED;
|
|
|
|
};
|
|
|
|
|
2014-03-03 16:32:17 +00:00
|
|
|
class SkDebugClipVisitor : public SkCanvas::ClipVisitor {
|
|
|
|
public:
|
|
|
|
SkDebugClipVisitor(SkCanvas* canvas) : fCanvas(canvas) {}
|
|
|
|
|
|
|
|
virtual void clipRect(const SkRect& r, SkRegion::Op, bool doAA) SK_OVERRIDE {
|
|
|
|
SkPaint p;
|
|
|
|
p.setColor(SK_ColorRED);
|
|
|
|
p.setStyle(SkPaint::kStroke_Style);
|
|
|
|
p.setAntiAlias(doAA);
|
|
|
|
fCanvas->drawRect(r, p);
|
|
|
|
}
|
|
|
|
virtual void clipRRect(const SkRRect& rr, SkRegion::Op, bool doAA) SK_OVERRIDE {
|
|
|
|
SkPaint p;
|
|
|
|
p.setColor(SK_ColorGREEN);
|
|
|
|
p.setStyle(SkPaint::kStroke_Style);
|
|
|
|
p.setAntiAlias(doAA);
|
|
|
|
fCanvas->drawRRect(rr, p);
|
|
|
|
}
|
|
|
|
virtual void clipPath(const SkPath& path, SkRegion::Op, bool doAA) SK_OVERRIDE {
|
|
|
|
SkPaint p;
|
|
|
|
p.setColor(SK_ColorBLUE);
|
|
|
|
p.setStyle(SkPaint::kStroke_Style);
|
|
|
|
p.setAntiAlias(doAA);
|
|
|
|
fCanvas->drawPath(path, p);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
SkCanvas* fCanvas;
|
|
|
|
|
|
|
|
private:
|
|
|
|
typedef SkCanvas::ClipVisitor INHERITED;
|
|
|
|
};
|
|
|
|
|
|
|
|
// set up the saveLayer commands so that the active ones
|
|
|
|
// return true in their 'active' method
|
2014-03-03 23:25:41 +00:00
|
|
|
void SkDebugCanvas::markActiveCommands(int index) {
|
|
|
|
fActiveLayers.rewind();
|
2014-03-03 16:32:17 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < fCommandVector.count(); ++i) {
|
|
|
|
fCommandVector[i]->setActive(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < index; ++i) {
|
|
|
|
SkDrawCommand::Action result = fCommandVector[i]->action();
|
2014-03-03 23:25:41 +00:00
|
|
|
if (SkDrawCommand::kPushLayer_Action == result) {
|
|
|
|
fActiveLayers.push(fCommandVector[i]);
|
|
|
|
} else if (SkDrawCommand::kPopLayer_Action == result) {
|
|
|
|
fActiveLayers.pop();
|
2014-03-03 16:32:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-03 23:25:41 +00:00
|
|
|
for (int i = 0; i < fActiveLayers.count(); ++i) {
|
|
|
|
fActiveLayers[i]->setActive(true);
|
|
|
|
}
|
|
|
|
|
2014-03-03 16:32:17 +00:00
|
|
|
}
|
|
|
|
|
2012-07-31 19:55:32 +00:00
|
|
|
void SkDebugCanvas::drawTo(SkCanvas* canvas, int index) {
|
2013-01-02 20:20:31 +00:00
|
|
|
SkASSERT(!fCommandVector.isEmpty());
|
|
|
|
SkASSERT(index < fCommandVector.count());
|
2013-12-04 13:42:46 +00:00
|
|
|
int i = 0;
|
2012-08-01 15:57:52 +00:00
|
|
|
|
2014-05-19 13:53:10 +00:00
|
|
|
bool pathOpsMode = getAllowSimplifyClip();
|
|
|
|
canvas->setAllowSimplifyClip(pathOpsMode);
|
2012-08-01 15:57:52 +00:00
|
|
|
// This only works assuming the canvas and device are the same ones that
|
|
|
|
// were previously drawn into because they need to preserve all saves
|
|
|
|
// and restores.
|
2013-12-04 13:42:46 +00:00
|
|
|
// The visibility filter also requires a full re-draw - otherwise we can
|
|
|
|
// end up drawing the filter repeatedly.
|
2014-10-16 21:28:28 +00:00
|
|
|
if (fIndex < index && !fFilter && !fMegaVizMode && !pathOpsMode && !fDrawNeedsReset) {
|
2012-08-01 15:57:52 +00:00
|
|
|
i = fIndex + 1;
|
|
|
|
} else {
|
2012-11-27 16:09:42 +00:00
|
|
|
for (int j = 0; j < fOutstandingSaveCount; j++) {
|
|
|
|
canvas->restore();
|
|
|
|
}
|
2012-12-06 21:47:40 +00:00
|
|
|
canvas->clear(SK_ColorTRANSPARENT);
|
2012-08-01 15:57:52 +00:00
|
|
|
canvas->resetMatrix();
|
2014-10-16 21:28:28 +00:00
|
|
|
SkRect rect = SkRect::MakeWH(SkIntToScalar(fWindowSize.fWidth),
|
2014-08-29 15:03:56 +00:00
|
|
|
SkIntToScalar(fWindowSize.fHeight));
|
|
|
|
canvas->clipRect(rect, SkRegion::kReplace_Op);
|
|
|
|
this->applyUserTransform(canvas);
|
2014-10-16 21:28:28 +00:00
|
|
|
fDrawNeedsReset = false;
|
2012-11-27 16:09:42 +00:00
|
|
|
fOutstandingSaveCount = 0;
|
2013-08-05 16:31:27 +00:00
|
|
|
}
|
2013-02-06 20:13:54 +00:00
|
|
|
|
2013-08-05 16:31:27 +00:00
|
|
|
// The setting of the draw filter has to go here (rather than in
|
|
|
|
// SkRasterWidget) due to the canvas restores this class performs.
|
|
|
|
// Since the draw filter is stored in the layer stack if we
|
|
|
|
// call setDrawFilter on anything but the root layer odd things happen.
|
|
|
|
if (fOverdrawViz) {
|
|
|
|
if (NULL == fOverdrawFilter) {
|
2013-10-17 17:56:10 +00:00
|
|
|
fOverdrawFilter = new SkOverdrawFilter;
|
2013-08-05 16:31:27 +00:00
|
|
|
}
|
2013-02-06 20:13:54 +00:00
|
|
|
|
2013-08-05 16:31:27 +00:00
|
|
|
if (fOverdrawFilter != canvas->getDrawFilter()) {
|
|
|
|
canvas->setDrawFilter(fOverdrawFilter);
|
2013-02-06 20:13:54 +00:00
|
|
|
}
|
2013-10-17 17:56:10 +00:00
|
|
|
} else if (fOverrideTexFiltering) {
|
|
|
|
if (NULL == fTexOverrideFilter) {
|
|
|
|
fTexOverrideFilter = new SkTexOverrideFilter;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fTexOverrideFilter != canvas->getDrawFilter()) {
|
|
|
|
canvas->setDrawFilter(fTexOverrideFilter);
|
|
|
|
}
|
2013-08-05 16:31:27 +00:00
|
|
|
} else {
|
|
|
|
canvas->setDrawFilter(NULL);
|
2012-08-01 15:57:52 +00:00
|
|
|
}
|
|
|
|
|
2014-03-03 16:32:17 +00:00
|
|
|
if (fMegaVizMode) {
|
2014-03-03 23:25:41 +00:00
|
|
|
this->markActiveCommands(index);
|
2014-03-03 16:32:17 +00:00
|
|
|
}
|
|
|
|
|
2012-08-01 15:57:52 +00:00
|
|
|
for (; i <= index; i++) {
|
2012-07-31 19:55:32 +00:00
|
|
|
if (i == index && fFilter) {
|
2014-08-29 15:03:56 +00:00
|
|
|
canvas->clear(0xAAFFFFFF);
|
2012-07-31 19:55:32 +00:00
|
|
|
}
|
2012-06-29 14:21:22 +00:00
|
|
|
|
2013-01-02 20:20:31 +00:00
|
|
|
if (fCommandVector[i]->isVisible()) {
|
2014-03-03 16:32:17 +00:00
|
|
|
if (fMegaVizMode && fCommandVector[i]->active()) {
|
2014-03-03 23:25:41 +00:00
|
|
|
// "active" commands execute their visualization behaviors:
|
|
|
|
// All active saveLayers get replaced with saves so all draws go to the
|
|
|
|
// visible canvas.
|
|
|
|
// All active culls draw their cull box
|
|
|
|
fCommandVector[i]->vizExecute(canvas);
|
2014-03-03 16:32:17 +00:00
|
|
|
} else {
|
2014-10-16 21:28:28 +00:00
|
|
|
fCommandVector[i]->setUserMatrix(fUserMatrix);
|
2014-03-03 16:32:17 +00:00
|
|
|
fCommandVector[i]->execute(canvas);
|
|
|
|
}
|
2014-03-03 23:25:41 +00:00
|
|
|
|
|
|
|
fCommandVector[i]->trackSaveState(&fOutstandingSaveCount);
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
}
|
2014-03-03 16:32:17 +00:00
|
|
|
|
|
|
|
if (fMegaVizMode) {
|
2014-12-12 16:46:25 +00:00
|
|
|
SkRect r = SkRect::MakeWH(SkIntToScalar(fWindowSize.fWidth),
|
2014-08-29 15:03:56 +00:00
|
|
|
SkIntToScalar(fWindowSize.fHeight));
|
2014-03-03 16:32:17 +00:00
|
|
|
r.outset(SK_Scalar1, SK_Scalar1);
|
|
|
|
|
|
|
|
canvas->save();
|
|
|
|
// nuke the CTM
|
2014-08-29 15:03:56 +00:00
|
|
|
canvas->resetMatrix();
|
2014-03-03 16:32:17 +00:00
|
|
|
// turn off clipping
|
|
|
|
canvas->clipRect(r, SkRegion::kReplace_Op);
|
|
|
|
|
|
|
|
// visualize existing clips
|
|
|
|
SkDebugClipVisitor visitor(canvas);
|
|
|
|
|
|
|
|
canvas->replayClips(&visitor);
|
|
|
|
|
|
|
|
canvas->restore();
|
|
|
|
}
|
2014-05-19 13:53:10 +00:00
|
|
|
if (pathOpsMode) {
|
|
|
|
this->resetClipStackData();
|
|
|
|
const SkClipStack* clipStack = canvas->getClipStack();
|
|
|
|
SkClipStack::Iter iter(*clipStack, SkClipStack::Iter::kBottom_IterStart);
|
|
|
|
const SkClipStack::Element* element;
|
|
|
|
SkPath devPath;
|
|
|
|
while ((element = iter.next())) {
|
|
|
|
SkClipStack::Element::Type type = element->getType();
|
|
|
|
SkPath operand;
|
|
|
|
if (type != SkClipStack::Element::kEmpty_Type) {
|
|
|
|
element->asPath(&operand);
|
|
|
|
}
|
|
|
|
SkRegion::Op elementOp = element->getOp();
|
|
|
|
this->addClipStackData(devPath, operand, elementOp);
|
|
|
|
if (elementOp == SkRegion::kReplace_Op) {
|
|
|
|
devPath = operand;
|
|
|
|
} else {
|
|
|
|
Op(devPath, operand, (SkPathOp) elementOp, &devPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this->lastClipStackData(devPath);
|
|
|
|
}
|
2012-08-03 17:32:05 +00:00
|
|
|
fMatrix = canvas->getTotalMatrix();
|
2014-03-08 03:57:19 +00:00
|
|
|
if (!canvas->getClipDeviceBounds(&fClip)) {
|
|
|
|
fClip.setEmpty();
|
|
|
|
}
|
2012-08-01 15:57:52 +00:00
|
|
|
fIndex = index;
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2013-04-01 18:18:49 +00:00
|
|
|
void SkDebugCanvas::deleteDrawCommandAt(int index) {
|
|
|
|
SkASSERT(index < fCommandVector.count());
|
|
|
|
delete fCommandVector[index];
|
|
|
|
fCommandVector.remove(index);
|
|
|
|
}
|
|
|
|
|
2012-06-29 14:21:22 +00:00
|
|
|
SkDrawCommand* SkDebugCanvas::getDrawCommandAt(int index) {
|
2013-01-02 20:20:31 +00:00
|
|
|
SkASSERT(index < fCommandVector.count());
|
|
|
|
return fCommandVector[index];
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2013-04-01 18:18:49 +00:00
|
|
|
void SkDebugCanvas::setDrawCommandAt(int index, SkDrawCommand* command) {
|
|
|
|
SkASSERT(index < fCommandVector.count());
|
|
|
|
delete fCommandVector[index];
|
|
|
|
fCommandVector[index] = command;
|
|
|
|
}
|
|
|
|
|
2014-11-09 00:18:56 +00:00
|
|
|
const SkTDArray<SkString*>* SkDebugCanvas::getCommandInfo(int index) const {
|
2013-01-02 20:20:31 +00:00
|
|
|
SkASSERT(index < fCommandVector.count());
|
|
|
|
return fCommandVector[index]->Info();
|
2012-07-17 15:40:51 +00:00
|
|
|
}
|
2012-06-29 14:21:22 +00:00
|
|
|
|
2012-07-17 15:40:51 +00:00
|
|
|
bool SkDebugCanvas::getDrawCommandVisibilityAt(int index) {
|
2013-01-02 20:20:31 +00:00
|
|
|
SkASSERT(index < fCommandVector.count());
|
|
|
|
return fCommandVector[index]->isVisible();
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2012-11-19 20:44:29 +00:00
|
|
|
const SkTDArray <SkDrawCommand*>& SkDebugCanvas::getDrawCommands() const {
|
2013-01-02 20:20:31 +00:00
|
|
|
return fCommandVector;
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2013-03-11 22:53:11 +00:00
|
|
|
SkTDArray <SkDrawCommand*>& SkDebugCanvas::getDrawCommands() {
|
|
|
|
return fCommandVector;
|
|
|
|
}
|
|
|
|
|
2013-10-18 07:01:59 +00:00
|
|
|
void SkDebugCanvas::overrideTexFiltering(bool overrideTexFiltering, SkPaint::FilterLevel level) {
|
2013-10-17 17:56:10 +00:00
|
|
|
if (NULL == fTexOverrideFilter) {
|
|
|
|
fTexOverrideFilter = new SkTexOverrideFilter;
|
|
|
|
}
|
|
|
|
|
2013-10-18 07:01:59 +00:00
|
|
|
fOverrideTexFiltering = overrideTexFiltering;
|
2013-10-17 17:56:10 +00:00
|
|
|
fTexOverrideFilter->setFilterLevel(level);
|
|
|
|
}
|
|
|
|
|
2014-02-28 18:19:39 +00:00
|
|
|
void SkDebugCanvas::onClipPath(const SkPath& path, SkRegion::Op op, ClipEdgeStyle edgeStyle) {
|
|
|
|
this->addDrawCommand(new SkClipPathCommand(path, op, kSoft_ClipEdgeStyle == edgeStyle));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 18:19:39 +00:00
|
|
|
void SkDebugCanvas::onClipRect(const SkRect& rect, SkRegion::Op op, ClipEdgeStyle edgeStyle) {
|
|
|
|
this->addDrawCommand(new SkClipRectCommand(rect, op, kSoft_ClipEdgeStyle == edgeStyle));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 18:19:39 +00:00
|
|
|
void SkDebugCanvas::onClipRRect(const SkRRect& rrect, SkRegion::Op op, ClipEdgeStyle edgeStyle) {
|
|
|
|
this->addDrawCommand(new SkClipRRectCommand(rrect, op, kSoft_ClipEdgeStyle == edgeStyle));
|
2013-01-02 20:20:31 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 18:19:39 +00:00
|
|
|
void SkDebugCanvas::onClipRegion(const SkRegion& region, SkRegion::Op op) {
|
|
|
|
this->addDrawCommand(new SkClipRegionCommand(region, op));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-03-13 20:03:58 +00:00
|
|
|
void SkDebugCanvas::didConcat(const SkMatrix& matrix) {
|
2014-03-25 17:32:26 +00:00
|
|
|
switch (matrix.getType()) {
|
|
|
|
case SkMatrix::kTranslate_Mask:
|
|
|
|
this->addDrawCommand(new SkTranslateCommand(matrix.getTranslateX(),
|
|
|
|
matrix.getTranslateY()));
|
|
|
|
break;
|
|
|
|
case SkMatrix::kScale_Mask:
|
|
|
|
this->addDrawCommand(new SkScaleCommand(matrix.getScaleX(),
|
|
|
|
matrix.getScaleY()));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
this->addDrawCommand(new SkConcatCommand(matrix));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-03-13 20:03:58 +00:00
|
|
|
this->INHERITED::didConcat(matrix);
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawBitmap(const SkBitmap& bitmap, SkScalar left,
|
|
|
|
SkScalar top, const SkPaint* paint = NULL) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawBitmapCommand(bitmap, left, top, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawBitmapRectToRect(const SkBitmap& bitmap,
|
|
|
|
const SkRect* src, const SkRect& dst,
|
|
|
|
const SkPaint* paint,
|
|
|
|
SkCanvas::DrawBitmapRectFlags flags) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawBitmapRectCommand(bitmap, src, dst, paint, flags));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawBitmapNine(const SkBitmap& bitmap,
|
|
|
|
const SkIRect& center, const SkRect& dst, const SkPaint* paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawBitmapNineCommand(bitmap, center, dst, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2013-05-29 13:24:23 +00:00
|
|
|
void SkDebugCanvas::beginCommentGroup(const char* description) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkBeginCommentGroupCommand(description));
|
2013-05-29 13:24:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SkDebugCanvas::addComment(const char* kywd, const char* value) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkCommentCommand(kywd, value));
|
2013-05-29 13:24:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SkDebugCanvas::endCommentGroup() {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkEndCommentGroupCommand());
|
2013-05-29 13:24:23 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawOval(const SkRect& oval, const SkPaint& paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawOvalCommand(oval, paint));
|
2013-01-02 20:20:31 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawPaint(const SkPaint& paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawPaintCommand(paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawPath(const SkPath& path, const SkPaint& paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawPathCommand(path, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-12-12 16:46:25 +00:00
|
|
|
void SkDebugCanvas::onDrawPicture(const SkPicture* picture,
|
|
|
|
const SkMatrix* matrix,
|
2014-08-13 17:46:23 +00:00
|
|
|
const SkPaint* paint) {
|
|
|
|
this->addDrawCommand(new SkDrawPictureCommand(picture, matrix, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawPoints(PointMode mode, size_t count,
|
|
|
|
const SkPoint pts[], const SkPaint& paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawPointsCommand(mode, count, pts, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-04-23 04:00:17 +00:00
|
|
|
void SkDebugCanvas::onDrawPosText(const void* text, size_t byteLength, const SkPoint pos[],
|
|
|
|
const SkPaint& paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawPosTextCommand(text, byteLength, pos, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-04-23 04:00:17 +00:00
|
|
|
void SkDebugCanvas::onDrawPosTextH(const void* text, size_t byteLength, const SkScalar xpos[],
|
|
|
|
SkScalar constY, const SkPaint& paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(
|
2013-06-18 20:20:55 +00:00
|
|
|
new SkDrawPosTextHCommand(text, byteLength, xpos, constY, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawRect(const SkRect& rect, const SkPaint& paint) {
|
2012-06-29 14:21:22 +00:00
|
|
|
// NOTE(chudy): Messing up when renamed to DrawRect... Why?
|
2013-06-18 20:20:55 +00:00
|
|
|
addDrawCommand(new SkDrawRectCommand(rect, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawRRect(const SkRRect& rrect, const SkPaint& paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawRRectCommand(rrect, paint));
|
2013-01-02 20:20:31 +00:00
|
|
|
}
|
|
|
|
|
2014-02-21 12:20:45 +00:00
|
|
|
void SkDebugCanvas::onDrawDRRect(const SkRRect& outer, const SkRRect& inner,
|
|
|
|
const SkPaint& paint) {
|
2014-02-24 17:28:55 +00:00
|
|
|
this->addDrawCommand(new SkDrawDRRectCommand(outer, inner, paint));
|
2014-02-21 12:20:45 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawSprite(const SkBitmap& bitmap, int left, int top,
|
|
|
|
const SkPaint* paint = NULL) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawSpriteCommand(bitmap, left, top, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-04-23 04:00:17 +00:00
|
|
|
void SkDebugCanvas::onDrawText(const void* text, size_t byteLength, SkScalar x, SkScalar y,
|
|
|
|
const SkPaint& paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawTextCommand(text, byteLength, x, y, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-04-23 04:00:17 +00:00
|
|
|
void SkDebugCanvas::onDrawTextOnPath(const void* text, size_t byteLength, const SkPath& path,
|
|
|
|
const SkMatrix* matrix, const SkPaint& paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(
|
2013-06-18 20:20:55 +00:00
|
|
|
new SkDrawTextOnPathCommand(text, byteLength, path, matrix, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-08-26 14:56:44 +00:00
|
|
|
void SkDebugCanvas::onDrawTextBlob(const SkTextBlob* blob, SkScalar x, SkScalar y,
|
|
|
|
const SkPaint& paint) {
|
|
|
|
this->addDrawCommand(new SkDrawTextBlobCommand(blob, x, y, paint));
|
|
|
|
}
|
|
|
|
|
2015-01-05 12:49:38 +00:00
|
|
|
void SkDebugCanvas::drawVertices(VertexMode vmode, int vertexCount,
|
|
|
|
const SkPoint vertices[], const SkPoint texs[], const SkColor colors[],
|
|
|
|
SkXfermode*, const uint16_t indices[], int indexCount,
|
|
|
|
const SkPaint& paint) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkDrawVerticesCommand(vmode, vertexCount, vertices,
|
|
|
|
texs, colors, NULL, indices, indexCount, paint));
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-03-12 20:21:48 +00:00
|
|
|
void SkDebugCanvas::willRestore() {
|
|
|
|
this->addDrawCommand(new SkRestoreCommand());
|
|
|
|
this->INHERITED::willRestore();
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-06-30 14:13:28 +00:00
|
|
|
void SkDebugCanvas::willSave() {
|
|
|
|
this->addDrawCommand(new SkSaveCommand());
|
|
|
|
this->INHERITED::willSave();
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-03-12 20:21:48 +00:00
|
|
|
SkCanvas::SaveLayerStrategy SkDebugCanvas::willSaveLayer(const SkRect* bounds, const SkPaint* paint,
|
|
|
|
SaveFlags flags) {
|
|
|
|
this->addDrawCommand(new SkSaveLayerCommand(bounds, paint, flags));
|
|
|
|
this->INHERITED::willSaveLayer(bounds, paint, flags);
|
|
|
|
// No need for a full layer.
|
|
|
|
return kNoLayer_SaveLayerStrategy;
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-03-13 20:03:58 +00:00
|
|
|
void SkDebugCanvas::didSetMatrix(const SkMatrix& matrix) {
|
2014-03-25 23:31:33 +00:00
|
|
|
this->addDrawCommand(new SkSetMatrixCommand(matrix));
|
2014-03-13 20:03:58 +00:00
|
|
|
this->INHERITED::didSetMatrix(matrix);
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SkDebugCanvas::toggleCommand(int index, bool toggle) {
|
2013-01-02 20:20:31 +00:00
|
|
|
SkASSERT(index < fCommandVector.count());
|
|
|
|
fCommandVector[index]->setVisible(toggle);
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
2014-05-19 13:53:10 +00:00
|
|
|
|
|
|
|
static const char* gFillTypeStrs[] = {
|
|
|
|
"kWinding_FillType",
|
|
|
|
"kEvenOdd_FillType",
|
|
|
|
"kInverseWinding_FillType",
|
|
|
|
"kInverseEvenOdd_FillType"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char* gOpStrs[] = {
|
|
|
|
"kDifference_PathOp",
|
|
|
|
"kIntersect_PathOp",
|
|
|
|
"kUnion_PathOp",
|
|
|
|
"kXor_PathOp",
|
|
|
|
"kReverseDifference_PathOp",
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char kHTML4SpaceIndent[] = " ";
|
|
|
|
|
|
|
|
void SkDebugCanvas::outputScalar(SkScalar num) {
|
|
|
|
if (num == (int) num) {
|
|
|
|
fClipStackData.appendf("%d", (int) num);
|
|
|
|
} else {
|
|
|
|
SkString str;
|
|
|
|
str.printf("%1.9g", num);
|
|
|
|
int width = (int) str.size();
|
|
|
|
const char* cStr = str.c_str();
|
|
|
|
while (cStr[width - 1] == '0') {
|
|
|
|
--width;
|
|
|
|
}
|
|
|
|
str.resize(width);
|
|
|
|
fClipStackData.appendf("%sf", str.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkDebugCanvas::outputPointsCommon(const SkPoint* pts, int count) {
|
|
|
|
for (int index = 0; index < count; ++index) {
|
|
|
|
this->outputScalar(pts[index].fX);
|
|
|
|
fClipStackData.appendf(", ");
|
|
|
|
this->outputScalar(pts[index].fY);
|
|
|
|
if (index + 1 < count) {
|
|
|
|
fClipStackData.appendf(", ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkDebugCanvas::outputPoints(const SkPoint* pts, int count) {
|
|
|
|
this->outputPointsCommon(pts, count);
|
|
|
|
fClipStackData.appendf(");<br>");
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkDebugCanvas::outputConicPoints(const SkPoint* pts, SkScalar weight) {
|
|
|
|
this->outputPointsCommon(pts, 2);
|
|
|
|
fClipStackData.appendf(", ");
|
|
|
|
this->outputScalar(weight);
|
|
|
|
fClipStackData.appendf(");<br>");
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkDebugCanvas::addPathData(const SkPath& path, const char* pathName) {
|
|
|
|
SkPath::RawIter iter(path);
|
|
|
|
SkPath::FillType fillType = path.getFillType();
|
|
|
|
fClipStackData.appendf("%sSkPath %s;<br>", kHTML4SpaceIndent, pathName);
|
|
|
|
fClipStackData.appendf("%s%s.setFillType(SkPath::%s);<br>", kHTML4SpaceIndent, pathName,
|
|
|
|
gFillTypeStrs[fillType]);
|
|
|
|
iter.setPath(path);
|
|
|
|
uint8_t verb;
|
|
|
|
SkPoint pts[4];
|
|
|
|
while ((verb = iter.next(pts)) != SkPath::kDone_Verb) {
|
|
|
|
switch (verb) {
|
|
|
|
case SkPath::kMove_Verb:
|
|
|
|
fClipStackData.appendf("%s%s.moveTo(", kHTML4SpaceIndent, pathName);
|
|
|
|
this->outputPoints(&pts[0], 1);
|
|
|
|
continue;
|
|
|
|
case SkPath::kLine_Verb:
|
|
|
|
fClipStackData.appendf("%s%s.lineTo(", kHTML4SpaceIndent, pathName);
|
|
|
|
this->outputPoints(&pts[1], 1);
|
|
|
|
break;
|
|
|
|
case SkPath::kQuad_Verb:
|
|
|
|
fClipStackData.appendf("%s%s.quadTo(", kHTML4SpaceIndent, pathName);
|
|
|
|
this->outputPoints(&pts[1], 2);
|
|
|
|
break;
|
|
|
|
case SkPath::kConic_Verb:
|
|
|
|
fClipStackData.appendf("%s%s.conicTo(", kHTML4SpaceIndent, pathName);
|
|
|
|
this->outputConicPoints(&pts[1], iter.conicWeight());
|
|
|
|
break;
|
|
|
|
case SkPath::kCubic_Verb:
|
|
|
|
fClipStackData.appendf("%s%s.cubicTo(", kHTML4SpaceIndent, pathName);
|
|
|
|
this->outputPoints(&pts[1], 3);
|
|
|
|
break;
|
|
|
|
case SkPath::kClose_Verb:
|
|
|
|
fClipStackData.appendf("%s%s.close();<br>", kHTML4SpaceIndent, pathName);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SkDEBUGFAIL("bad verb");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkDebugCanvas::addClipStackData(const SkPath& devPath, const SkPath& operand,
|
|
|
|
SkRegion::Op elementOp) {
|
|
|
|
if (elementOp == SkRegion::kReplace_Op) {
|
|
|
|
if (!lastClipStackData(devPath)) {
|
|
|
|
fSaveDevPath = operand;
|
|
|
|
}
|
|
|
|
fCalledAddStackData = false;
|
|
|
|
} else {
|
|
|
|
fClipStackData.appendf("<br>static void test(skiatest::Reporter* reporter,"
|
|
|
|
" const char* filename) {<br>");
|
|
|
|
addPathData(fCalledAddStackData ? devPath : fSaveDevPath, "path");
|
|
|
|
addPathData(operand, "pathB");
|
|
|
|
fClipStackData.appendf("%stestPathOp(reporter, path, pathB, %s, filename);<br>",
|
|
|
|
kHTML4SpaceIndent, gOpStrs[elementOp]);
|
|
|
|
fClipStackData.appendf("}<br>");
|
|
|
|
fCalledAddStackData = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SkDebugCanvas::lastClipStackData(const SkPath& devPath) {
|
|
|
|
if (fCalledAddStackData) {
|
|
|
|
fClipStackData.appendf("<br>");
|
|
|
|
addPathData(devPath, "pathOut");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|