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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "SkObjectParser.h"
|
2013-01-28 20:21:59 +00:00
|
|
|
#include "SkData.h"
|
|
|
|
#include "SkFontDescriptor.h"
|
2013-01-02 20:20:31 +00:00
|
|
|
#include "SkRRect.h"
|
2013-01-15 20:17:47 +00:00
|
|
|
#include "SkShader.h"
|
2013-01-07 15:48:19 +00:00
|
|
|
#include "SkStream.h"
|
2013-01-28 20:21:59 +00:00
|
|
|
#include "SkStringUtils.h"
|
|
|
|
#include "SkTypeface.h"
|
2013-03-07 20:30:32 +00:00
|
|
|
#include "SkUtils.h"
|
2012-06-29 14:21:22 +00:00
|
|
|
|
|
|
|
/* TODO(chudy): Replace all std::strings with char */
|
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::BitmapToString(const SkBitmap& bitmap) {
|
2012-10-23 12:13:35 +00:00
|
|
|
SkString* mBitmap = new SkString("SkBitmap: ");
|
|
|
|
mBitmap->append("W: ");
|
|
|
|
mBitmap->appendS32(bitmap.width());
|
|
|
|
mBitmap->append(" H: ");
|
|
|
|
mBitmap->appendS32(bitmap.height());
|
|
|
|
|
|
|
|
const char* gConfigStrings[] = {
|
2013-12-02 13:50:38 +00:00
|
|
|
"None", "A8", "Index8", "RGB565", "ARGB4444", "ARGB8888"
|
2012-10-23 12:13:35 +00:00
|
|
|
};
|
2013-12-02 13:50:38 +00:00
|
|
|
SkASSERT(SkBitmap::kConfigCount == SK_ARRAY_COUNT(gConfigStrings));
|
2012-10-23 12:13:35 +00:00
|
|
|
|
|
|
|
mBitmap->append(" Config: ");
|
2013-10-31 17:28:30 +00:00
|
|
|
mBitmap->append(gConfigStrings[bitmap.config()]);
|
2012-10-23 12:13:35 +00:00
|
|
|
|
|
|
|
if (bitmap.isOpaque()) {
|
|
|
|
mBitmap->append(" opaque");
|
|
|
|
} else {
|
|
|
|
mBitmap->append(" not-opaque");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bitmap.isImmutable()) {
|
|
|
|
mBitmap->append(" immutable");
|
|
|
|
} else {
|
|
|
|
mBitmap->append(" not-immutable");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bitmap.isVolatile()) {
|
|
|
|
mBitmap->append(" volatile");
|
|
|
|
} else {
|
|
|
|
mBitmap->append(" not-volatile");
|
|
|
|
}
|
|
|
|
|
|
|
|
mBitmap->append(" genID: ");
|
|
|
|
mBitmap->appendS32(bitmap.getGenerationID());
|
|
|
|
|
2012-06-29 14:21:22 +00:00
|
|
|
return mBitmap;
|
|
|
|
}
|
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::BoolToString(bool doAA) {
|
|
|
|
SkString* mBool = new SkString("Bool doAA: ");
|
2012-06-29 14:21:22 +00:00
|
|
|
if (doAA) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mBool->append("True");
|
2012-06-29 14:21:22 +00:00
|
|
|
} else {
|
2012-08-07 20:41:37 +00:00
|
|
|
mBool->append("False");
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
2012-08-07 20:41:37 +00:00
|
|
|
return mBool;
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::CustomTextToString(const char* text) {
|
|
|
|
SkString* mText = new SkString(text);
|
|
|
|
return mText;
|
|
|
|
}
|
|
|
|
|
|
|
|
SkString* SkObjectParser::IntToString(int x, const char* text) {
|
|
|
|
SkString* mInt = new SkString(text);
|
|
|
|
mInt->append(" ");
|
|
|
|
mInt->appendScalar(SkIntToScalar(x));
|
|
|
|
return mInt;
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::IRectToString(const SkIRect& rect) {
|
|
|
|
SkString* mRect = new SkString("SkIRect: ");
|
|
|
|
mRect->append("L: ");
|
2012-10-23 12:13:35 +00:00
|
|
|
mRect->appendS32(rect.left());
|
2012-08-07 20:41:37 +00:00
|
|
|
mRect->append(", T: ");
|
2012-10-23 12:13:35 +00:00
|
|
|
mRect->appendS32(rect.top());
|
2012-08-07 20:41:37 +00:00
|
|
|
mRect->append(", R: ");
|
2012-10-23 12:13:35 +00:00
|
|
|
mRect->appendS32(rect.right());
|
2012-08-07 20:41:37 +00:00
|
|
|
mRect->append(", B: ");
|
2012-10-23 12:13:35 +00:00
|
|
|
mRect->appendS32(rect.bottom());
|
2012-08-07 20:41:37 +00:00
|
|
|
return mRect;
|
|
|
|
}
|
2012-06-29 14:21:22 +00:00
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::MatrixToString(const SkMatrix& matrix) {
|
2013-02-14 13:53:53 +00:00
|
|
|
SkString* str = new SkString("SkMatrix: ");
|
2013-02-14 14:40:27 +00:00
|
|
|
#ifdef SK_DEVELOPER
|
2013-02-14 13:53:53 +00:00
|
|
|
matrix.toString(str);
|
2013-02-14 14:40:27 +00:00
|
|
|
#endif
|
2013-02-14 13:53:53 +00:00
|
|
|
return str;
|
2012-08-07 20:41:37 +00:00
|
|
|
}
|
2012-06-29 14:21:22 +00:00
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::PaintToString(const SkPaint& paint) {
|
2013-02-14 13:53:53 +00:00
|
|
|
SkString* str = new SkString;
|
2013-02-14 14:40:27 +00:00
|
|
|
#ifdef SK_DEVELOPER
|
2013-02-14 13:53:53 +00:00
|
|
|
paint.toString(str);
|
2013-02-14 14:40:27 +00:00
|
|
|
#endif
|
2013-02-14 13:53:53 +00:00
|
|
|
return str;
|
2012-08-07 20:41:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SkString* SkObjectParser::PathToString(const SkPath& path) {
|
2012-10-18 13:30:18 +00:00
|
|
|
SkString* mPath = new SkString("Path (");
|
|
|
|
|
2012-12-05 19:34:33 +00:00
|
|
|
static const char* gFillStrings[] = {
|
|
|
|
"Winding", "EvenOdd", "InverseWinding", "InverseEvenOdd"
|
|
|
|
};
|
|
|
|
|
|
|
|
mPath->append(gFillStrings[path.getFillType()]);
|
|
|
|
mPath->append(", ");
|
|
|
|
|
2012-10-18 13:30:18 +00:00
|
|
|
static const char* gConvexityStrings[] = {
|
2012-10-19 02:01:19 +00:00
|
|
|
"Unknown", "Convex", "Concave"
|
2012-10-18 13:30:18 +00:00
|
|
|
};
|
|
|
|
SkASSERT(SkPath::kConcave_Convexity == 2);
|
|
|
|
|
|
|
|
mPath->append(gConvexityStrings[path.getConvexity()]);
|
|
|
|
mPath->append(", ");
|
|
|
|
|
2012-10-23 12:13:35 +00:00
|
|
|
if (path.isRect(NULL)) {
|
|
|
|
mPath->append("isRect, ");
|
|
|
|
} else {
|
|
|
|
mPath->append("isNotRect, ");
|
|
|
|
}
|
|
|
|
|
2012-10-18 13:30:18 +00:00
|
|
|
mPath->appendS32(path.countVerbs());
|
|
|
|
mPath->append("V, ");
|
|
|
|
mPath->appendS32(path.countPoints());
|
|
|
|
mPath->append("P): ");
|
|
|
|
|
|
|
|
static const char* gVerbStrings[] = {
|
2013-06-04 21:51:06 +00:00
|
|
|
"Move", "Line", "Quad", "Conic", "Cubic", "Close", "Done"
|
2012-10-18 13:30:18 +00:00
|
|
|
};
|
2013-06-04 21:51:06 +00:00
|
|
|
static const int gPtsPerVerb[] = { 1, 1, 2, 2, 3, 0, 0 };
|
|
|
|
static const int gPtOffsetPerVerb[] = { 0, 1, 1, 1, 1, 0, 0 };
|
|
|
|
SkASSERT(SkPath::kDone_Verb == 6);
|
2012-10-18 13:30:18 +00:00
|
|
|
|
|
|
|
SkPath::Iter iter(const_cast<SkPath&>(path), false);
|
|
|
|
SkPath::Verb verb;
|
|
|
|
SkPoint points[4];
|
|
|
|
|
2012-10-19 02:01:19 +00:00
|
|
|
for(verb = iter.next(points, false);
|
|
|
|
verb != SkPath::kDone_Verb;
|
2012-10-18 13:30:18 +00:00
|
|
|
verb = iter.next(points, false)) {
|
|
|
|
|
|
|
|
mPath->append(gVerbStrings[verb]);
|
|
|
|
mPath->append(" ");
|
|
|
|
|
|
|
|
for (int i = 0; i < gPtsPerVerb[verb]; ++i) {
|
|
|
|
mPath->append("(");
|
|
|
|
mPath->appendScalar(points[gPtOffsetPerVerb[verb]+i].fX);
|
|
|
|
mPath->append(", ");
|
|
|
|
mPath->appendScalar(points[gPtOffsetPerVerb[verb]+i].fY);
|
2013-06-04 21:51:06 +00:00
|
|
|
mPath->append(")");
|
2012-10-18 13:30:18 +00:00
|
|
|
}
|
2013-06-04 21:51:06 +00:00
|
|
|
|
|
|
|
if (SkPath::kConic_Verb == verb) {
|
|
|
|
mPath->append("(");
|
|
|
|
mPath->appendScalar(iter.conicWeight());
|
|
|
|
mPath->append(")");
|
|
|
|
}
|
|
|
|
|
|
|
|
mPath->append(" ");
|
2012-08-07 20:41:37 +00:00
|
|
|
}
|
2012-10-18 13:30:18 +00:00
|
|
|
|
2012-11-06 16:45:36 +00:00
|
|
|
SkString* boundStr = SkObjectParser::RectToString(path.getBounds(), " Bound: ");
|
|
|
|
|
|
|
|
if (NULL != boundStr) {
|
|
|
|
mPath->append(*boundStr);
|
|
|
|
SkDELETE(boundStr);
|
|
|
|
}
|
|
|
|
|
2012-06-29 14:21:22 +00:00
|
|
|
return mPath;
|
|
|
|
}
|
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::PointsToString(const SkPoint pts[], size_t count) {
|
|
|
|
SkString* mPoints = new SkString("SkPoints pts[]: ");
|
2012-06-29 14:21:22 +00:00
|
|
|
for (unsigned int i = 0; i < count; i++) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mPoints->append("(");
|
|
|
|
mPoints->appendScalar(pts[i].fX);
|
|
|
|
mPoints->append(",");
|
|
|
|
mPoints->appendScalar(pts[i].fY);
|
|
|
|
mPoints->append(")");
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
2012-08-07 20:41:37 +00:00
|
|
|
return mPoints;
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::PointModeToString(SkCanvas::PointMode mode) {
|
|
|
|
SkString* mMode = new SkString("SkCanvas::PointMode: ");
|
2012-08-01 16:10:06 +00:00
|
|
|
if (mode == SkCanvas::kPoints_PointMode) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mMode->append("kPoints_PointMode");
|
2012-08-01 16:10:06 +00:00
|
|
|
} else if (mode == SkCanvas::kLines_PointMode) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mMode->append("kLines_Mode");
|
2012-08-01 16:10:06 +00:00
|
|
|
} else if (mode == SkCanvas::kPolygon_PointMode) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mMode->append("kPolygon_PointMode");
|
2012-08-01 16:10:06 +00:00
|
|
|
}
|
|
|
|
return mMode;
|
|
|
|
}
|
|
|
|
|
2012-11-06 16:45:36 +00:00
|
|
|
SkString* SkObjectParser::RectToString(const SkRect& rect, const char* title) {
|
|
|
|
|
|
|
|
SkString* mRect = new SkString;
|
2012-11-08 02:03:56 +00:00
|
|
|
|
2012-11-06 16:45:36 +00:00
|
|
|
if (NULL == title) {
|
|
|
|
mRect->append("SkRect: ");
|
|
|
|
} else {
|
|
|
|
mRect->append(title);
|
|
|
|
}
|
2012-08-07 20:41:37 +00:00
|
|
|
mRect->append("(");
|
|
|
|
mRect->appendScalar(rect.left());
|
|
|
|
mRect->append(", ");
|
|
|
|
mRect->appendScalar(rect.top());
|
|
|
|
mRect->append(", ");
|
|
|
|
mRect->appendScalar(rect.right());
|
|
|
|
mRect->append(", ");
|
|
|
|
mRect->appendScalar(rect.bottom());
|
|
|
|
mRect->append(")");
|
2012-06-29 14:21:22 +00:00
|
|
|
return mRect;
|
|
|
|
}
|
|
|
|
|
2013-01-02 20:20:31 +00:00
|
|
|
SkString* SkObjectParser::RRectToString(const SkRRect& rrect, const char* title) {
|
|
|
|
|
|
|
|
SkString* mRRect = new SkString;
|
|
|
|
|
|
|
|
if (NULL == title) {
|
|
|
|
mRRect->append("SkRRect (");
|
|
|
|
if (rrect.isEmpty()) {
|
|
|
|
mRRect->append("empty");
|
|
|
|
} else if (rrect.isRect()) {
|
|
|
|
mRRect->append("rect");
|
|
|
|
} else if (rrect.isOval()) {
|
|
|
|
mRRect->append("oval");
|
|
|
|
} else if (rrect.isSimple()) {
|
|
|
|
mRRect->append("simple");
|
|
|
|
} else {
|
|
|
|
SkASSERT(rrect.isComplex());
|
|
|
|
mRRect->append("complex");
|
|
|
|
}
|
|
|
|
mRRect->append("): ");
|
|
|
|
} else {
|
|
|
|
mRRect->append(title);
|
|
|
|
}
|
|
|
|
mRRect->append("(");
|
|
|
|
mRRect->appendScalar(rrect.rect().left());
|
|
|
|
mRRect->append(", ");
|
|
|
|
mRRect->appendScalar(rrect.rect().top());
|
|
|
|
mRRect->append(", ");
|
|
|
|
mRRect->appendScalar(rrect.rect().right());
|
|
|
|
mRRect->append(", ");
|
|
|
|
mRRect->appendScalar(rrect.rect().bottom());
|
|
|
|
mRRect->append(") radii: (");
|
|
|
|
for (int i = 0; i < 4; ++i) {
|
|
|
|
const SkVector& radii = rrect.radii((SkRRect::Corner) i);
|
|
|
|
mRRect->appendScalar(radii.fX);
|
|
|
|
mRRect->append(", ");
|
|
|
|
mRRect->appendScalar(radii.fY);
|
|
|
|
if (i < 3) {
|
|
|
|
mRRect->append(", ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mRRect->append(")");
|
|
|
|
return mRRect;
|
|
|
|
}
|
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::RegionOpToString(SkRegion::Op op) {
|
|
|
|
SkString* mOp = new SkString("SkRegion::Op: ");
|
2012-06-29 14:21:22 +00:00
|
|
|
if (op == SkRegion::kDifference_Op) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mOp->append("kDifference_Op");
|
2012-06-29 14:21:22 +00:00
|
|
|
} else if (op == SkRegion::kIntersect_Op) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mOp->append("kIntersect_Op");
|
2012-06-29 14:21:22 +00:00
|
|
|
} else if (op == SkRegion::kUnion_Op) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mOp->append("kUnion_Op");
|
2012-06-29 14:21:22 +00:00
|
|
|
} else if (op == SkRegion::kXOR_Op) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mOp->append("kXOR_Op");
|
2012-06-29 14:21:22 +00:00
|
|
|
} else if (op == SkRegion::kReverseDifference_Op) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mOp->append("kReverseDifference_Op");
|
2012-06-29 14:21:22 +00:00
|
|
|
} else if (op == SkRegion::kReplace_Op) {
|
2012-08-07 20:41:37 +00:00
|
|
|
mOp->append("kReplace_Op");
|
2012-06-29 14:21:22 +00:00
|
|
|
} else {
|
2012-08-07 20:41:37 +00:00
|
|
|
mOp->append("Unknown Type");
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
return mOp;
|
|
|
|
}
|
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::RegionToString(const SkRegion& region) {
|
|
|
|
SkString* mRegion = new SkString("SkRegion: Data unavailable.");
|
|
|
|
return mRegion;
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::SaveFlagsToString(SkCanvas::SaveFlags flags) {
|
|
|
|
SkString* mFlags = new SkString("SkCanvas::SaveFlags: ");
|
2013-07-11 16:25:55 +00:00
|
|
|
if (flags & SkCanvas::kMatrix_SaveFlag) {
|
|
|
|
mFlags->append("kMatrix_SaveFlag ");
|
|
|
|
}
|
|
|
|
if (flags & SkCanvas::kClip_SaveFlag) {
|
|
|
|
mFlags->append("kClip_SaveFlag ");
|
|
|
|
}
|
|
|
|
if (flags & SkCanvas::kHasAlphaLayer_SaveFlag) {
|
|
|
|
mFlags->append("kHasAlphaLayer_SaveFlag ");
|
|
|
|
}
|
|
|
|
if (flags & SkCanvas::kFullColorLayer_SaveFlag) {
|
|
|
|
mFlags->append("kFullColorLayer_SaveFlag ");
|
|
|
|
}
|
|
|
|
if (flags & SkCanvas::kClipToLayer_SaveFlag) {
|
|
|
|
mFlags->append("kClipToLayer_SaveFlag ");
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|
|
|
|
return mFlags;
|
|
|
|
}
|
|
|
|
|
2012-08-07 20:41:37 +00:00
|
|
|
SkString* SkObjectParser::ScalarToString(SkScalar x, const char* text) {
|
|
|
|
SkString* mScalar = new SkString(text);
|
|
|
|
mScalar->append(" ");
|
|
|
|
mScalar->appendScalar(x);
|
2012-06-29 14:21:22 +00:00
|
|
|
return mScalar;
|
|
|
|
}
|
|
|
|
|
2013-03-07 20:30:32 +00:00
|
|
|
SkString* SkObjectParser::TextToString(const void* text, size_t byteLength,
|
|
|
|
SkPaint::TextEncoding encoding) {
|
|
|
|
|
|
|
|
SkString* decodedText = new SkString();
|
|
|
|
switch (encoding) {
|
|
|
|
case SkPaint::kUTF8_TextEncoding: {
|
|
|
|
decodedText->append("UTF-8: ");
|
|
|
|
decodedText->append((const char*)text, byteLength);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SkPaint::kUTF16_TextEncoding: {
|
|
|
|
decodedText->append("UTF-16: ");
|
2013-11-22 07:02:24 +00:00
|
|
|
size_t sizeNeeded = SkUTF16_ToUTF8((uint16_t*)text,
|
|
|
|
SkToS32(byteLength / 2),
|
2013-11-21 14:24:16 +00:00
|
|
|
NULL);
|
2013-08-26 21:20:04 +00:00
|
|
|
SkAutoSTMalloc<0x100, char> utf8(sizeNeeded);
|
2013-11-21 14:24:16 +00:00
|
|
|
SkUTF16_ToUTF8((uint16_t*)text, SkToS32(byteLength / 2), utf8);
|
2013-03-07 20:30:32 +00:00
|
|
|
decodedText->append(utf8, sizeNeeded);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SkPaint::kUTF32_TextEncoding: {
|
|
|
|
decodedText->append("UTF-32: ");
|
|
|
|
const SkUnichar* begin = (const SkUnichar*)text;
|
|
|
|
const SkUnichar* end = (const SkUnichar*)((const char*)text + byteLength);
|
|
|
|
for (const SkUnichar* unichar = begin; unichar < end; ++unichar) {
|
|
|
|
decodedText->appendUnichar(*unichar);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SkPaint::kGlyphID_TextEncoding: {
|
|
|
|
decodedText->append("GlyphID: ");
|
|
|
|
const uint16_t* begin = (const uint16_t*)text;
|
|
|
|
const uint16_t* end = (const uint16_t*)((const char*)text + byteLength);
|
|
|
|
for (const uint16_t* glyph = begin; glyph < end; ++glyph) {
|
|
|
|
decodedText->append("0x");
|
|
|
|
decodedText->appendHex(*glyph);
|
|
|
|
decodedText->append(" ");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
decodedText->append("Unknown text encoding.");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return decodedText;
|
2012-06-29 14:21:22 +00:00
|
|
|
}
|