Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(126)

Unified Diff: src/core/SkTextBlob.cpp

Issue 473633002: SkTextBlob (Closed) Base URL: https://chromium.googlesource.com/skia.git@master
Patch Set: More const API, minimal docs. Created 6 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: src/core/SkTextBlob.cpp
diff --git a/src/core/SkTextBlob.cpp b/src/core/SkTextBlob.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..2ad29836e1d5387810a3594a10fb2b218c18aa11
--- /dev/null
+++ b/src/core/SkTextBlob.cpp
@@ -0,0 +1,177 @@
+/*
+ * Copyright 2014 Google Inc.
+ *
+ * Use of this source code is governed by a BSD-style license that can be
+ * found in the LICENSE file.
+ */
+
+#include "SkTextBlob.h"
+
+#include "SkDevice.h"
+
+const SkTextChunk* SkTextChunk::Create(const uint16_t* glyphs, size_t count, const SkPaint& paint,
+ const SkRect* bounds) {
+ if (NULL == glyphs || 0 == count) {
+ return NULL;
+ }
+
+ return SkNEW_ARGS(SkTextChunk, (glyphs, count, paint, bounds));
+}
+
+const SkTextChunk* SkTextChunk::Create(const uint16_t* glyphs, size_t count, const SkScalar* pos,
+ const SkPaint& paint, const SkRect* bounds) {
+ if (NULL == glyphs || NULL == pos || 0 == count) {
+ return NULL;
+ }
+
+ return SkNEW_ARGS(SkTextChunk, (glyphs, count, pos, paint, bounds));
+}
+
+const SkTextChunk* SkTextChunk::Create(const uint16_t* glyphs, size_t count, const SkPoint* pos,
+ const SkPaint& paint, const SkRect* bounds) {
+ if (NULL == glyphs || NULL == pos || 0 == count) {
+ return NULL;
+ }
+
+ return SkNEW_ARGS(SkTextChunk, (glyphs, count, pos, paint, bounds));
+}
+
+SkTextChunk::SkTextChunk(const uint16_t *glyphs, size_t count, const SkPaint &paint,
+ const SkRect* bounds)
+ : fPos(NULL)
+ , fFont(paint)
+ , fScalarsPerPos(0) {
+
+ SkASSERT(SkPaint::kGlyphID_TextEncoding == paint.getTextEncoding());
+ init(glyphs, count, bounds, NULL);
+}
+
+SkTextChunk::SkTextChunk(const uint16_t *glyphs, size_t count, const SkScalar* pos,
+ const SkPaint &paint, const SkRect* bounds)
+ : fPos(NULL)
+ , fFont(paint)
+ , fScalarsPerPos(1) {
+
+ SkASSERT(SkPaint::kGlyphID_TextEncoding == paint.getTextEncoding());
+ init(glyphs, count, bounds, pos);
+}
+
+SkTextChunk::SkTextChunk(const uint16_t *glyphs, size_t count, const SkPoint* pos,
+ const SkPaint &paint, const SkRect* bounds)
+ : fPos(NULL)
+ , fFont(paint)
+ , fScalarsPerPos(2) {
+
+ SK_COMPILE_ASSERT(sizeof(SkScalar) * 2 == sizeof(SkPoint), point_is_two_scalars);
+
+ SkASSERT(SkPaint::kGlyphID_TextEncoding == paint.getTextEncoding());
+ init(glyphs, count, bounds, (const SkScalar*)pos);
mtklein 2014/08/14 14:00:42 Can't hurt to label this explicitly as reinterpret
f(malita) 2014/08/14 14:29:26 Done.
+}
+
+SkTextChunk::~SkTextChunk() {
+ sk_free(fGlyphs);
+ sk_free(fPos);
+}
+
+void SkTextChunk::init(const uint16_t* glyphs, size_t count, const SkRect* bounds,
+ const SkScalar* pos) {
+ SkASSERT(glyphs);
+ SkASSERT(count > 0);
+
+ size_t storageSize = sizeof(uint16_t) * count;
+
+ fGlyphCount = count;
+ fGlyphs = static_cast<uint16_t*>(sk_malloc_throw(storageSize));
+ memcpy(fGlyphs, glyphs, storageSize);
+
+ fBoundsDirty = NULL == bounds;
+ if (!fBoundsDirty) {
+ fBounds = *bounds;
+ }
+
+ if (NULL != pos) {
+ size_t posStorageSize = sizeof(SkScalar) * fScalarsPerPos * count;
+ fPos = static_cast<SkScalar*>(sk_malloc_throw(posStorageSize));
+ memcpy(fPos, pos, posStorageSize);
+ }
+}
+
+const SkRect& SkTextChunk::bounds() const {
+ if (fBoundsDirty) {
+ fFont.measureText(fGlyphs, fGlyphCount, &fBounds);
+ fBoundsDirty = false;
+ }
+
+ return fBounds;
+}
+
+void SkTextChunk::draw(SkBaseDevice* device, const SkDraw& draw, const SkPaint& paint) const {
+ SkASSERT(NULL != device);
+ SkASSERT(SkPaint::kGlyphID_TextEncoding == paint.getTextEncoding());
+
+ size_t length = sizeof(uint16_t) * fGlyphCount;
+ if (NULL != fPos) {
+ SkASSERT(1 == fScalarsPerPos || 2 == fScalarsPerPos);
+ device->drawPosText(draw, fGlyphs, length, fPos, 0, fScalarsPerPos, paint);
+ } else {
+ SkASSERT(0 == fScalarsPerPos);
+ device->drawText(draw, fGlyphs, length, 0, 0, paint);
+ }
+}
+
+const SkTextBlob* SkTextBlob::Create(const SkTextChunk *chunk) {
+ SkTDArray<const SkTextChunk*> chunks;
+ chunks.setReserve(1);
+ *chunks.append() = chunk;
+ return SkNEW_ARGS(SkTextBlob, (chunks));
+}
+
+const SkTextBlob* SkTextBlob::Create(const SkTDArray<const SkTextChunk*>& chunks) {
+ return SkNEW_ARGS(SkTextBlob, (chunks));
+}
+
+SkTextBlob::SkTextBlob(const SkTDArray<const SkTextChunk*>& chunks)
+ : fChunks(chunks)
+ , fUniqueID(SK_InvalidGenID) {
+}
+
+SkTextBlob::~SkTextBlob() {
+ Iter it(this);
+ while (const SkTextChunk* chunk = it.next()) {
+ SkDELETE(chunk);
+ }
+}
+
+uint32_t SkTextBlob::uniqueID() const {
+ static int32_t gTextBlobGenerationID; // = 0;
+
+ // do a loop in case our global wraps around, as we never want to
+ // return SK_InvalidGenID
+ while (SK_InvalidGenID == fUniqueID) {
+ fUniqueID = sk_atomic_inc(&gTextBlobGenerationID) + 1;
+ }
+
+ return fUniqueID;
+}
+
+SkTextBlobBuilder::SkTextBlobBuilder() {
+}
+
+SkTextBlobBuilder::~SkTextBlobBuilder() {
+ // unused chunks.
+ for (int i = 0; i < fChunks.count(); ++i) {
+ SkDELETE(fChunks[i]);
+ }
+}
+
+const SkTextBlob* SkTextBlobBuilder::build() {
+ const SkTextBlob* blob = SkTextBlob::Create(fChunks);
+ fChunks.rewind();
+ return blob;
+}
+
+void SkTextBlobBuilder::addChunk(const SkTextChunk* chunk) {
+ *fChunks.append() = chunk;
+}
+
+
« include/core/SkTextBlob.h ('K') | « src/core/SkRecords.h ('k') | src/pipe/SkGPipeWrite.cpp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698