1 /*
2 * Copyright 2016 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8 #include "include/core/SkFontMetrics.h"
9 #include "include/core/SkStream.h"
10 #include "include/core/SkTypeface.h"
11 #include "include/private/SkTo.h"
12 #include "modules/skshaper/include/SkShaper.h"
13 #include "src/utils/SkUTF.h"
14
15 #ifndef USE_SKIA_TXT
16
17 class SkShaperPrimitive : public SkShaper {
18 public:
SkShaperPrimitive()19 SkShaperPrimitive() {}
20 private:
21 void shape(const char* utf8, size_t utf8Bytes,
22 const SkFont& srcFont,
23 bool leftToRight,
24 SkScalar width,
25 RunHandler*) const override;
26
27 void shape(const char* utf8, size_t utf8Bytes,
28 FontRunIterator&,
29 BiDiRunIterator&,
30 ScriptRunIterator&,
31 LanguageRunIterator&,
32 SkScalar width,
33 RunHandler*) const override;
34
35 void shape(const char* utf8, size_t utf8Bytes,
36 FontRunIterator&,
37 BiDiRunIterator&,
38 ScriptRunIterator&,
39 LanguageRunIterator&,
40 const Feature*, size_t featureSize,
41 SkScalar width,
42 RunHandler*) const override;
43 };
44
MakePrimitive()45 std::unique_ptr<SkShaper> SkShaper::MakePrimitive() {
46 return std::make_unique<SkShaperPrimitive>();
47 }
48
is_breaking_whitespace(SkUnichar c)49 static inline bool is_breaking_whitespace(SkUnichar c) {
50 switch (c) {
51 case 0x0020: // SPACE
52 //case 0x00A0: // NO-BREAK SPACE
53 case 0x1680: // OGHAM SPACE MARK
54 case 0x180E: // MONGOLIAN VOWEL SEPARATOR
55 case 0x2000: // EN QUAD
56 case 0x2001: // EM QUAD
57 case 0x2002: // EN SPACE (nut)
58 case 0x2003: // EM SPACE (mutton)
59 case 0x2004: // THREE-PER-EM SPACE (thick space)
60 case 0x2005: // FOUR-PER-EM SPACE (mid space)
61 case 0x2006: // SIX-PER-EM SPACE
62 case 0x2007: // FIGURE SPACE
63 case 0x2008: // PUNCTUATION SPACE
64 case 0x2009: // THIN SPACE
65 case 0x200A: // HAIR SPACE
66 case 0x200B: // ZERO WIDTH SPACE
67 case 0x202F: // NARROW NO-BREAK SPACE
68 case 0x205F: // MEDIUM MATHEMATICAL SPACE
69 case 0x3000: // IDEOGRAPHIC SPACE
70 //case 0xFEFF: // ZERO WIDTH NO-BREAK SPACE
71 return true;
72 default:
73 return false;
74 }
75 }
76
linebreak(const char text[],const char stop[],const SkFont & font,SkScalar width,SkScalar * advance,size_t * trailing)77 static size_t linebreak(const char text[], const char stop[],
78 const SkFont& font, SkScalar width,
79 SkScalar* advance,
80 size_t* trailing)
81 {
82 SkScalar accumulatedWidth = 0;
83 int glyphIndex = 0;
84 const char* start = text;
85 const char* wordStart = text;
86 bool prevWS = true;
87 *trailing = 0;
88
89 while (text < stop) {
90 const char* prevText = text;
91 SkUnichar uni = SkUTF::NextUTF8(&text, stop);
92 accumulatedWidth += advance[glyphIndex++];
93 bool currWS = is_breaking_whitespace(uni);
94
95 if (!currWS && prevWS) {
96 wordStart = prevText;
97 }
98 prevWS = currWS;
99
100 if (width < accumulatedWidth) {
101 bool consumeWhitespace = false;
102 if (currWS) {
103 // previous fit, put this and following whitespace in trailing
104 if (prevText == start) {
105 // don't put this in trailing if it's the first thing
106 prevText = text;
107 }
108 consumeWhitespace = true;
109 } else if (wordStart != start) {
110 // backup to the last whitespace that fit
111 text = wordStart;
112 } else if (prevText > start) {
113 // backup to just before the glyph that didn't fit
114 text = prevText;
115 } else {
116 // let it overflow, put any following whitespace in trailing
117 prevText = text;
118 consumeWhitespace = true;
119 }
120 if (consumeWhitespace) {
121 const char* next = text;
122 while (next < stop && is_breaking_whitespace(SkUTF::NextUTF8(&next, stop))) {
123 text = next;
124 }
125 if (trailing) {
126 *trailing = text - prevText;
127 }
128 }
129 break;
130 }
131 }
132
133 return text - start;
134 }
135
shape(const char * utf8,size_t utf8Bytes,FontRunIterator & font,BiDiRunIterator & bidi,ScriptRunIterator &,LanguageRunIterator &,SkScalar width,RunHandler * handler) const136 void SkShaperPrimitive::shape(const char* utf8, size_t utf8Bytes,
137 FontRunIterator& font,
138 BiDiRunIterator& bidi,
139 ScriptRunIterator&,
140 LanguageRunIterator&,
141 SkScalar width,
142 RunHandler* handler) const
143 {
144 SkFont skfont;
145 if (!font.atEnd()) {
146 font.consume();
147 skfont = font.currentFont();
148 } else {
149 skfont.setTypeface(sk_ref_sp(skfont.getTypefaceOrDefault()));
150 }
151 SkASSERT(skfont.getTypeface());
152 bool skbidi = 0;
153 if (!bidi.atEnd()) {
154 bidi.consume();
155 skbidi = (bidi.currentLevel() % 2) == 0;
156 }
157 return this->shape(utf8, utf8Bytes, skfont, skbidi, width, handler);
158 }
159
shape(const char * utf8,size_t utf8Bytes,FontRunIterator & font,BiDiRunIterator & bidi,ScriptRunIterator &,LanguageRunIterator &,const Feature *,size_t,SkScalar width,RunHandler * handler) const160 void SkShaperPrimitive::shape(const char* utf8, size_t utf8Bytes,
161 FontRunIterator& font,
162 BiDiRunIterator& bidi,
163 ScriptRunIterator&,
164 LanguageRunIterator&,
165 const Feature*, size_t,
166 SkScalar width,
167 RunHandler* handler) const {
168 font.consume();
169 SkASSERT(font.currentFont().getTypeface());
170 bidi.consume();
171 return this->shape(utf8, utf8Bytes, font.currentFont(), (bidi.currentLevel() % 2) == 0,
172 width, handler);
173 }
174
shape(const char * utf8,size_t utf8Bytes,const SkFont & font,bool leftToRight,SkScalar width,RunHandler * handler) const175 void SkShaperPrimitive::shape(const char* utf8, size_t utf8Bytes,
176 const SkFont& font,
177 bool leftToRight,
178 SkScalar width,
179 RunHandler* handler) const {
180 sk_ignore_unused_variable(leftToRight);
181
182 int glyphCount = font.countText(utf8, utf8Bytes, SkTextEncoding::kUTF8);
183 if (glyphCount < 0) {
184 return;
185 }
186
187 std::unique_ptr<SkGlyphID[]> glyphs(new SkGlyphID[glyphCount]);
188 font.textToGlyphs(utf8, utf8Bytes, SkTextEncoding::kUTF8, glyphs.get(), glyphCount);
189
190 std::unique_ptr<SkScalar[]> advances(new SkScalar[glyphCount]);
191 font.getWidthsBounds(glyphs.get(), glyphCount, advances.get(), nullptr, nullptr);
192
193 size_t glyphOffset = 0;
194 size_t utf8Offset = 0;
195 do {
196 size_t bytesCollapsed;
197 size_t bytesConsumed = linebreak(utf8, utf8 + utf8Bytes, font, width,
198 advances.get() + glyphOffset, &bytesCollapsed);
199 size_t bytesVisible = bytesConsumed - bytesCollapsed;
200
201 size_t numGlyphs = SkUTF::CountUTF8(utf8, bytesVisible);
202 const RunHandler::RunInfo info = {
203 font,
204 0,
205 { font.measureText(utf8, bytesVisible, SkTextEncoding::kUTF8), 0 },
206 numGlyphs,
207 RunHandler::Range(utf8Offset, bytesVisible)
208 };
209 handler->beginLine();
210 if (info.glyphCount) {
211 handler->runInfo(info);
212 }
213 handler->commitRunInfo();
214 if (info.glyphCount) {
215 const auto buffer = handler->runBuffer(info);
216
217 memcpy(buffer.glyphs, glyphs.get() + glyphOffset, info.glyphCount * sizeof(SkGlyphID));
218 SkPoint position = buffer.point;
219 for (size_t i = 0; i < info.glyphCount; ++i) {
220 buffer.positions[i] = position;
221 position.fX += advances[i + glyphOffset];
222 }
223 if (buffer.clusters) {
224 const char* txtPtr = utf8;
225 for (size_t i = 0; i < info.glyphCount; ++i) {
226 // Each character maps to exactly one glyph.
227 buffer.clusters[i] = SkToU32(txtPtr - utf8 + utf8Offset);
228 SkUTF::NextUTF8(&txtPtr, utf8 + utf8Bytes);
229 }
230 }
231 handler->commitRunBuffer(info);
232 }
233 handler->commitLine();
234
235 glyphOffset += SkUTF::CountUTF8(utf8, bytesConsumed);
236 utf8Offset += bytesConsumed;
237 utf8 += bytesConsumed;
238 utf8Bytes -= bytesConsumed;
239 } while (0 < utf8Bytes);
240
241 return;
242 }
243
244 #endif
245