blob: 92f78afd54c9abbfc01d742c91aa39b0f734dd3b [file] [log] [blame]
/*
* Copyright 2016 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "include/core/SkFont.h"
#include "include/core/SkFontTypes.h"
#include "include/core/SkPoint.h"
#include "include/core/SkScalar.h"
#include "include/core/SkTypes.h"
#include "include/private/base/SkTo.h"
#include "modules/skshaper/include/SkShaper.h"
#include "src/base/SkUTF.h"
#if !defined(SK_DISABLE_LEGACY_SKSHAPER_FUNCTIONS)
#include "include/core/SkFontMgr.h"
#endif
#include <cstdint>
#include <cstring>
#include <memory>
class SkShaperPrimitive : public SkShaper {
public:
SkShaperPrimitive() {}
private:
#if !defined(SK_DISABLE_LEGACY_SKSHAPER_FUNCTIONS)
void shape(const char* utf8, size_t utf8Bytes,
const SkFont& srcFont,
bool leftToRight,
SkScalar width,
RunHandler*) const override;
void shape(const char* utf8, size_t utf8Bytes,
FontRunIterator&,
BiDiRunIterator&,
ScriptRunIterator&,
LanguageRunIterator&,
SkScalar width,
RunHandler*) const override;
#endif
void shape(const char* utf8, size_t utf8Bytes,
FontRunIterator&,
BiDiRunIterator&,
ScriptRunIterator&,
LanguageRunIterator&,
const Feature*, size_t featureSize,
SkScalar width,
RunHandler*) const override;
};
static inline bool is_breaking_whitespace(SkUnichar c) {
switch (c) {
case 0x0020: // SPACE
//case 0x00A0: // NO-BREAK SPACE
case 0x1680: // OGHAM SPACE MARK
case 0x180E: // MONGOLIAN VOWEL SEPARATOR
case 0x2000: // EN QUAD
case 0x2001: // EM QUAD
case 0x2002: // EN SPACE (nut)
case 0x2003: // EM SPACE (mutton)
case 0x2004: // THREE-PER-EM SPACE (thick space)
case 0x2005: // FOUR-PER-EM SPACE (mid space)
case 0x2006: // SIX-PER-EM SPACE
case 0x2007: // FIGURE SPACE
case 0x2008: // PUNCTUATION SPACE
case 0x2009: // THIN SPACE
case 0x200A: // HAIR SPACE
case 0x200B: // ZERO WIDTH SPACE
case 0x202F: // NARROW NO-BREAK SPACE
case 0x205F: // MEDIUM MATHEMATICAL SPACE
case 0x3000: // IDEOGRAPHIC SPACE
//case 0xFEFF: // ZERO WIDTH NO-BREAK SPACE
return true;
default:
return false;
}
}
static size_t linebreak(const char text[], const char stop[],
const SkFont& font, SkScalar width,
SkScalar* advance,
size_t* trailing)
{
SkScalar accumulatedWidth = 0;
int glyphIndex = 0;
const char* start = text;
const char* wordStart = text;
bool prevWS = true;
*trailing = 0;
while (text < stop) {
const char* prevText = text;
SkUnichar uni = SkUTF::NextUTF8(&text, stop);
accumulatedWidth += advance[glyphIndex++];
bool currWS = is_breaking_whitespace(uni);
if (!currWS && prevWS) {
wordStart = prevText;
}
prevWS = currWS;
if (width < accumulatedWidth) {
bool consumeWhitespace = false;
if (currWS) {
// previous fit, put this and following whitespace in trailing
if (prevText == start) {
// don't put this in trailing if it's the first thing
prevText = text;
}
consumeWhitespace = true;
} else if (wordStart != start) {
// backup to the last whitespace that fit
text = wordStart;
} else if (prevText > start) {
// backup to just before the glyph that didn't fit
text = prevText;
} else {
// let it overflow, put any following whitespace in trailing
prevText = text;
consumeWhitespace = true;
}
if (consumeWhitespace) {
const char* next = text;
while (next < stop && is_breaking_whitespace(SkUTF::NextUTF8(&next, stop))) {
text = next;
}
if (trailing) {
*trailing = text - prevText;
}
}
break;
}
}
return text - start;
}
#if !defined(SK_DISABLE_LEGACY_SKSHAPER_FUNCTIONS)
void SkShaperPrimitive::shape(const char* utf8,
size_t utf8Bytes,
FontRunIterator& font,
BiDiRunIterator& bidi,
ScriptRunIterator& script,
LanguageRunIterator& lang,
SkScalar width,
RunHandler* handler) const {
return this->shape(utf8, utf8Bytes, font, bidi, script, lang, nullptr, 0, width, handler);
}
void SkShaperPrimitive::shape(const char* utf8,
size_t utf8Bytes,
const SkFont& font,
bool leftToRight,
SkScalar width,
RunHandler* handler) const {
std::unique_ptr<FontRunIterator> fontRuns(
MakeFontMgrRunIterator(utf8, utf8Bytes, font, nullptr));
if (!fontRuns) {
return;
}
// bidi, script, and lang are all unused so we can construct them with empty data.
TrivialBiDiRunIterator bidi{0, 0};
TrivialScriptRunIterator script{0, 0};
TrivialLanguageRunIterator lang{nullptr, 0};
return this->shape(utf8, utf8Bytes, *fontRuns, bidi, script, lang, nullptr, 0, width, handler);
}
#endif
void SkShaperPrimitive::shape(const char* utf8,
size_t utf8Bytes,
FontRunIterator& fontRuns,
BiDiRunIterator&,
ScriptRunIterator&,
LanguageRunIterator&,
const Feature*,
size_t,
SkScalar width,
RunHandler* handler) const {
SkFont font;
if (!fontRuns.atEnd()) {
fontRuns.consume();
font = fontRuns.currentFont();
}
SkASSERT(font.getTypeface());
int glyphCount = font.countText(utf8, utf8Bytes, SkTextEncoding::kUTF8);
if (glyphCount < 0) {
return;
}
std::unique_ptr<SkGlyphID[]> glyphs(new SkGlyphID[glyphCount]);
font.textToGlyphs(utf8, utf8Bytes, SkTextEncoding::kUTF8, glyphs.get(), glyphCount);
std::unique_ptr<SkScalar[]> advances(new SkScalar[glyphCount]);
font.getWidthsBounds(glyphs.get(), glyphCount, advances.get(), nullptr, nullptr);
size_t glyphOffset = 0;
size_t utf8Offset = 0;
do {
size_t bytesCollapsed;
size_t bytesConsumed = linebreak(utf8, utf8 + utf8Bytes, font, width,
advances.get() + glyphOffset, &bytesCollapsed);
size_t bytesVisible = bytesConsumed - bytesCollapsed;
size_t numGlyphs = SkUTF::CountUTF8(utf8, bytesVisible);
const RunHandler::RunInfo info = {
font,
0,
{ font.measureText(utf8, bytesVisible, SkTextEncoding::kUTF8), 0 },
numGlyphs,
RunHandler::Range(utf8Offset, bytesVisible)
};
handler->beginLine();
if (info.glyphCount) {
handler->runInfo(info);
}
handler->commitRunInfo();
if (info.glyphCount) {
const auto buffer = handler->runBuffer(info);
memcpy(buffer.glyphs, glyphs.get() + glyphOffset, info.glyphCount * sizeof(SkGlyphID));
SkPoint position = buffer.point;
for (size_t i = 0; i < info.glyphCount; ++i) {
buffer.positions[i] = position;
position.fX += advances[i + glyphOffset];
}
if (buffer.clusters) {
const char* txtPtr = utf8;
for (size_t i = 0; i < info.glyphCount; ++i) {
// Each character maps to exactly one glyph.
buffer.clusters[i] = SkToU32(txtPtr - utf8 + utf8Offset);
SkUTF::NextUTF8(&txtPtr, utf8 + utf8Bytes);
}
}
handler->commitRunBuffer(info);
}
handler->commitLine();
glyphOffset += SkUTF::CountUTF8(utf8, bytesConsumed);
utf8Offset += bytesConsumed;
utf8 += bytesConsumed;
utf8Bytes -= bytesConsumed;
} while (0 < utf8Bytes);
}
#if !defined(SK_DISABLE_LEGACY_SKSHAPER_FUNCTIONS)
std::unique_ptr<SkShaper> SkShaper::MakePrimitive() { return SkShapers::Primitive::PrimitiveText(); }
#endif
namespace SkShapers::Primitive {
std::unique_ptr<SkShaper> PrimitiveText() { return std::make_unique<SkShaperPrimitive>(); }
} // namespace SkShapers