impl breakText in SkFont
Bug: skia: Change-Id: I449e0b29f98dc194a335be476f6dbdd69efad16b Reviewed-on: https://skia-review.googlesource.com/c/174504 Commit-Queue: Mike Reed <reed@google.com> Reviewed-by: Mike Reed <reed@google.com>
This commit is contained in:
parent
ef211e600c
commit
70fbc36512
@ -347,59 +347,8 @@ SkScalar SkPaint::measureText(const void* textData, size_t length, SkRect* bound
|
||||
|
||||
size_t SkPaint::breakText(const void* textD, size_t length, SkScalar maxWidth,
|
||||
SkScalar* measuredWidth) const {
|
||||
if (0 == length || 0 >= maxWidth) {
|
||||
if (measuredWidth) {
|
||||
*measuredWidth = 0;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (0 == fTextSize) {
|
||||
if (measuredWidth) {
|
||||
*measuredWidth = 0;
|
||||
}
|
||||
return length;
|
||||
}
|
||||
|
||||
SkASSERT(textD != nullptr);
|
||||
const char* text = (const char*)textD;
|
||||
const char* stop = text + length;
|
||||
|
||||
SkCanonicalizePaint canon(*this);
|
||||
const SkPaint& paint = canon.getPaint();
|
||||
SkScalar scale = canon.getScale();
|
||||
|
||||
// adjust max in case we changed the textSize in paint
|
||||
if (scale) {
|
||||
maxWidth /= scale;
|
||||
}
|
||||
|
||||
const SkFont font = SkFont::LEGACY_ExtractFromPaint(paint);
|
||||
auto cache = SkStrikeCache::FindOrCreateStrikeWithNoDeviceExclusive(font, paint);
|
||||
|
||||
SkFontPriv::GlyphCacheProc glyphCacheProc = SkFontPriv::GetGlyphCacheProc(
|
||||
static_cast<SkTextEncoding>(paint.getTextEncoding()), false);
|
||||
SkScalar width = 0;
|
||||
|
||||
while (text < stop) {
|
||||
const char* curr = text;
|
||||
SkScalar x = advance(glyphCacheProc(cache.get(), &text, stop));
|
||||
if ((width += x) > maxWidth) {
|
||||
width -= x;
|
||||
text = curr;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (measuredWidth) {
|
||||
if (scale) {
|
||||
width *= scale;
|
||||
}
|
||||
*measuredWidth = width;
|
||||
}
|
||||
|
||||
// return the number of bytes measured
|
||||
return text - stop + length;
|
||||
return SkFont::LEGACY_ExtractFromPaint(*this).breakText(textD, length,
|
||||
(SkTextEncoding)this->getTextEncoding(), maxWidth, measuredWidth);
|
||||
}
|
||||
|
||||
SkScalar SkPaint::getFontMetrics(SkFontMetrics* metrics) const {
|
||||
|
@ -7,35 +7,19 @@
|
||||
|
||||
#include "SkAutoMalloc.h"
|
||||
#include "SkFont.h"
|
||||
#include "SkPaint.h"
|
||||
#include "Test.h"
|
||||
|
||||
static size_t break_text(const SkPaint& paint, const void* text, size_t length, SkScalar maxw,
|
||||
SkScalar* measuredw, skiatest::Reporter* reporter) {
|
||||
size_t n = paint.breakText(text, length, maxw, measuredw);
|
||||
SkScalar measuredw2;
|
||||
size_t n2 = SkFont::LEGACY_ExtractFromPaint(paint).breakText(text, length,
|
||||
(SkTextEncoding)paint.getTextEncoding(), maxw, &measuredw2);
|
||||
REPORTER_ASSERT(reporter, n == n2);
|
||||
if (measuredw) {
|
||||
REPORTER_ASSERT(reporter, *measuredw == measuredw2);
|
||||
}
|
||||
return n;
|
||||
}
|
||||
|
||||
static void test_monotonic(skiatest::Reporter* reporter,
|
||||
const SkPaint& paint,
|
||||
const char* msg) {
|
||||
static void test_monotonic(skiatest::Reporter* reporter, const SkFont& font, const char* msg) {
|
||||
const char* text = "sdfkljAKLDFJKEWkldfjlk#$%&sdfs.dsj";
|
||||
const size_t length = strlen(text);
|
||||
const SkScalar width = paint.measureText(text, length);
|
||||
const SkScalar width = font.measureText(text, length, kUTF8_SkTextEncoding);
|
||||
|
||||
SkScalar mm = 0;
|
||||
size_t nn = 0;
|
||||
const SkScalar step = SkMaxScalar(width / 10, SK_Scalar1);
|
||||
for (SkScalar w = 0; w <= width; w += step) {
|
||||
SkScalar m;
|
||||
const size_t n = break_text(paint, text, length, w, &m, reporter);
|
||||
const size_t n = font.breakText(text, length, kUTF8_SkTextEncoding, w, &m);
|
||||
|
||||
REPORTER_ASSERT(reporter, n <= length, msg);
|
||||
REPORTER_ASSERT(reporter, m <= width, msg);
|
||||
@ -56,44 +40,41 @@ static void test_monotonic(skiatest::Reporter* reporter,
|
||||
}
|
||||
}
|
||||
|
||||
static void test_eq_measure_text(skiatest::Reporter* reporter,
|
||||
const SkPaint& paint,
|
||||
static void test_eq_measure_text(skiatest::Reporter* reporter, const SkFont& font,
|
||||
const char* msg) {
|
||||
const char* text = "The ultimate measure of a man is not where he stands in moments of comfort "
|
||||
"and convenience, but where he stands at times of challenge and controversy.";
|
||||
const size_t length = strlen(text);
|
||||
const SkScalar width = paint.measureText(text, length);
|
||||
const SkScalar width = font.measureText(text, length, kUTF8_SkTextEncoding);
|
||||
|
||||
SkScalar mm;
|
||||
const size_t length2 = break_text(paint, text, length, width, &mm, reporter);
|
||||
const size_t length2 = font.breakText(text, length, kUTF8_SkTextEncoding, width, &mm);
|
||||
REPORTER_ASSERT(reporter, length2 == length, msg);
|
||||
REPORTER_ASSERT(reporter, mm == width, msg);
|
||||
}
|
||||
|
||||
static void test_long_text(skiatest::Reporter* reporter,
|
||||
const SkPaint& paint,
|
||||
const char* msg) {
|
||||
static void test_long_text(skiatest::Reporter* reporter, const SkFont& font, const char* msg) {
|
||||
static const int kSize = 16 * 1024;
|
||||
SkAutoMalloc block(kSize);
|
||||
memset(block.get(), 'a', kSize - 1);
|
||||
char* text = static_cast<char*>(block.get());
|
||||
text[kSize - 1] = '\0';
|
||||
const SkScalar width = paint.measureText(text, kSize);
|
||||
const SkScalar width = font.measureText(text, kSize, kUTF8_SkTextEncoding);
|
||||
|
||||
SkScalar mm;
|
||||
const size_t length = break_text(paint, text, kSize, width, &mm, reporter);
|
||||
const size_t length = font.breakText(text, kSize, kUTF8_SkTextEncoding, width, &mm);
|
||||
REPORTER_ASSERT(reporter, length == kSize, msg);
|
||||
REPORTER_ASSERT(reporter, mm == width, msg);
|
||||
}
|
||||
|
||||
DEF_TEST(PaintBreakText, reporter) {
|
||||
SkPaint paint;
|
||||
test_monotonic(reporter, paint, "default");
|
||||
test_eq_measure_text(reporter, paint, "default");
|
||||
test_long_text(reporter, paint, "default");
|
||||
paint.setTextSize(SkIntToScalar(1 << 17));
|
||||
test_monotonic(reporter, paint, "huge text size");
|
||||
test_eq_measure_text(reporter, paint, "huge text size");
|
||||
paint.setTextSize(0);
|
||||
test_monotonic(reporter, paint, "zero text size");
|
||||
SkFont font;
|
||||
test_monotonic(reporter, font, "default");
|
||||
test_eq_measure_text(reporter, font, "default");
|
||||
test_long_text(reporter, font, "default");
|
||||
font.setSize(SkIntToScalar(1 << 17));
|
||||
test_monotonic(reporter, font, "huge text size");
|
||||
test_eq_measure_text(reporter, font, "huge text size");
|
||||
font.setSize(0);
|
||||
test_monotonic(reporter, font, "zero text size");
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user