• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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