| /* |
| * Copyright 2012 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "tools/debugger/DrawCommand.h" |
| |
| #include <algorithm> |
| #include "include/core/SkColorFilter.h" |
| #include "include/core/SkDrawable.h" |
| #include "include/core/SkImageFilter.h" |
| #include "include/core/SkPathEffect.h" |
| #include "include/core/SkPicture.h" |
| #include "include/core/SkTypeface.h" |
| #include "include/effects/SkDashPathEffect.h" |
| #include "include/encode/SkPngEncoder.h" |
| #include "include/private/SkShadowFlags.h" |
| #include "include/private/SkTHash.h" |
| #include "src/core/SkAutoMalloc.h" |
| #include "src/core/SkCanvasPriv.h" |
| #include "src/core/SkClipOpPriv.h" |
| #include "src/core/SkLatticeIter.h" |
| #include "src/core/SkMaskFilterBase.h" |
| #include "src/core/SkPaintDefaults.h" |
| #include "src/core/SkPaintPriv.h" |
| #include "src/core/SkReadBuffer.h" |
| #include "src/core/SkRectPriv.h" |
| #include "src/core/SkTextBlobPriv.h" |
| #include "src/core/SkWriteBuffer.h" |
| #include "src/image/SkImage_Base.h" |
| #include "tools/debugger/DebugLayerManager.h" |
| #include "tools/debugger/JsonWriteBuffer.h" |
| |
| #ifdef SK_SUPPORT_GPU |
| #include "include/gpu/GrDirectContext.h" |
| #else |
| class GrDirectContext; |
| #endif |
| |
| #define DEBUGCANVAS_ATTRIBUTE_COMMAND "command" |
| #define DEBUGCANVAS_ATTRIBUTE_VISIBLE "visible" |
| #define DEBUGCANVAS_ATTRIBUTE_MATRIX "matrix" |
| #define DEBUGCANVAS_ATTRIBUTE_DRAWDEPTHTRANS "drawDepthTranslation" |
| #define DEBUGCANVAS_ATTRIBUTE_COORDS "coords" |
| #define DEBUGCANVAS_ATTRIBUTE_EDGING "edging" |
| #define DEBUGCANVAS_ATTRIBUTE_HINTING "hinting" |
| #define DEBUGCANVAS_ATTRIBUTE_BOUNDS "bounds" |
| #define DEBUGCANVAS_ATTRIBUTE_PAINT "paint" |
| #define DEBUGCANVAS_ATTRIBUTE_OUTER "outer" |
| #define DEBUGCANVAS_ATTRIBUTE_INNER "inner" |
| #define DEBUGCANVAS_ATTRIBUTE_MODE "mode" |
| #define DEBUGCANVAS_ATTRIBUTE_POINTS "points" |
| #define DEBUGCANVAS_ATTRIBUTE_PATH "path" |
| #define DEBUGCANVAS_ATTRIBUTE_TEXT "text" |
| #define DEBUGCANVAS_ATTRIBUTE_COLOR "color" |
| #define DEBUGCANVAS_ATTRIBUTE_ALPHA "alpha" |
| #define DEBUGCANVAS_ATTRIBUTE_BLENDMODE "blendMode" |
| #define DEBUGCANVAS_ATTRIBUTE_STYLE "style" |
| #define DEBUGCANVAS_ATTRIBUTE_STROKEWIDTH "strokeWidth" |
| #define DEBUGCANVAS_ATTRIBUTE_STROKEMITER "strokeMiter" |
| #define DEBUGCANVAS_ATTRIBUTE_STROKEJOIN "strokeJoin" |
| #define DEBUGCANVAS_ATTRIBUTE_CAP "cap" |
| #define DEBUGCANVAS_ATTRIBUTE_ANTIALIAS "antiAlias" |
| #define DEBUGCANVAS_ATTRIBUTE_DITHER "dither" |
| #define DEBUGCANVAS_ATTRIBUTE_FAKEBOLDTEXT "fakeBoldText" |
| #define DEBUGCANVAS_ATTRIBUTE_LINEARTEXT "linearText" |
| #define DEBUGCANVAS_ATTRIBUTE_SUBPIXELTEXT "subpixelText" |
| #define DEBUGCANVAS_ATTRIBUTE_DEVKERNTEXT "devKernText" |
| #define DEBUGCANVAS_ATTRIBUTE_LCDRENDERTEXT "lcdRenderText" |
| #define DEBUGCANVAS_ATTRIBUTE_EMBEDDEDBITMAPTEXT "embeddedBitmapText" |
| #define DEBUGCANVAS_ATTRIBUTE_AUTOHINTING "forceAutoHinting" |
| #define DEBUGCANVAS_ATTRIBUTE_REGION "region" |
| #define DEBUGCANVAS_ATTRIBUTE_REGIONOP "op" |
| #define DEBUGCANVAS_ATTRIBUTE_EDGESTYLE "edgeStyle" |
| #define DEBUGCANVAS_ATTRIBUTE_DEVICEREGION "deviceRegion" |
| #define DEBUGCANVAS_ATTRIBUTE_BLUR "blur" |
| #define DEBUGCANVAS_ATTRIBUTE_SIGMA "sigma" |
| #define DEBUGCANVAS_ATTRIBUTE_QUALITY "quality" |
| #define DEBUGCANVAS_ATTRIBUTE_TEXTSIZE "textSize" |
| #define DEBUGCANVAS_ATTRIBUTE_TEXTSCALEX "textScaleX" |
| #define DEBUGCANVAS_ATTRIBUTE_TEXTSKEWX "textSkewX" |
| #define DEBUGCANVAS_ATTRIBUTE_DASHING "dashing" |
| #define DEBUGCANVAS_ATTRIBUTE_INTERVALS "intervals" |
| #define DEBUGCANVAS_ATTRIBUTE_PHASE "phase" |
| #define DEBUGCANVAS_ATTRIBUTE_FILLTYPE "fillType" |
| #define DEBUGCANVAS_ATTRIBUTE_VERBS "verbs" |
| #define DEBUGCANVAS_ATTRIBUTE_NAME "name" |
| #define DEBUGCANVAS_ATTRIBUTE_DATA "data" |
| #define DEBUGCANVAS_ATTRIBUTE_VALUES "values" |
| #define DEBUGCANVAS_ATTRIBUTE_SHADER "shader" |
| #define DEBUGCANVAS_ATTRIBUTE_PATHEFFECT "pathEffect" |
| #define DEBUGCANVAS_ATTRIBUTE_MASKFILTER "maskFilter" |
| #define DEBUGCANVAS_ATTRIBUTE_XFERMODE "xfermode" |
| #define DEBUGCANVAS_ATTRIBUTE_BACKDROP "backdrop" |
| #define DEBUGCANVAS_ATTRIBUTE_COLORFILTER "colorfilter" |
| #define DEBUGCANVAS_ATTRIBUTE_IMAGEFILTER "imagefilter" |
| #define DEBUGCANVAS_ATTRIBUTE_IMAGE "image" |
| #define DEBUGCANVAS_ATTRIBUTE_IMAGE_INDEX "imageIndex" |
| #define DEBUGCANVAS_ATTRIBUTE_BITMAP "bitmap" |
| #define DEBUGCANVAS_ATTRIBUTE_SRC "src" |
| #define DEBUGCANVAS_ATTRIBUTE_DST "dst" |
| #define DEBUGCANVAS_ATTRIBUTE_CENTER "center" |
| #define DEBUGCANVAS_ATTRIBUTE_STRICT "strict" |
| #define DEBUGCANVAS_ATTRIBUTE_DESCRIPTION "description" |
| #define DEBUGCANVAS_ATTRIBUTE_X "x" |
| #define DEBUGCANVAS_ATTRIBUTE_Y "y" |
| #define DEBUGCANVAS_ATTRIBUTE_RUNS "runs" |
| #define DEBUGCANVAS_ATTRIBUTE_POSITIONS "positions" |
| #define DEBUGCANVAS_ATTRIBUTE_GLYPHS "glyphs" |
| #define DEBUGCANVAS_ATTRIBUTE_FONT "font" |
| #define DEBUGCANVAS_ATTRIBUTE_TYPEFACE "typeface" |
| #define DEBUGCANVAS_ATTRIBUTE_CUBICS "cubics" |
| #define DEBUGCANVAS_ATTRIBUTE_COLORS "colors" |
| #define DEBUGCANVAS_ATTRIBUTE_TEXTURECOORDS "textureCoords" |
| #define DEBUGCANVAS_ATTRIBUTE_STARTANGLE "startAngle" |
| #define DEBUGCANVAS_ATTRIBUTE_SWEEPANGLE "sweepAngle" |
| #define DEBUGCANVAS_ATTRIBUTE_USECENTER "useCenter" |
| #define DEBUGCANVAS_ATTRIBUTE_SHORTDESC "shortDesc" |
| #define DEBUGCANVAS_ATTRIBUTE_UNIQUE_ID "uniqueID" |
| #define DEBUGCANVAS_ATTRIBUTE_WIDTH "width" |
| #define DEBUGCANVAS_ATTRIBUTE_HEIGHT "height" |
| #define DEBUGCANVAS_ATTRIBUTE_ALPHA "alpha" |
| #define DEBUGCANVAS_ATTRIBUTE_LATTICE "lattice" |
| #define DEBUGCANVAS_ATTRIBUTE_LATTICEXCOUNT "xCount" |
| #define DEBUGCANVAS_ATTRIBUTE_LATTICEYCOUNT "yCount" |
| #define DEBUGCANVAS_ATTRIBUTE_LATTICEXDIVS "xDivs" |
| #define DEBUGCANVAS_ATTRIBUTE_LATTICEYDIVS "yDivs" |
| #define DEBUGCANVAS_ATTRIBUTE_LATTICEFLAGS "flags" |
| #define DEBUGCANVAS_ATTRIBUTE_ZPLANE "zPlane" |
| #define DEBUGCANVAS_ATTRIBUTE_LIGHTPOSITION "lightPositions" |
| #define DEBUGCANVAS_ATTRIBUTE_AMBIENTCOLOR "ambientColor" |
| #define DEBUGCANVAS_ATTRIBUTE_SPOTCOLOR "spotColor" |
| #define DEBUGCANVAS_ATTRIBUTE_LIGHTRADIUS "lightRadius" |
| #define DEBUGCANVAS_ATTRIBUTE_LAYERNODEID "layerNodeId" |
| |
| #define DEBUGCANVAS_VERB_MOVE "move" |
| #define DEBUGCANVAS_VERB_LINE "line" |
| #define DEBUGCANVAS_VERB_QUAD "quad" |
| #define DEBUGCANVAS_VERB_CUBIC "cubic" |
| #define DEBUGCANVAS_VERB_CONIC "conic" |
| #define DEBUGCANVAS_VERB_CLOSE "close" |
| |
| #define DEBUGCANVAS_STYLE_FILL "fill" |
| #define DEBUGCANVAS_STYLE_STROKE "stroke" |
| #define DEBUGCANVAS_STYLE_STROKEANDFILL "strokeAndFill" |
| |
| #define DEBUGCANVAS_POINTMODE_POINTS "points" |
| #define DEBUGCANVAS_POINTMODE_LINES "lines" |
| #define DEBUGCANVAS_POINTMODE_POLYGON "polygon" |
| |
| #define DEBUGCANVAS_REGIONOP_DIFFERENCE "difference" |
| #define DEBUGCANVAS_REGIONOP_INTERSECT "intersect" |
| #define DEBUGCANVAS_REGIONOP_UNION "union" |
| #define DEBUGCANVAS_REGIONOP_XOR "xor" |
| #define DEBUGCANVAS_REGIONOP_REVERSE_DIFFERENCE "reverseDifference" |
| #define DEBUGCANVAS_REGIONOP_REPLACE "replace" |
| |
| #define DEBUGCANVAS_BLURSTYLE_NORMAL "normal" |
| #define DEBUGCANVAS_BLURSTYLE_SOLID "solid" |
| #define DEBUGCANVAS_BLURSTYLE_OUTER "outer" |
| #define DEBUGCANVAS_BLURSTYLE_INNER "inner" |
| |
| #define DEBUGCANVAS_BLURQUALITY_LOW "low" |
| #define DEBUGCANVAS_BLURQUALITY_HIGH "high" |
| |
| #define DEBUGCANVAS_FILLTYPE_WINDING "winding" |
| #define DEBUGCANVAS_FILLTYPE_EVENODD "evenOdd" |
| #define DEBUGCANVAS_FILLTYPE_INVERSEWINDING "inverseWinding" |
| #define DEBUGCANVAS_FILLTYPE_INVERSEEVENODD "inverseEvenOdd" |
| |
| #define DEBUGCANVAS_CAP_BUTT "butt" |
| #define DEBUGCANVAS_CAP_ROUND "round" |
| #define DEBUGCANVAS_CAP_SQUARE "square" |
| |
| #define DEBUGCANVAS_MITER_JOIN "miter" |
| #define DEBUGCANVAS_ROUND_JOIN "round" |
| #define DEBUGCANVAS_BEVEL_JOIN "bevel" |
| |
| #define DEBUGCANVAS_COLORTYPE_ARGB4444 "ARGB4444" |
| #define DEBUGCANVAS_COLORTYPE_RGBA8888 "RGBA8888" |
| #define DEBUGCANVAS_COLORTYPE_BGRA8888 "BGRA8888" |
| #define DEBUGCANVAS_COLORTYPE_565 "565" |
| #define DEBUGCANVAS_COLORTYPE_GRAY8 "Gray8" |
| #define DEBUGCANVAS_COLORTYPE_INDEX8 "Index8" |
| #define DEBUGCANVAS_COLORTYPE_ALPHA8 "Alpha8" |
| |
| #define DEBUGCANVAS_ALPHATYPE_OPAQUE "opaque" |
| #define DEBUGCANVAS_ALPHATYPE_PREMUL "premul" |
| #define DEBUGCANVAS_ALPHATYPE_UNPREMUL "unpremul" |
| #define DEBUGCANVAS_ALPHATYPE_UNKNOWN "unknown" |
| |
| #define DEBUGCANVAS_HINTING_NONE "none" |
| #define DEBUGCANVAS_HINTING_SLIGHT "slight" |
| #define DEBUGCANVAS_HINTING_NORMAL "normal" |
| #define DEBUGCANVAS_HINTING_FULL "full" |
| |
| #define DEBUGCANVAS_EDGING_ALIAS "alias" |
| #define DEBUGCANVAS_EDGING_ANTIALIAS "antialias" |
| #define DEBUGCANVAS_EDGING_SUBPIXELANTIALIAS "subpixelantialias" |
| |
| #define DEBUGCANVAS_SHADOWFLAG_TRANSPARENT_OCC "transparentOccluder" |
| #define DEBUGCANVAS_SHADOWFLAG_GEOMETRIC_ONLY "geometricOnly" |
| |
| static SkString* str_append(SkString* str, const SkRect& r) { |
| str->appendf(" [%g %g %g %g]", r.left(), r.top(), r.right(), r.bottom()); |
| return str; |
| } |
| |
| DrawCommand::DrawCommand(OpType type) : fOpType(type), fVisible(true) {} |
| |
| const char* DrawCommand::GetCommandString(OpType type) { |
| switch (type) { |
| case kBeginDrawPicture_OpType: return "BeginDrawPicture"; |
| case kClear_OpType: return "DrawClear"; |
| case kClipPath_OpType: return "ClipPath"; |
| case kClipRegion_OpType: return "ClipRegion"; |
| case kClipRect_OpType: return "ClipRect"; |
| case kClipRRect_OpType: return "ClipRRect"; |
| case kConcat_OpType: return "Concat"; |
| case kConcat44_OpType: return "Concat44"; |
| case kDrawAnnotation_OpType: return "DrawAnnotation"; |
| case kDrawBitmap_OpType: return "DrawBitmap"; |
| case kDrawBitmapRect_OpType: return "DrawBitmapRect"; |
| case kDrawDRRect_OpType: return "DrawDRRect"; |
| case kDrawImage_OpType: return "DrawImage"; |
| case kDrawImageLattice_OpType: return "DrawImageLattice"; |
| case kDrawImageRect_OpType: return "DrawImageRect"; |
| case kDrawImageRectLayer_OpType: return "DrawImageRectLayer"; |
| case kDrawOval_OpType: return "DrawOval"; |
| case kDrawPaint_OpType: return "DrawPaint"; |
| case kDrawPatch_OpType: return "DrawPatch"; |
| case kDrawPath_OpType: return "DrawPath"; |
| case kDrawArc_OpType: return "DrawArc"; |
| case kDrawPoints_OpType: return "DrawPoints"; |
| case kDrawRect_OpType: return "DrawRect"; |
| case kDrawRRect_OpType: return "DrawRRect"; |
| case kDrawRegion_OpType: return "DrawRegion"; |
| case kDrawShadow_OpType: return "DrawShadow"; |
| case kDrawTextBlob_OpType: return "DrawTextBlob"; |
| case kDrawVertices_OpType: return "DrawVertices"; |
| case kDrawAtlas_OpType: return "DrawAtlas"; |
| case kDrawDrawable_OpType: return "DrawDrawable"; |
| case kDrawEdgeAAQuad_OpType: return "DrawEdgeAAQuad"; |
| case kDrawEdgeAAImageSet_OpType: return "DrawEdgeAAImageSet"; |
| case kEndDrawPicture_OpType: return "EndDrawPicture"; |
| case kRestore_OpType: return "Restore"; |
| case kSave_OpType: return "Save"; |
| case kSaveLayer_OpType: return "SaveLayer"; |
| case kSetMatrix_OpType: return "SetMatrix"; |
| case kSetM44_OpType: return "SetM44"; |
| default: |
| SkDebugf("OpType error 0x%08x\n", type); |
| SkASSERT(0); |
| break; |
| } |
| SkDEBUGFAIL("DrawType UNUSED\n"); |
| return nullptr; |
| } |
| |
| void DrawCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_COMMAND, this->GetCommandString(fOpType)); |
| writer.appendBool(DEBUGCANVAS_ATTRIBUTE_VISIBLE, this->isVisible()); |
| } |
| |
| namespace { |
| |
| void xlate_and_scale_to_bounds(SkCanvas* canvas, const SkRect& bounds) { |
| const SkISize& size = canvas->getBaseLayerSize(); |
| |
| static const SkScalar kInsetFrac = 0.9f; // Leave a border around object |
| |
| canvas->translate(size.fWidth / 2.0f, size.fHeight / 2.0f); |
| if (bounds.width() > bounds.height()) { |
| canvas->scale(SkDoubleToScalar((kInsetFrac * size.fWidth) / bounds.width()), |
| SkDoubleToScalar((kInsetFrac * size.fHeight) / bounds.width())); |
| } else { |
| canvas->scale(SkDoubleToScalar((kInsetFrac * size.fWidth) / bounds.height()), |
| SkDoubleToScalar((kInsetFrac * size.fHeight) / bounds.height())); |
| } |
| canvas->translate(-bounds.centerX(), -bounds.centerY()); |
| } |
| |
| void render_path(SkCanvas* canvas, const SkPath& path) { |
| canvas->clear(0xFFFFFFFF); |
| |
| const SkRect& bounds = path.getBounds(); |
| if (bounds.isEmpty()) { |
| return; |
| } |
| |
| SkAutoCanvasRestore acr(canvas, true); |
| xlate_and_scale_to_bounds(canvas, bounds); |
| |
| SkPaint p; |
| p.setColor(SK_ColorBLACK); |
| p.setStyle(SkPaint::kStroke_Style); |
| |
| canvas->drawPath(path, p); |
| } |
| |
| void render_region(SkCanvas* canvas, const SkRegion& region) { |
| canvas->clear(0xFFFFFFFF); |
| |
| const SkIRect& bounds = region.getBounds(); |
| if (bounds.isEmpty()) { |
| return; |
| } |
| |
| SkAutoCanvasRestore acr(canvas, true); |
| xlate_and_scale_to_bounds(canvas, SkRect::Make(bounds)); |
| |
| SkPaint p; |
| p.setColor(SK_ColorBLACK); |
| p.setStyle(SkPaint::kStroke_Style); |
| |
| canvas->drawRegion(region, p); |
| } |
| |
| void render_rrect(SkCanvas* canvas, const SkRRect& rrect) { |
| canvas->clear(0xFFFFFFFF); |
| canvas->save(); |
| |
| const SkRect& bounds = rrect.getBounds(); |
| |
| xlate_and_scale_to_bounds(canvas, bounds); |
| |
| SkPaint p; |
| p.setColor(SK_ColorBLACK); |
| p.setStyle(SkPaint::kStroke_Style); |
| |
| canvas->drawRRect(rrect, p); |
| canvas->restore(); |
| } |
| |
| void render_drrect(SkCanvas* canvas, const SkRRect& outer, const SkRRect& inner) { |
| canvas->clear(0xFFFFFFFF); |
| canvas->save(); |
| |
| const SkRect& bounds = outer.getBounds(); |
| |
| xlate_and_scale_to_bounds(canvas, bounds); |
| |
| SkPaint p; |
| p.setColor(SK_ColorBLACK); |
| p.setStyle(SkPaint::kStroke_Style); |
| |
| canvas->drawDRRect(outer, inner, p); |
| canvas->restore(); |
| } |
| |
| void render_shadow(SkCanvas* canvas, const SkPath& path, SkDrawShadowRec rec) { |
| canvas->clear(0xFFFFFFFF); |
| |
| const SkRect& bounds = path.getBounds(); |
| if (bounds.isEmpty()) { |
| return; |
| } |
| |
| SkAutoCanvasRestore acr(canvas, true); |
| xlate_and_scale_to_bounds(canvas, bounds); |
| |
| rec.fAmbientColor = SK_ColorBLACK; |
| rec.fSpotColor = SK_ColorBLACK; |
| canvas->private_draw_shadow_rec(path, rec); |
| } |
| |
| static const char* const gBlendModeMap[] = { |
| "clear", "src", "dst", "srcOver", "dstOver", "srcIn", "dstIn", |
| "srcOut", "dstOut", "srcATop", "dstATop", "xor", "plus", "modulate", |
| |
| "screen", |
| |
| "overlay", "darken", "lighten", "colorDodge", "colorBurn", "hardLight", "softLight", |
| "difference", "exclusion", "multiply", |
| |
| "hue", "saturation", "color", "luminosity", |
| }; |
| |
| static_assert(SK_ARRAY_COUNT(gBlendModeMap) == static_cast<size_t>(SkBlendMode::kLastMode) + 1, |
| "blendMode mismatch"); |
| static_assert(SK_ARRAY_COUNT(gBlendModeMap) == static_cast<size_t>(SkBlendMode::kLuminosity) + 1, |
| "blendMode mismatch"); |
| |
| void apply_paint_blend_mode(const SkPaint& paint, SkJSONWriter& writer) { |
| const auto mode = paint.getBlendMode(); |
| if (mode != SkBlendMode::kSrcOver) { |
| SkASSERT(static_cast<size_t>(mode) < SK_ARRAY_COUNT(gBlendModeMap)); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_BLENDMODE, |
| gBlendModeMap[static_cast<size_t>(mode)]); |
| } |
| } |
| |
| }; // namespace |
| |
| void DrawCommand::MakeJsonColor(SkJSONWriter& writer, const SkColor color) { |
| writer.beginArray(nullptr, false); |
| writer.appendS32(SkColorGetA(color)); |
| writer.appendS32(SkColorGetR(color)); |
| writer.appendS32(SkColorGetG(color)); |
| writer.appendS32(SkColorGetB(color)); |
| writer.endArray(); |
| } |
| |
| void DrawCommand::MakeJsonColor4f(SkJSONWriter& writer, const SkColor4f& color) { |
| writer.beginArray(nullptr, false); |
| writer.appendFloat(color.fA); |
| writer.appendFloat(color.fR); |
| writer.appendFloat(color.fG); |
| writer.appendFloat(color.fB); |
| writer.endArray(); |
| } |
| |
| void DrawCommand::MakeJsonPoint(SkJSONWriter& writer, const SkPoint& point) { |
| writer.beginArray(nullptr, false); |
| writer.appendFloat(point.x()); |
| writer.appendFloat(point.y()); |
| writer.endArray(); |
| } |
| |
| void DrawCommand::MakeJsonPoint(SkJSONWriter& writer, SkScalar x, SkScalar y) { |
| writer.beginArray(nullptr, false); |
| writer.appendFloat(x); |
| writer.appendFloat(y); |
| writer.endArray(); |
| } |
| |
| void DrawCommand::MakeJsonPoint3(SkJSONWriter& writer, const SkPoint3& point) { |
| writer.beginArray(nullptr, false); |
| writer.appendFloat(point.x()); |
| writer.appendFloat(point.y()); |
| writer.appendFloat(point.z()); |
| writer.endArray(); |
| } |
| |
| void DrawCommand::MakeJsonRect(SkJSONWriter& writer, const SkRect& rect) { |
| writer.beginArray(nullptr, false); |
| writer.appendFloat(rect.left()); |
| writer.appendFloat(rect.top()); |
| writer.appendFloat(rect.right()); |
| writer.appendFloat(rect.bottom()); |
| writer.endArray(); |
| } |
| |
| void DrawCommand::MakeJsonIRect(SkJSONWriter& writer, const SkIRect& rect) { |
| writer.beginArray(nullptr, false); |
| writer.appendS32(rect.left()); |
| writer.appendS32(rect.top()); |
| writer.appendS32(rect.right()); |
| writer.appendS32(rect.bottom()); |
| writer.endArray(); |
| } |
| |
| static void make_json_rrect(SkJSONWriter& writer, const SkRRect& rrect) { |
| writer.beginArray(nullptr, false); |
| DrawCommand::MakeJsonRect(writer, rrect.rect()); |
| DrawCommand::MakeJsonPoint(writer, rrect.radii(SkRRect::kUpperLeft_Corner)); |
| DrawCommand::MakeJsonPoint(writer, rrect.radii(SkRRect::kUpperRight_Corner)); |
| DrawCommand::MakeJsonPoint(writer, rrect.radii(SkRRect::kLowerRight_Corner)); |
| DrawCommand::MakeJsonPoint(writer, rrect.radii(SkRRect::kLowerLeft_Corner)); |
| writer.endArray(); |
| } |
| |
| void DrawCommand::MakeJsonMatrix(SkJSONWriter& writer, const SkMatrix& matrix) { |
| writer.beginArray(); |
| for (int r = 0; r < 3; ++r) { |
| writer.beginArray(nullptr, false); |
| for (int c = 0; c < 3; ++c) { |
| writer.appendFloat(matrix[r * 3 + c]); |
| } |
| writer.endArray(); |
| } |
| writer.endArray(); |
| } |
| |
| void DrawCommand::MakeJsonMatrix44(SkJSONWriter& writer, const SkM44& matrix) { |
| writer.beginArray(); |
| for (int r = 0; r < 4; ++r) { |
| writer.beginArray(nullptr, false); |
| for (int c = 0; c < 4; ++c) { |
| writer.appendFloat(matrix.rc(r, c)); |
| } |
| writer.endArray(); |
| } |
| writer.endArray(); |
| } |
| |
| void DrawCommand::MakeJsonPath(SkJSONWriter& writer, const SkPath& path) { |
| writer.beginObject(); |
| switch (path.getFillType()) { |
| case SkPathFillType::kWinding: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_FILLTYPE, DEBUGCANVAS_FILLTYPE_WINDING); |
| break; |
| case SkPathFillType::kEvenOdd: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_FILLTYPE, DEBUGCANVAS_FILLTYPE_EVENODD); |
| break; |
| case SkPathFillType::kInverseWinding: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_FILLTYPE, |
| DEBUGCANVAS_FILLTYPE_INVERSEWINDING); |
| break; |
| case SkPathFillType::kInverseEvenOdd: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_FILLTYPE, |
| DEBUGCANVAS_FILLTYPE_INVERSEEVENODD); |
| break; |
| } |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_VERBS); |
| SkPath::Iter iter(path, false); |
| SkPoint pts[4]; |
| SkPath::Verb verb; |
| while ((verb = iter.next(pts)) != SkPath::kDone_Verb) { |
| if (verb == SkPath::kClose_Verb) { |
| writer.appendString(DEBUGCANVAS_VERB_CLOSE); |
| continue; |
| } |
| writer.beginObject(); // verb |
| switch (verb) { |
| case SkPath::kLine_Verb: { |
| writer.appendName(DEBUGCANVAS_VERB_LINE); |
| MakeJsonPoint(writer, pts[1]); |
| break; |
| } |
| case SkPath::kQuad_Verb: { |
| writer.beginArray(DEBUGCANVAS_VERB_QUAD); |
| MakeJsonPoint(writer, pts[1]); |
| MakeJsonPoint(writer, pts[2]); |
| writer.endArray(); // quad coords |
| break; |
| } |
| case SkPath::kCubic_Verb: { |
| writer.beginArray(DEBUGCANVAS_VERB_CUBIC); |
| MakeJsonPoint(writer, pts[1]); |
| MakeJsonPoint(writer, pts[2]); |
| MakeJsonPoint(writer, pts[3]); |
| writer.endArray(); // cubic coords |
| break; |
| } |
| case SkPath::kConic_Verb: { |
| writer.beginArray(DEBUGCANVAS_VERB_CONIC); |
| MakeJsonPoint(writer, pts[1]); |
| MakeJsonPoint(writer, pts[2]); |
| writer.appendFloat(iter.conicWeight()); |
| writer.endArray(); // conic coords |
| break; |
| } |
| case SkPath::kMove_Verb: { |
| writer.appendName(DEBUGCANVAS_VERB_MOVE); |
| MakeJsonPoint(writer, pts[0]); |
| break; |
| } |
| case SkPath::kClose_Verb: |
| case SkPath::kDone_Verb: |
| // Unreachable |
| break; |
| } |
| writer.endObject(); // verb |
| } |
| writer.endArray(); // verbs |
| writer.endObject(); // path |
| } |
| |
| void DrawCommand::MakeJsonRegion(SkJSONWriter& writer, const SkRegion& region) { |
| // TODO: Actually serialize the rectangles, rather than just devolving to path |
| SkPath path; |
| region.getBoundaryPath(&path); |
| MakeJsonPath(writer, path); |
| } |
| |
| static const char* regionop_name(SkClipOp op) { |
| switch (op) { |
| case kDifference_SkClipOp: return DEBUGCANVAS_REGIONOP_DIFFERENCE; |
| case kIntersect_SkClipOp: return DEBUGCANVAS_REGIONOP_INTERSECT; |
| case kUnion_SkClipOp: return DEBUGCANVAS_REGIONOP_UNION; |
| case kXOR_SkClipOp: return DEBUGCANVAS_REGIONOP_XOR; |
| case kReverseDifference_SkClipOp: return DEBUGCANVAS_REGIONOP_REVERSE_DIFFERENCE; |
| case kReplace_SkClipOp: return DEBUGCANVAS_REGIONOP_REPLACE; |
| default: SkASSERT(false); return "<invalid region op>"; |
| } |
| } |
| |
| static const char* pointmode_name(SkCanvas::PointMode mode) { |
| switch (mode) { |
| case SkCanvas::kPoints_PointMode: return DEBUGCANVAS_POINTMODE_POINTS; |
| case SkCanvas::kLines_PointMode: return DEBUGCANVAS_POINTMODE_LINES; |
| case SkCanvas::kPolygon_PointMode: return DEBUGCANVAS_POINTMODE_POLYGON; |
| default: SkASSERT(false); return "<invalid point mode>"; |
| } |
| } |
| |
| static void store_scalar(SkJSONWriter& writer, |
| const char* key, |
| SkScalar value, |
| SkScalar defaultValue) { |
| if (value != defaultValue) { |
| writer.appendFloat(key, value); |
| } |
| } |
| |
| static void store_bool(SkJSONWriter& writer, const char* key, bool value, bool defaultValue) { |
| if (value != defaultValue) { |
| writer.appendBool(key, value); |
| } |
| } |
| |
| static SkString encode_data(const void* bytes, |
| size_t count, |
| const char* contentType, |
| UrlDataManager& urlDataManager) { |
| sk_sp<SkData> data(SkData::MakeWithCopy(bytes, count)); |
| return urlDataManager.addData(data.get(), contentType); |
| } |
| |
| void DrawCommand::flatten(const SkFlattenable* flattenable, |
| SkJSONWriter& writer, |
| UrlDataManager& urlDataManager) { |
| SkBinaryWriteBuffer buffer; |
| flattenable->flatten(buffer); |
| void* data = sk_malloc_throw(buffer.bytesWritten()); |
| buffer.writeToMemory(data); |
| SkString url = |
| encode_data(data, buffer.bytesWritten(), "application/octet-stream", urlDataManager); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_NAME, flattenable->getTypeName()); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_DATA, url.c_str()); |
| |
| writer.beginObject(DEBUGCANVAS_ATTRIBUTE_VALUES); |
| JsonWriteBuffer jsonBuffer(&writer, &urlDataManager); |
| flattenable->flatten(jsonBuffer); |
| writer.endObject(); // values |
| |
| sk_free(data); |
| } |
| |
| void DrawCommand::WritePNG(SkBitmap bitmap, SkWStream& out) { |
| SkPixmap pm; |
| SkAssertResult(bitmap.peekPixels(&pm)); |
| |
| SkPngEncoder::Options options; |
| options.fZLibLevel = 1; |
| options.fFilterFlags = SkPngEncoder::FilterFlag::kNone; |
| SkPngEncoder::Encode(&out, pm, options); |
| } |
| |
| // flattens an image to a Json stream, also called from shader flatten |
| bool DrawCommand::flatten(const SkImage& image, |
| SkJSONWriter& writer, |
| UrlDataManager& urlDataManager) { |
| // For MSKP files, there is no need to encode the image, |
| // just report its id. |
| if (urlDataManager.hasImageIndex()) { |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_IMAGE_INDEX); |
| writer.appendU64(urlDataManager.lookupImage(&image)); |
| return true; |
| } |
| |
| writer.beginObject(DEBUGCANVAS_ATTRIBUTE_IMAGE); |
| size_t rowBytes = 4 * image.width(); |
| SkAutoMalloc buffer(rowBytes * image.height()); |
| SkImageInfo dstInfo = |
| SkImageInfo::Make(image.dimensions(), kN32_SkColorType, kPremul_SkAlphaType); |
| // "cheat" for this debug tool and use image's context |
| GrDirectContext* dContext = nullptr; |
| #ifdef SK_SUPPORT_GPU |
| dContext = GrAsDirectContext(as_IB(&image)->context()); |
| #endif |
| if (!image.readPixels(dContext, dstInfo, buffer.get(), rowBytes, 0, 0)) { |
| SkDebugf("DrawCommand::flatten SkImage: readPixels failed\n"); |
| writer.endObject(); |
| return false; |
| } |
| |
| SkBitmap bm; |
| bm.installPixels(dstInfo, buffer.get(), rowBytes); |
| |
| SkDynamicMemoryWStream out; |
| DrawCommand::WritePNG(bm, out); |
| sk_sp<SkData> encoded = out.detachAsData(); |
| if (encoded == nullptr) { |
| SkDebugf("DrawCommand::flatten SkImage: could not encode image as PNG\n"); |
| writer.endObject(); |
| return false; |
| } |
| auto dataPtr = encoded->data(); |
| if (!dataPtr) { |
| SkDebugf("DrawCommand::flatten SkImage: encoding as PNG produced zero length data\n"); |
| writer.endObject(); |
| return false; |
| } |
| SkString url = encode_data(encoded->data(), encoded->size(), "image/png", urlDataManager); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_DATA, url.c_str()); |
| writer.endObject(); |
| return true; |
| } |
| |
| static const char* color_type_name(SkColorType colorType) { |
| switch (colorType) { |
| case kARGB_4444_SkColorType: return DEBUGCANVAS_COLORTYPE_ARGB4444; |
| case kRGBA_8888_SkColorType: return DEBUGCANVAS_COLORTYPE_RGBA8888; |
| case kBGRA_8888_SkColorType: return DEBUGCANVAS_COLORTYPE_BGRA8888; |
| case kRGB_565_SkColorType: return DEBUGCANVAS_COLORTYPE_565; |
| case kGray_8_SkColorType: return DEBUGCANVAS_COLORTYPE_GRAY8; |
| case kAlpha_8_SkColorType: return DEBUGCANVAS_COLORTYPE_ALPHA8; |
| default: SkASSERT(false); return DEBUGCANVAS_COLORTYPE_RGBA8888; |
| } |
| } |
| |
| static const char* alpha_type_name(SkAlphaType alphaType) { |
| switch (alphaType) { |
| case kOpaque_SkAlphaType: return DEBUGCANVAS_ALPHATYPE_OPAQUE; |
| case kPremul_SkAlphaType: return DEBUGCANVAS_ALPHATYPE_PREMUL; |
| case kUnpremul_SkAlphaType: return DEBUGCANVAS_ALPHATYPE_UNPREMUL; |
| default: SkASSERT(false); return DEBUGCANVAS_ALPHATYPE_OPAQUE; |
| } |
| } |
| |
| bool DrawCommand::flatten(const SkBitmap& bitmap, |
| SkJSONWriter& writer, |
| UrlDataManager& urlDataManager) { |
| sk_sp<SkImage> image(bitmap.asImage()); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_COLOR, color_type_name(bitmap.colorType())); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_ALPHA, alpha_type_name(bitmap.alphaType())); |
| // Image will appear to have no uses, TODO(nifong): provide the user with a useful explanation |
| bool success = flatten(*image, writer, urlDataManager); |
| return success; |
| } |
| |
| static void apply_font_hinting(const SkFont& font, SkJSONWriter& writer) { |
| SkFontHinting hinting = font.getHinting(); |
| if (hinting != SkPaintDefaults_Hinting) { |
| switch (hinting) { |
| case SkFontHinting::kNone: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_HINTING, DEBUGCANVAS_HINTING_NONE); |
| break; |
| case SkFontHinting::kSlight: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_HINTING, DEBUGCANVAS_HINTING_SLIGHT); |
| break; |
| case SkFontHinting::kNormal: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_HINTING, DEBUGCANVAS_HINTING_NORMAL); |
| break; |
| case SkFontHinting::kFull: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_HINTING, DEBUGCANVAS_HINTING_FULL); |
| break; |
| } |
| } |
| } |
| |
| static void apply_font_edging(const SkFont& font, SkJSONWriter& writer) { |
| switch (font.getEdging()) { |
| case SkFont::Edging::kAlias: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_EDGING, DEBUGCANVAS_EDGING_ALIAS); |
| break; |
| case SkFont::Edging::kAntiAlias: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_EDGING, DEBUGCANVAS_EDGING_ANTIALIAS); |
| break; |
| case SkFont::Edging::kSubpixelAntiAlias: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_EDGING, DEBUGCANVAS_EDGING_SUBPIXELANTIALIAS); |
| break; |
| } |
| } |
| |
| static void apply_paint_color(const SkPaint& paint, SkJSONWriter& writer) { |
| SkColor color = paint.getColor(); |
| if (color != SK_ColorBLACK) { |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COLOR); |
| DrawCommand::MakeJsonColor(writer, color); |
| } |
| } |
| |
| static void apply_paint_style(const SkPaint& paint, SkJSONWriter& writer) { |
| SkPaint::Style style = paint.getStyle(); |
| if (style != SkPaint::kFill_Style) { |
| switch (style) { |
| case SkPaint::kStroke_Style: { |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_STYLE, DEBUGCANVAS_STYLE_STROKE); |
| break; |
| } |
| case SkPaint::kStrokeAndFill_Style: { |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_STYLE, DEBUGCANVAS_STYLE_STROKEANDFILL); |
| break; |
| } |
| default: SkASSERT(false); |
| } |
| } |
| } |
| |
| static void apply_paint_cap(const SkPaint& paint, SkJSONWriter& writer) { |
| SkPaint::Cap cap = paint.getStrokeCap(); |
| if (cap != SkPaint::kDefault_Cap) { |
| switch (cap) { |
| case SkPaint::kButt_Cap: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_CAP, DEBUGCANVAS_CAP_BUTT); |
| break; |
| case SkPaint::kRound_Cap: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_CAP, DEBUGCANVAS_CAP_ROUND); |
| break; |
| case SkPaint::kSquare_Cap: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_CAP, DEBUGCANVAS_CAP_SQUARE); |
| break; |
| default: SkASSERT(false); |
| } |
| } |
| } |
| |
| static void apply_paint_join(const SkPaint& paint, SkJSONWriter& writer) { |
| SkPaint::Join join = paint.getStrokeJoin(); |
| if (join != SkPaint::kDefault_Join) { |
| switch (join) { |
| case SkPaint::kMiter_Join: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_STROKEJOIN, DEBUGCANVAS_MITER_JOIN); |
| break; |
| case SkPaint::kRound_Join: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_STROKEJOIN, DEBUGCANVAS_ROUND_JOIN); |
| break; |
| case SkPaint::kBevel_Join: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_STROKEJOIN, DEBUGCANVAS_BEVEL_JOIN); |
| break; |
| default: SkASSERT(false); |
| } |
| } |
| } |
| |
| static void apply_paint_maskfilter(const SkPaint& paint, |
| SkJSONWriter& writer, |
| UrlDataManager& urlDataManager) { |
| SkMaskFilter* maskFilter = paint.getMaskFilter(); |
| if (maskFilter != nullptr) { |
| SkMaskFilterBase::BlurRec blurRec; |
| if (as_MFB(maskFilter)->asABlur(&blurRec)) { |
| writer.beginObject(DEBUGCANVAS_ATTRIBUTE_BLUR); |
| writer.appendFloat(DEBUGCANVAS_ATTRIBUTE_SIGMA, blurRec.fSigma); |
| switch (blurRec.fStyle) { |
| case SkBlurStyle::kNormal_SkBlurStyle: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_STYLE, DEBUGCANVAS_BLURSTYLE_NORMAL); |
| break; |
| case SkBlurStyle::kSolid_SkBlurStyle: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_STYLE, DEBUGCANVAS_BLURSTYLE_SOLID); |
| break; |
| case SkBlurStyle::kOuter_SkBlurStyle: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_STYLE, DEBUGCANVAS_BLURSTYLE_OUTER); |
| break; |
| case SkBlurStyle::kInner_SkBlurStyle: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_STYLE, DEBUGCANVAS_BLURSTYLE_INNER); |
| break; |
| default: SkASSERT(false); |
| } |
| writer.endObject(); // blur |
| } else { |
| writer.beginObject(DEBUGCANVAS_ATTRIBUTE_MASKFILTER); |
| DrawCommand::flatten(maskFilter, writer, urlDataManager); |
| writer.endObject(); // maskFilter |
| } |
| } |
| } |
| |
| static void apply_paint_patheffect(const SkPaint& paint, |
| SkJSONWriter& writer, |
| UrlDataManager& urlDataManager) { |
| SkPathEffect* pathEffect = paint.getPathEffect(); |
| if (pathEffect != nullptr) { |
| SkPathEffect::DashInfo dashInfo; |
| SkPathEffect::DashType dashType = pathEffect->asADash(&dashInfo); |
| if (dashType == SkPathEffect::kDash_DashType) { |
| dashInfo.fIntervals = (SkScalar*)sk_malloc_throw(dashInfo.fCount * sizeof(SkScalar)); |
| pathEffect->asADash(&dashInfo); |
| writer.beginObject(DEBUGCANVAS_ATTRIBUTE_DASHING); |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_INTERVALS, false); |
| for (int32_t i = 0; i < dashInfo.fCount; i++) { |
| writer.appendFloat(dashInfo.fIntervals[i]); |
| } |
| writer.endArray(); // intervals |
| sk_free(dashInfo.fIntervals); |
| writer.appendFloat(DEBUGCANVAS_ATTRIBUTE_PHASE, dashInfo.fPhase); |
| writer.endObject(); // dashing |
| } else { |
| writer.beginObject(DEBUGCANVAS_ATTRIBUTE_PATHEFFECT); |
| DrawCommand::flatten(pathEffect, writer, urlDataManager); |
| writer.endObject(); // pathEffect |
| } |
| } |
| } |
| |
| static void apply_font_typeface(const SkFont& font, |
| SkJSONWriter& writer, |
| UrlDataManager& urlDataManager) { |
| SkTypeface* typeface = font.getTypefaceOrDefault(); |
| if (typeface != nullptr) { |
| writer.beginObject(DEBUGCANVAS_ATTRIBUTE_TYPEFACE); |
| SkDynamicMemoryWStream buffer; |
| typeface->serialize(&buffer); |
| void* data = sk_malloc_throw(buffer.bytesWritten()); |
| buffer.copyTo(data); |
| SkString url = encode_data( |
| data, buffer.bytesWritten(), "application/octet-stream", urlDataManager); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_DATA, url.c_str()); |
| sk_free(data); |
| writer.endObject(); |
| } |
| } |
| |
| static void apply_flattenable(const char* key, |
| SkFlattenable* flattenable, |
| SkJSONWriter& writer, |
| UrlDataManager& urlDataManager) { |
| if (flattenable != nullptr) { |
| writer.beginObject(key); |
| DrawCommand::flatten(flattenable, writer, urlDataManager); |
| writer.endObject(); |
| } |
| } |
| |
| void DrawCommand::MakeJsonPaint(SkJSONWriter& writer, |
| const SkPaint& paint, |
| UrlDataManager& urlDataManager) { |
| writer.beginObject(); |
| store_scalar(writer, DEBUGCANVAS_ATTRIBUTE_STROKEWIDTH, paint.getStrokeWidth(), 0.0f); |
| store_scalar(writer, |
| DEBUGCANVAS_ATTRIBUTE_STROKEMITER, |
| paint.getStrokeMiter(), |
| SkPaintDefaults_MiterLimit); |
| store_bool(writer, DEBUGCANVAS_ATTRIBUTE_ANTIALIAS, paint.isAntiAlias(), false); |
| store_bool(writer, DEBUGCANVAS_ATTRIBUTE_DITHER, paint.isDither(), false); |
| |
| apply_paint_color(paint, writer); |
| apply_paint_style(paint, writer); |
| apply_paint_blend_mode(paint, writer); |
| apply_paint_cap(paint, writer); |
| apply_paint_join(paint, writer); |
| apply_paint_patheffect(paint, writer, urlDataManager); |
| apply_paint_maskfilter(paint, writer, urlDataManager); |
| apply_flattenable(DEBUGCANVAS_ATTRIBUTE_SHADER, paint.getShader(), writer, urlDataManager); |
| apply_flattenable( |
| DEBUGCANVAS_ATTRIBUTE_IMAGEFILTER, paint.getImageFilter(), writer, urlDataManager); |
| apply_flattenable( |
| DEBUGCANVAS_ATTRIBUTE_COLORFILTER, paint.getColorFilter(), writer, urlDataManager); |
| writer.endObject(); // paint |
| } |
| |
| static void MakeJsonFont(const SkFont& font, SkJSONWriter& writer, UrlDataManager& urlDataManager) { |
| writer.beginObject(); |
| store_bool(writer, DEBUGCANVAS_ATTRIBUTE_FAKEBOLDTEXT, font.isEmbolden(), false); |
| store_bool(writer, DEBUGCANVAS_ATTRIBUTE_LINEARTEXT, font.isLinearMetrics(), false); |
| store_bool(writer, DEBUGCANVAS_ATTRIBUTE_SUBPIXELTEXT, font.isSubpixel(), false); |
| store_bool(writer, DEBUGCANVAS_ATTRIBUTE_EMBEDDEDBITMAPTEXT, font.isEmbeddedBitmaps(), false); |
| store_bool(writer, DEBUGCANVAS_ATTRIBUTE_AUTOHINTING, font.isForceAutoHinting(), false); |
| |
| store_scalar(writer, DEBUGCANVAS_ATTRIBUTE_TEXTSIZE, font.getSize(), SkPaintDefaults_TextSize); |
| store_scalar(writer, DEBUGCANVAS_ATTRIBUTE_TEXTSCALEX, font.getScaleX(), SK_Scalar1); |
| store_scalar(writer, DEBUGCANVAS_ATTRIBUTE_TEXTSCALEX, font.getSkewX(), 0.0f); |
| apply_font_edging(font, writer); |
| apply_font_hinting(font, writer); |
| apply_font_typeface(font, writer, urlDataManager); |
| writer.endObject(); // font |
| } |
| |
| void DrawCommand::MakeJsonLattice(SkJSONWriter& writer, const SkCanvas::Lattice& lattice) { |
| writer.beginObject(); |
| writer.appendS32(DEBUGCANVAS_ATTRIBUTE_LATTICEXCOUNT, lattice.fXCount); |
| writer.appendS32(DEBUGCANVAS_ATTRIBUTE_LATTICEYCOUNT, lattice.fYCount); |
| if (nullptr != lattice.fBounds) { |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_BOUNDS); |
| MakeJsonIRect(writer, *lattice.fBounds); |
| } |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_LATTICEXDIVS); |
| for (int i = 0; i < lattice.fXCount; i++) { |
| writer.appendS32(lattice.fXDivs[i]); |
| } |
| writer.endArray(); // xdivs |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_LATTICEYDIVS); |
| for (int i = 0; i < lattice.fYCount; i++) { |
| writer.appendS32(lattice.fYDivs[i]); |
| } |
| writer.endArray(); // ydivs |
| if (nullptr != lattice.fRectTypes) { |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_LATTICEFLAGS); |
| int flagCount = 0; |
| for (int row = 0; row < lattice.fYCount + 1; row++) { |
| writer.beginArray(); |
| for (int column = 0; column < lattice.fXCount + 1; column++) { |
| writer.appendS32(lattice.fRectTypes[flagCount++]); |
| } |
| writer.endArray(); // row |
| } |
| writer.endArray(); |
| } |
| writer.endObject(); |
| } |
| |
| ClearCommand::ClearCommand(SkColor color) : INHERITED(kClear_OpType) { fColor = color; } |
| |
| void ClearCommand::execute(SkCanvas* canvas) const { canvas->clear(fColor); } |
| |
| void ClearCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COLOR); |
| MakeJsonColor(writer, fColor); |
| } |
| |
| ClipPathCommand::ClipPathCommand(const SkPath& path, SkClipOp op, bool doAA) |
| : INHERITED(kClipPath_OpType) { |
| fPath = path; |
| fOp = op; |
| fDoAA = doAA; |
| } |
| |
| void ClipPathCommand::execute(SkCanvas* canvas) const { canvas->clipPath(fPath, fOp, fDoAA); } |
| |
| bool ClipPathCommand::render(SkCanvas* canvas) const { |
| render_path(canvas, fPath); |
| return true; |
| } |
| |
| void ClipPathCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PATH); |
| MakeJsonPath(writer, fPath); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_REGIONOP, regionop_name(fOp)); |
| writer.appendBool(DEBUGCANVAS_ATTRIBUTE_ANTIALIAS, fDoAA); |
| } |
| |
| ClipRegionCommand::ClipRegionCommand(const SkRegion& region, SkClipOp op) |
| : INHERITED(kClipRegion_OpType) { |
| fRegion = region; |
| fOp = op; |
| } |
| |
| void ClipRegionCommand::execute(SkCanvas* canvas) const { canvas->clipRegion(fRegion, fOp); } |
| |
| void ClipRegionCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_REGION); |
| MakeJsonRegion(writer, fRegion); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_REGIONOP, regionop_name(fOp)); |
| } |
| |
| ClipRectCommand::ClipRectCommand(const SkRect& rect, SkClipOp op, bool doAA) |
| : INHERITED(kClipRect_OpType) { |
| fRect = rect; |
| fOp = op; |
| fDoAA = doAA; |
| } |
| |
| void ClipRectCommand::execute(SkCanvas* canvas) const { canvas->clipRect(fRect, fOp, fDoAA); } |
| |
| void ClipRectCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COORDS); |
| MakeJsonRect(writer, fRect); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_REGIONOP, regionop_name(fOp)); |
| writer.appendBool(DEBUGCANVAS_ATTRIBUTE_ANTIALIAS, fDoAA); |
| |
| SkString desc; |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, str_append(&desc, fRect)->c_str()); |
| } |
| |
| ClipRRectCommand::ClipRRectCommand(const SkRRect& rrect, SkClipOp op, bool doAA) |
| : INHERITED(kClipRRect_OpType) { |
| fRRect = rrect; |
| fOp = op; |
| fDoAA = doAA; |
| } |
| |
| void ClipRRectCommand::execute(SkCanvas* canvas) const { canvas->clipRRect(fRRect, fOp, fDoAA); } |
| |
| bool ClipRRectCommand::render(SkCanvas* canvas) const { |
| render_rrect(canvas, fRRect); |
| return true; |
| } |
| |
| void ClipRRectCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COORDS); |
| make_json_rrect(writer, fRRect); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_REGIONOP, regionop_name(fOp)); |
| writer.appendBool(DEBUGCANVAS_ATTRIBUTE_ANTIALIAS, fDoAA); |
| } |
| |
| ClipShaderCommand::ClipShaderCommand(sk_sp<SkShader> cs, SkClipOp op) |
| : INHERITED(kClipShader_OpType) { |
| fShader = cs; |
| fOp = op; |
| } |
| |
| void ClipShaderCommand::execute(SkCanvas* canvas) const { canvas->clipShader(fShader, fOp); } |
| |
| bool ClipShaderCommand::render(SkCanvas* canvas) const { |
| SkPaint paint; |
| paint.setShader(fShader); |
| canvas->drawPaint(paint); |
| return true; |
| } |
| |
| void ClipShaderCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| apply_flattenable(DEBUGCANVAS_ATTRIBUTE_SHADER, fShader.get(), writer, urlDataManager); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_REGIONOP, regionop_name(fOp)); |
| } |
| |
| ConcatCommand::ConcatCommand(const SkMatrix& matrix) : INHERITED(kConcat_OpType) { |
| fMatrix = matrix; |
| } |
| |
| void ConcatCommand::execute(SkCanvas* canvas) const { canvas->concat(fMatrix); } |
| |
| namespace { |
| void writeMatrixType(SkJSONWriter& writer, const SkMatrix& m) { |
| switch (m.getType()) { |
| case SkMatrix::kTranslate_Mask: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, " (translate)"); |
| break; |
| case SkMatrix::kScale_Mask: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, " (scale)"); |
| break; |
| case SkMatrix::kAffine_Mask: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, " (rotation or skew)"); |
| break; |
| case SkMatrix::kPerspective_Mask: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, " (perspective)"); |
| break; |
| default: |
| break; |
| } |
| } |
| } |
| |
| void ConcatCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_MATRIX); |
| MakeJsonMatrix(writer, fMatrix); |
| writeMatrixType(writer, fMatrix); |
| } |
| |
| Concat44Command::Concat44Command(const SkM44& matrix) : INHERITED(kConcat44_OpType) { |
| fMatrix = matrix; |
| } |
| |
| void Concat44Command::execute(SkCanvas* canvas) const { canvas->concat(fMatrix); } |
| |
| void Concat44Command::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_MATRIX); |
| MakeJsonMatrix44(writer, fMatrix); |
| } |
| |
| //// |
| |
| DrawAnnotationCommand::DrawAnnotationCommand(const SkRect& rect, |
| const char key[], |
| sk_sp<SkData> value) |
| : INHERITED(kDrawAnnotation_OpType), fRect(rect), fKey(key), fValue(std::move(value)) {} |
| |
| void DrawAnnotationCommand::execute(SkCanvas* canvas) const { |
| canvas->drawAnnotation(fRect, fKey.c_str(), fValue); |
| } |
| |
| void DrawAnnotationCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COORDS); |
| MakeJsonRect(writer, fRect); |
| writer.appendString("key", fKey.c_str()); |
| if (fValue) { |
| writer.appendString("value", std::string( |
| static_cast<const char*>(fValue->data()), fValue->size() |
| ).c_str()); |
| } |
| |
| SkString desc; |
| str_append(&desc, fRect)->appendf(" %s", fKey.c_str()); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, desc.c_str()); |
| } |
| |
| //// |
| |
| DrawImageCommand::DrawImageCommand(const SkImage* image, |
| SkScalar left, |
| SkScalar top, |
| const SkSamplingOptions& sampling, |
| const SkPaint* paint) |
| : INHERITED(kDrawImage_OpType) |
| , fImage(SkRef(image)) |
| , fLeft(left) |
| , fTop(top) |
| , fSampling(sampling) |
| , fPaint(paint) {} |
| |
| void DrawImageCommand::execute(SkCanvas* canvas) const { |
| canvas->drawImage(fImage.get(), fLeft, fTop, fSampling, fPaint.getMaybeNull()); |
| } |
| |
| bool DrawImageCommand::render(SkCanvas* canvas) const { |
| SkAutoCanvasRestore acr(canvas, true); |
| canvas->clear(0xFFFFFFFF); |
| |
| xlate_and_scale_to_bounds( |
| canvas, |
| SkRect::MakeXYWH( |
| fLeft, fTop, SkIntToScalar(fImage->width()), SkIntToScalar(fImage->height()))); |
| this->execute(canvas); |
| return true; |
| } |
| |
| uint64_t DrawImageCommand::imageId(UrlDataManager& udm) const { |
| return udm.lookupImage(fImage.get()); |
| } |
| |
| void DrawImageCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| flatten(*fImage, writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COORDS); |
| MakeJsonPoint(writer, fLeft, fTop); |
| if (fPaint.isValid()) { |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, *fPaint, urlDataManager); |
| } |
| |
| writer.appendU32(DEBUGCANVAS_ATTRIBUTE_UNIQUE_ID, fImage->uniqueID()); |
| writer.appendS32(DEBUGCANVAS_ATTRIBUTE_WIDTH, fImage->width()); |
| writer.appendS32(DEBUGCANVAS_ATTRIBUTE_HEIGHT, fImage->height()); |
| switch (fImage->alphaType()) { |
| case kOpaque_SkAlphaType: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_ALPHA, DEBUGCANVAS_ALPHATYPE_OPAQUE); |
| break; |
| case kPremul_SkAlphaType: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_ALPHA, DEBUGCANVAS_ALPHATYPE_PREMUL); |
| break; |
| case kUnpremul_SkAlphaType: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_ALPHA, DEBUGCANVAS_ALPHATYPE_UNPREMUL); |
| break; |
| default: |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_ALPHA, DEBUGCANVAS_ALPHATYPE_UNKNOWN); |
| break; |
| } |
| } |
| |
| DrawImageLatticeCommand::DrawImageLatticeCommand(const SkImage* image, |
| const SkCanvas::Lattice& lattice, |
| const SkRect& dst, |
| SkFilterMode filter, |
| const SkPaint* paint) |
| : INHERITED(kDrawImageLattice_OpType) |
| , fImage(SkRef(image)) |
| , fLattice(lattice) |
| , fDst(dst) |
| , fFilter(filter) |
| , fPaint(paint) {} |
| |
| void DrawImageLatticeCommand::execute(SkCanvas* canvas) const { |
| canvas->drawImageLattice(fImage.get(), fLattice, fDst, fFilter, fPaint.getMaybeNull()); |
| } |
| |
| bool DrawImageLatticeCommand::render(SkCanvas* canvas) const { |
| SkAutoCanvasRestore acr(canvas, true); |
| canvas->clear(0xFFFFFFFF); |
| |
| xlate_and_scale_to_bounds(canvas, fDst); |
| |
| this->execute(canvas); |
| return true; |
| } |
| |
| uint64_t DrawImageLatticeCommand::imageId(UrlDataManager& udm) const { |
| return udm.lookupImage(fImage.get()); |
| } |
| |
| void DrawImageLatticeCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| flatten(*fImage, writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_LATTICE); |
| MakeJsonLattice(writer, fLattice); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_DST); |
| MakeJsonRect(writer, fDst); |
| if (fPaint.isValid()) { |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, *fPaint, urlDataManager); |
| } |
| |
| SkString desc; |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, str_append(&desc, fDst)->c_str()); |
| } |
| |
| DrawImageRectCommand::DrawImageRectCommand(const SkImage* image, |
| const SkRect& src, |
| const SkRect& dst, |
| const SkSamplingOptions& sampling, |
| const SkPaint* paint, |
| SkCanvas::SrcRectConstraint constraint) |
| : INHERITED(kDrawImageRect_OpType) |
| , fImage(SkRef(image)) |
| , fSrc(src) |
| , fDst(dst) |
| , fSampling(sampling) |
| , fPaint(paint) |
| , fConstraint(constraint) {} |
| |
| void DrawImageRectCommand::execute(SkCanvas* canvas) const { |
| canvas->drawImageRect(fImage.get(), fSrc, fDst, fSampling, fPaint.getMaybeNull(), fConstraint); |
| } |
| |
| bool DrawImageRectCommand::render(SkCanvas* canvas) const { |
| SkAutoCanvasRestore acr(canvas, true); |
| canvas->clear(0xFFFFFFFF); |
| |
| xlate_and_scale_to_bounds(canvas, fDst); |
| |
| this->execute(canvas); |
| return true; |
| } |
| |
| uint64_t DrawImageRectCommand::imageId(UrlDataManager& udm) const { |
| return udm.lookupImage(fImage.get()); |
| } |
| |
| void DrawImageRectCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| flatten(*fImage, writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_SRC); |
| MakeJsonRect(writer, fSrc); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_DST); |
| MakeJsonRect(writer, fDst); |
| if (fPaint.isValid()) { |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, *fPaint, urlDataManager); |
| } |
| if (fConstraint == SkCanvas::kStrict_SrcRectConstraint) { |
| writer.appendBool(DEBUGCANVAS_ATTRIBUTE_STRICT, true); |
| } |
| |
| SkString desc; |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, str_append(&desc, fDst)->c_str()); |
| } |
| |
| DrawImageRectLayerCommand::DrawImageRectLayerCommand(DebugLayerManager* layerManager, |
| const int nodeId, |
| const int frame, |
| const SkRect& src, |
| const SkRect& dst, |
| const SkSamplingOptions& sampling, |
| const SkPaint* paint, |
| SkCanvas::SrcRectConstraint constraint) |
| : INHERITED(kDrawImageRectLayer_OpType) |
| , fLayerManager(layerManager) |
| , fNodeId(nodeId) |
| , fFrame(frame) |
| , fSrc(src) |
| , fDst(dst) |
| , fSampling(sampling) |
| , fPaint(paint) |
| , fConstraint(constraint) {} |
| |
| void DrawImageRectLayerCommand::execute(SkCanvas* canvas) const { |
| sk_sp<SkImage> snapshot = fLayerManager->getLayerAsImage(fNodeId, fFrame); |
| canvas->drawImageRect(snapshot.get(), fSrc, fDst, SkSamplingOptions(), fPaint.getMaybeNull(), fConstraint); |
| } |
| |
| bool DrawImageRectLayerCommand::render(SkCanvas* canvas) const { |
| SkAutoCanvasRestore acr(canvas, true); |
| canvas->clear(0xFFFFFFFF); |
| |
| xlate_and_scale_to_bounds(canvas, fDst); |
| |
| this->execute(canvas); |
| return true; |
| } |
| |
| void DrawImageRectLayerCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| |
| // Don't append an image attribute here, the image can be rendered in as many different ways |
| // as there are commands in the layer, at least. the urlDataManager would save each one under |
| // a different URL. |
| // Append the node id, and the layer inspector of the debugger will know what to do with it. |
| writer.appendS64(DEBUGCANVAS_ATTRIBUTE_LAYERNODEID, fNodeId); |
| |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_SRC); |
| MakeJsonRect(writer, fSrc); |
| |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_DST); |
| MakeJsonRect(writer, fDst); |
| if (fPaint.isValid()) { |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, *fPaint, urlDataManager); |
| } |
| if (fConstraint == SkCanvas::kStrict_SrcRectConstraint) { |
| writer.appendBool(DEBUGCANVAS_ATTRIBUTE_STRICT, true); |
| } |
| |
| SkString desc; |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, str_append(&desc, fDst)->c_str()); |
| } |
| |
| DrawOvalCommand::DrawOvalCommand(const SkRect& oval, const SkPaint& paint) |
| : INHERITED(kDrawOval_OpType) { |
| fOval = oval; |
| fPaint = paint; |
| } |
| |
| void DrawOvalCommand::execute(SkCanvas* canvas) const { canvas->drawOval(fOval, fPaint); } |
| |
| bool DrawOvalCommand::render(SkCanvas* canvas) const { |
| canvas->clear(0xFFFFFFFF); |
| canvas->save(); |
| |
| xlate_and_scale_to_bounds(canvas, fOval); |
| |
| SkPaint p; |
| p.setColor(SK_ColorBLACK); |
| p.setStyle(SkPaint::kStroke_Style); |
| |
| canvas->drawOval(fOval, p); |
| canvas->restore(); |
| |
| return true; |
| } |
| |
| void DrawOvalCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COORDS); |
| MakeJsonRect(writer, fOval); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| } |
| |
| DrawArcCommand::DrawArcCommand(const SkRect& oval, |
| SkScalar startAngle, |
| SkScalar sweepAngle, |
| bool useCenter, |
| const SkPaint& paint) |
| : INHERITED(kDrawArc_OpType) { |
| fOval = oval; |
| fStartAngle = startAngle; |
| fSweepAngle = sweepAngle; |
| fUseCenter = useCenter; |
| fPaint = paint; |
| } |
| |
| void DrawArcCommand::execute(SkCanvas* canvas) const { |
| canvas->drawArc(fOval, fStartAngle, fSweepAngle, fUseCenter, fPaint); |
| } |
| |
| bool DrawArcCommand::render(SkCanvas* canvas) const { |
| canvas->clear(0xFFFFFFFF); |
| canvas->save(); |
| |
| xlate_and_scale_to_bounds(canvas, fOval); |
| |
| SkPaint p; |
| p.setColor(SK_ColorBLACK); |
| p.setStyle(SkPaint::kStroke_Style); |
| |
| canvas->drawArc(fOval, fStartAngle, fSweepAngle, fUseCenter, p); |
| canvas->restore(); |
| |
| return true; |
| } |
| |
| void DrawArcCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COORDS); |
| MakeJsonRect(writer, fOval); |
| writer.appendFloat(DEBUGCANVAS_ATTRIBUTE_STARTANGLE, fStartAngle); |
| writer.appendFloat(DEBUGCANVAS_ATTRIBUTE_SWEEPANGLE, fSweepAngle); |
| writer.appendBool(DEBUGCANVAS_ATTRIBUTE_USECENTER, fUseCenter); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| } |
| |
| DrawPaintCommand::DrawPaintCommand(const SkPaint& paint) : INHERITED(kDrawPaint_OpType) { |
| fPaint = paint; |
| } |
| |
| void DrawPaintCommand::execute(SkCanvas* canvas) const { canvas->drawPaint(fPaint); } |
| |
| bool DrawPaintCommand::render(SkCanvas* canvas) const { |
| canvas->clear(0xFFFFFFFF); |
| canvas->drawPaint(fPaint); |
| return true; |
| } |
| |
| void DrawPaintCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| } |
| |
| DrawBehindCommand::DrawBehindCommand(const SkPaint& paint) : INHERITED(kDrawPaint_OpType) { |
| fPaint = paint; |
| } |
| |
| void DrawBehindCommand::execute(SkCanvas* canvas) const { |
| SkCanvasPriv::DrawBehind(canvas, fPaint); |
| } |
| |
| bool DrawBehindCommand::render(SkCanvas* canvas) const { |
| canvas->clear(0xFFFFFFFF); |
| SkCanvasPriv::DrawBehind(canvas, fPaint); |
| return true; |
| } |
| |
| void DrawBehindCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| } |
| |
| DrawPathCommand::DrawPathCommand(const SkPath& path, const SkPaint& paint) |
| : INHERITED(kDrawPath_OpType) { |
| fPath = path; |
| fPaint = paint; |
| } |
| |
| void DrawPathCommand::execute(SkCanvas* canvas) const { canvas->drawPath(fPath, fPaint); } |
| |
| bool DrawPathCommand::render(SkCanvas* canvas) const { |
| render_path(canvas, fPath); |
| return true; |
| } |
| |
| void DrawPathCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PATH); |
| MakeJsonPath(writer, fPath); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| } |
| |
| DrawRegionCommand::DrawRegionCommand(const SkRegion& region, const SkPaint& paint) |
| : INHERITED(kDrawRegion_OpType) { |
| fRegion = region; |
| fPaint = paint; |
| } |
| |
| void DrawRegionCommand::execute(SkCanvas* canvas) const { canvas->drawRegion(fRegion, fPaint); } |
| |
| bool DrawRegionCommand::render(SkCanvas* canvas) const { |
| render_region(canvas, fRegion); |
| return true; |
| } |
| |
| void DrawRegionCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_REGION); |
| MakeJsonRegion(writer, fRegion); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| } |
| |
| BeginDrawPictureCommand::BeginDrawPictureCommand(const SkPicture* picture, |
| const SkMatrix* matrix, |
| const SkPaint* paint) |
| : INHERITED(kBeginDrawPicture_OpType) |
| , fPicture(SkRef(picture)) |
| , fMatrix(matrix) |
| , fPaint(paint) {} |
| |
| void BeginDrawPictureCommand::execute(SkCanvas* canvas) const { |
| if (fPaint.isValid()) { |
| SkRect bounds = fPicture->cullRect(); |
| if (fMatrix.isValid()) { |
| fMatrix->mapRect(&bounds); |
| } |
| canvas->saveLayer(&bounds, fPaint.get()); |
| } |
| |
| if (fMatrix.isValid()) { |
| if (!fPaint.isValid()) { |
| canvas->save(); |
| } |
| canvas->concat(*fMatrix); |
| } |
| } |
| |
| bool BeginDrawPictureCommand::render(SkCanvas* canvas) const { |
| canvas->clear(0xFFFFFFFF); |
| canvas->save(); |
| |
| xlate_and_scale_to_bounds(canvas, fPicture->cullRect()); |
| |
| canvas->drawPicture(fPicture.get()); |
| |
| canvas->restore(); |
| |
| return true; |
| } |
| |
| EndDrawPictureCommand::EndDrawPictureCommand(bool restore) |
| : INHERITED(kEndDrawPicture_OpType), fRestore(restore) {} |
| |
| void EndDrawPictureCommand::execute(SkCanvas* canvas) const { |
| if (fRestore) { |
| canvas->restore(); |
| } |
| } |
| |
| DrawPointsCommand::DrawPointsCommand(SkCanvas::PointMode mode, |
| size_t count, |
| const SkPoint pts[], |
| const SkPaint& paint) |
| : INHERITED(kDrawPoints_OpType), fMode(mode), fPts(pts, count), fPaint(paint) {} |
| |
| void DrawPointsCommand::execute(SkCanvas* canvas) const { |
| canvas->drawPoints(fMode, fPts.count(), fPts.begin(), fPaint); |
| } |
| |
| bool DrawPointsCommand::render(SkCanvas* canvas) const { |
| canvas->clear(0xFFFFFFFF); |
| canvas->save(); |
| |
| SkRect bounds; |
| |
| bounds.setEmpty(); |
| for (int i = 0; i < fPts.count(); ++i) { |
| SkRectPriv::GrowToInclude(&bounds, fPts[i]); |
| } |
| |
| xlate_and_scale_to_bounds(canvas, bounds); |
| |
| SkPaint p; |
| p.setColor(SK_ColorBLACK); |
| p.setStyle(SkPaint::kStroke_Style); |
| |
| canvas->drawPoints(fMode, fPts.count(), fPts.begin(), p); |
| canvas->restore(); |
| |
| return true; |
| } |
| |
| void DrawPointsCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_MODE, pointmode_name(fMode)); |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_POINTS); |
| for (int i = 0; i < fPts.count(); i++) { |
| MakeJsonPoint(writer, fPts[i]); |
| } |
| writer.endArray(); // points |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| } |
| |
| DrawTextBlobCommand::DrawTextBlobCommand(sk_sp<SkTextBlob> blob, |
| SkScalar x, |
| SkScalar y, |
| const SkPaint& paint) |
| : INHERITED(kDrawTextBlob_OpType) |
| , fBlob(std::move(blob)) |
| , fXPos(x) |
| , fYPos(y) |
| , fPaint(paint) {} |
| |
| void DrawTextBlobCommand::execute(SkCanvas* canvas) const { |
| canvas->drawTextBlob(fBlob, fXPos, fYPos, fPaint); |
| } |
| |
| bool DrawTextBlobCommand::render(SkCanvas* canvas) const { |
| canvas->clear(SK_ColorWHITE); |
| canvas->save(); |
| |
| SkRect bounds = fBlob->bounds().makeOffset(fXPos, fYPos); |
| xlate_and_scale_to_bounds(canvas, bounds); |
| |
| canvas->drawTextBlob(fBlob, fXPos, fYPos, fPaint); |
| |
| canvas->restore(); |
| |
| return true; |
| } |
| |
| void DrawTextBlobCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_RUNS); |
| SkTextBlobRunIterator iter(fBlob.get()); |
| while (!iter.done()) { |
| writer.beginObject(); // run |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_GLYPHS); |
| for (uint32_t i = 0; i < iter.glyphCount(); i++) { |
| writer.appendU32(iter.glyphs()[i]); |
| } |
| writer.endArray(); // glyphs |
| if (iter.positioning() != SkTextBlobRunIterator::kDefault_Positioning) { |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_POSITIONS); |
| const SkScalar* iterPositions = iter.pos(); |
| for (uint32_t i = 0; i < iter.glyphCount(); i++) { |
| switch (iter.positioning()) { |
| case SkTextBlobRunIterator::kFull_Positioning: |
| MakeJsonPoint(writer, iterPositions[i * 2], iterPositions[i * 2 + 1]); |
| break; |
| case SkTextBlobRunIterator::kHorizontal_Positioning: |
| writer.appendFloat(iterPositions[i]); |
| break; |
| case SkTextBlobRunIterator::kDefault_Positioning: break; |
| case SkTextBlobRunIterator::kRSXform_Positioning: |
| // TODO_RSXFORM_BLOB |
| break; |
| } |
| } |
| writer.endArray(); // positions |
| } |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_FONT); |
| MakeJsonFont(iter.font(), writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COORDS); |
| MakeJsonPoint(writer, iter.offset()); |
| |
| writer.endObject(); // run |
| iter.next(); |
| } |
| writer.endArray(); // runs |
| writer.appendFloat(DEBUGCANVAS_ATTRIBUTE_X, fXPos); |
| writer.appendFloat(DEBUGCANVAS_ATTRIBUTE_Y, fYPos); |
| SkRect bounds = fBlob->bounds(); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COORDS); |
| MakeJsonRect(writer, bounds); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| |
| SkString desc; |
| // make the bounds local by applying the x,y |
| bounds.offset(fXPos, fYPos); |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, str_append(&desc, bounds)->c_str()); |
| } |
| |
| DrawPatchCommand::DrawPatchCommand(const SkPoint cubics[12], |
| const SkColor colors[4], |
| const SkPoint texCoords[4], |
| SkBlendMode bmode, |
| const SkPaint& paint) |
| : INHERITED(kDrawPatch_OpType), fBlendMode(bmode) { |
| memcpy(fCubics, cubics, sizeof(fCubics)); |
| if (colors != nullptr) { |
| memcpy(fColors, colors, sizeof(fColors)); |
| fColorsPtr = fColors; |
| } else { |
| fColorsPtr = nullptr; |
| } |
| if (texCoords != nullptr) { |
| memcpy(fTexCoords, texCoords, sizeof(fTexCoords)); |
| fTexCoordsPtr = fTexCoords; |
| } else { |
| fTexCoordsPtr = nullptr; |
| } |
| fPaint = paint; |
| } |
| |
| void DrawPatchCommand::execute(SkCanvas* canvas) const { |
| canvas->drawPatch(fCubics, fColorsPtr, fTexCoordsPtr, fBlendMode, fPaint); |
| } |
| |
| void DrawPatchCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_CUBICS); |
| for (int i = 0; i < 12; i++) { |
| MakeJsonPoint(writer, fCubics[i]); |
| } |
| writer.endArray(); // cubics |
| if (fColorsPtr != nullptr) { |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_COLORS); |
| for (int i = 0; i < 4; i++) { |
| MakeJsonColor(writer, fColorsPtr[i]); |
| } |
| writer.endArray(); // colors |
| } |
| if (fTexCoordsPtr != nullptr) { |
| writer.beginArray(DEBUGCANVAS_ATTRIBUTE_TEXTURECOORDS); |
| for (int i = 0; i < 4; i++) { |
| MakeJsonPoint(writer, fTexCoords[i]); |
| } |
| writer.endArray(); // texCoords |
| } |
| // fBlendMode |
| } |
| |
| DrawRectCommand::DrawRectCommand(const SkRect& rect, const SkPaint& paint) |
| : INHERITED(kDrawRect_OpType) { |
| fRect = rect; |
| fPaint = paint; |
| } |
| |
| void DrawRectCommand::execute(SkCanvas* canvas) const { canvas->drawRect(fRect, fPaint); } |
| |
| void DrawRectCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COORDS); |
| MakeJsonRect(writer, fRect); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| |
| SkString desc; |
| writer.appendString(DEBUGCANVAS_ATTRIBUTE_SHORTDESC, str_append(&desc, fRect)->c_str()); |
| } |
| |
| DrawRRectCommand::DrawRRectCommand(const SkRRect& rrect, const SkPaint& paint) |
| : INHERITED(kDrawRRect_OpType) { |
| fRRect = rrect; |
| fPaint = paint; |
| } |
| |
| void DrawRRectCommand::execute(SkCanvas* canvas) const { canvas->drawRRect(fRRect, fPaint); } |
| |
| bool DrawRRectCommand::render(SkCanvas* canvas) const { |
| render_rrect(canvas, fRRect); |
| return true; |
| } |
| |
| void DrawRRectCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_COORDS); |
| make_json_rrect(writer, fRRect); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| } |
| |
| DrawDRRectCommand::DrawDRRectCommand(const SkRRect& outer, |
| const SkRRect& inner, |
| const SkPaint& paint) |
| : INHERITED(kDrawDRRect_OpType) { |
| fOuter = outer; |
| fInner = inner; |
| fPaint = paint; |
| } |
| |
| void DrawDRRectCommand::execute(SkCanvas* canvas) const { |
| canvas->drawDRRect(fOuter, fInner, fPaint); |
| } |
| |
| bool DrawDRRectCommand::render(SkCanvas* canvas) const { |
| render_drrect(canvas, fOuter, fInner); |
| return true; |
| } |
| |
| void DrawDRRectCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_OUTER); |
| make_json_rrect(writer, fOuter); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_INNER); |
| make_json_rrect(writer, fInner); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, fPaint, urlDataManager); |
| } |
| |
| DrawShadowCommand::DrawShadowCommand(const SkPath& path, const SkDrawShadowRec& rec) |
| : INHERITED(kDrawShadow_OpType) { |
| fPath = path; |
| fShadowRec = rec; |
| } |
| |
| void DrawShadowCommand::execute(SkCanvas* canvas) const { |
| canvas->private_draw_shadow_rec(fPath, fShadowRec); |
| } |
| |
| bool DrawShadowCommand::render(SkCanvas* canvas) const { |
| render_shadow(canvas, fPath, fShadowRec); |
| return true; |
| } |
| |
| void DrawShadowCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| |
| bool geometricOnly = SkToBool(fShadowRec.fFlags & SkShadowFlags::kGeometricOnly_ShadowFlag); |
| bool transparentOccluder = |
| SkToBool(fShadowRec.fFlags & SkShadowFlags::kTransparentOccluder_ShadowFlag); |
| |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PATH); |
| MakeJsonPath(writer, fPath); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_ZPLANE); |
| MakeJsonPoint3(writer, fShadowRec.fZPlaneParams); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_LIGHTPOSITION); |
| MakeJsonPoint3(writer, fShadowRec.fLightPos); |
| writer.appendFloat(DEBUGCANVAS_ATTRIBUTE_LIGHTRADIUS, fShadowRec.fLightRadius); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_AMBIENTCOLOR); |
| MakeJsonColor(writer, fShadowRec.fAmbientColor); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_SPOTCOLOR); |
| MakeJsonColor(writer, fShadowRec.fSpotColor); |
| store_bool(writer, DEBUGCANVAS_SHADOWFLAG_TRANSPARENT_OCC, transparentOccluder, false); |
| store_bool(writer, DEBUGCANVAS_SHADOWFLAG_GEOMETRIC_ONLY, geometricOnly, false); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////////////////////////// |
| |
| DrawEdgeAAQuadCommand::DrawEdgeAAQuadCommand(const SkRect& rect, |
| const SkPoint clip[], |
| SkCanvas::QuadAAFlags aa, |
| const SkColor4f& color, |
| SkBlendMode mode) |
| : INHERITED(kDrawEdgeAAQuad_OpType) |
| , fRect(rect) |
| , fHasClip(clip != nullptr) |
| , fAA(aa) |
| , fColor(color) |
| , fMode(mode) { |
| if (clip) { |
| for (int i = 0; i < 4; ++i) { |
| fClip[i] = clip[i]; |
| } |
| } |
| } |
| |
| void DrawEdgeAAQuadCommand::execute(SkCanvas* canvas) const { |
| canvas->experimental_DrawEdgeAAQuad(fRect, fHasClip ? fClip : nullptr, fAA, fColor, fMode); |
| } |
| |
| DrawEdgeAAImageSetCommand::DrawEdgeAAImageSetCommand(const SkCanvas::ImageSetEntry set[], |
| int count, |
| const SkPoint dstClips[], |
| const SkMatrix preViewMatrices[], |
| const SkSamplingOptions& sampling, |
| const SkPaint* paint, |
| SkCanvas::SrcRectConstraint constraint) |
| : INHERITED(kDrawEdgeAAImageSet_OpType) |
| , fSet(count) |
| , fCount(count) |
| , fSampling(sampling) |
| , fPaint(paint) |
| , fConstraint(constraint) { |
| int totalDstClipCount, totalMatrixCount; |
| SkCanvasPriv::GetDstClipAndMatrixCounts(set, count, &totalDstClipCount, &totalMatrixCount); |
| |
| std::copy_n(set, count, fSet.get()); |
| fDstClips.reset(totalDstClipCount); |
| std::copy_n(dstClips, totalDstClipCount, fDstClips.get()); |
| fPreViewMatrices.reset(totalMatrixCount); |
| std::copy_n(preViewMatrices, totalMatrixCount, fPreViewMatrices.get()); |
| } |
| |
| void DrawEdgeAAImageSetCommand::execute(SkCanvas* canvas) const { |
| canvas->experimental_DrawEdgeAAImageSet(fSet.get(), |
| fCount, |
| fDstClips.get(), |
| fPreViewMatrices.get(), |
| fSampling, |
| fPaint.getMaybeNull(), |
| fConstraint); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////////////////////////// |
| |
| DrawDrawableCommand::DrawDrawableCommand(SkDrawable* drawable, const SkMatrix* matrix) |
| : INHERITED(kDrawDrawable_OpType), fDrawable(SkRef(drawable)), fMatrix(matrix) {} |
| |
| void DrawDrawableCommand::execute(SkCanvas* canvas) const { |
| canvas->drawDrawable(fDrawable.get(), fMatrix.getMaybeNull()); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////////////////////////// |
| |
| DrawVerticesCommand::DrawVerticesCommand(sk_sp<SkVertices> vertices, |
| SkBlendMode bmode, |
| const SkPaint& paint) |
| : INHERITED(kDrawVertices_OpType) |
| , fVertices(std::move(vertices)) |
| , fBlendMode(bmode) |
| , fPaint(paint) {} |
| |
| void DrawVerticesCommand::execute(SkCanvas* canvas) const { |
| canvas->drawVertices(fVertices, fBlendMode, fPaint); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////////////////////////// |
| |
| DrawAtlasCommand::DrawAtlasCommand(const SkImage* image, |
| const SkRSXform xform[], |
| const SkRect tex[], |
| const SkColor colors[], |
| int count, |
| SkBlendMode bmode, |
| const SkSamplingOptions& sampling, |
| const SkRect* cull, |
| const SkPaint* paint) |
| : INHERITED(kDrawAtlas_OpType) |
| , fImage(SkRef(image)) |
| , fXform(xform, count) |
| , fTex(tex, count) |
| , fColors(colors, colors ? count : 0) |
| , fBlendMode(bmode) |
| , fSampling(sampling) |
| , fCull(cull) |
| , fPaint(paint) {} |
| |
| void DrawAtlasCommand::execute(SkCanvas* canvas) const { |
| canvas->drawAtlas(fImage.get(), |
| fXform.begin(), |
| fTex.begin(), |
| fColors.isEmpty() ? nullptr : fColors.begin(), |
| fXform.count(), |
| fBlendMode, |
| fSampling, |
| fCull.getMaybeNull(), |
| fPaint.getMaybeNull()); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////////////////////////// |
| |
| RestoreCommand::RestoreCommand() : INHERITED(kRestore_OpType) {} |
| |
| void RestoreCommand::execute(SkCanvas* canvas) const { canvas->restore(); } |
| |
| SaveCommand::SaveCommand() : INHERITED(kSave_OpType) {} |
| |
| void SaveCommand::execute(SkCanvas* canvas) const { canvas->save(); } |
| |
| SaveLayerCommand::SaveLayerCommand(const SkCanvas::SaveLayerRec& rec) |
| : INHERITED(kSaveLayer_OpType) |
| , fBounds(rec.fBounds) |
| , fPaint(rec.fPaint) |
| , fBackdrop(SkSafeRef(rec.fBackdrop)) |
| , fSaveLayerFlags(rec.fSaveLayerFlags) {} |
| |
| void SaveLayerCommand::execute(SkCanvas* canvas) const { |
| canvas->saveLayer( |
| SkCanvas::SaveLayerRec(fBounds.getMaybeNull(), fPaint.getMaybeNull(), fSaveLayerFlags)); |
| } |
| |
| void SaveLayerCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| if (fBounds.isValid()) { |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_BOUNDS); |
| MakeJsonRect(writer, *fBounds); |
| } |
| if (fPaint.isValid()) { |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_PAINT); |
| MakeJsonPaint(writer, *fPaint, urlDataManager); |
| } |
| if (fBackdrop != nullptr) { |
| writer.beginObject(DEBUGCANVAS_ATTRIBUTE_BACKDROP); |
| flatten(fBackdrop.get(), writer, urlDataManager); |
| writer.endObject(); // backdrop |
| } |
| if (fSaveLayerFlags != 0) { |
| SkDebugf("unsupported: saveLayer flags\n"); |
| SkASSERT(false); |
| } |
| } |
| |
| SetMatrixCommand::SetMatrixCommand(const SkMatrix& matrix) : INHERITED(kSetMatrix_OpType) { |
| fMatrix = matrix; |
| } |
| |
| void SetMatrixCommand::execute(SkCanvas* canvas) const { canvas->setMatrix(fMatrix); } |
| |
| void SetMatrixCommand::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_MATRIX); |
| MakeJsonMatrix(writer, fMatrix); |
| writeMatrixType(writer, fMatrix); |
| } |
| |
| SetM44Command::SetM44Command(const SkM44& matrix) : INHERITED(kSetM44_OpType) { |
| fMatrix = matrix; |
| } |
| |
| void SetM44Command::execute(SkCanvas* canvas) const { canvas->setMatrix(fMatrix); } |
| |
| void SetM44Command::toJSON(SkJSONWriter& writer, UrlDataManager& urlDataManager) const { |
| INHERITED::toJSON(writer, urlDataManager); |
| writer.appendName(DEBUGCANVAS_ATTRIBUTE_MATRIX); |
| MakeJsonMatrix44(writer, fMatrix); |
| } |