1 /*
2 *
3 * Copyright 2019, The Android Open Source Project
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 */
17
18 #include <teeui/font_rendering.h>
19
20 namespace teeui {
21
isBreakable(unsigned long codePoint)22 bool isBreakable(unsigned long codePoint) {
23 switch (codePoint) {
24 case 9:
25 case 0xA:
26 case 0xB:
27 case 0xC:
28 case 0xD:
29 case 0x20:
30 case 0x85:
31 case 0x1680: // Ogham Space Mark
32 case 0x180E: // Mongolian Vowel Separator
33 case 0x2000: // EN Quad
34 case 0x2001: // EM Quad
35 case 0x2002: // EN Space
36 case 0x2003: // EM Space
37 case 0x2004: // three per em space
38 case 0x2005: // four per em space
39 case 0x2006: // six per em space
40 case 0x2008: // Punctuation space
41 case 0x2009: // Thin Space
42 case 0x200A: // Hair Space
43 case 0x200B: // Zero Width Space
44 case 0x200C: // Zero Width Non-Joiner
45 case 0x200D: // Zero Width Joiner
46 case 0x2028: // Line Separator
47 case 0x2029: // Paragraph Separator
48 case 0x205F: // Medium Mathematical Space
49 case 0x3000: // Ideographic Space
50 return true;
51 default:
52 return false;
53 }
54 }
55
isNewline(unsigned long codePoint)56 bool isNewline(unsigned long codePoint) {
57 return codePoint == '\n';
58 }
59
setCharSize(signed long char_size,unsigned int dpi)60 Error TextFace::setCharSize(signed long char_size, unsigned int dpi) {
61 if (!face_) return Error::NotInitialized;
62 auto error = FT_Set_Char_Size(*face_, 0, char_size, 0, dpi);
63 if (error) return Error::CharSizeNotSet;
64 return Error::OK;
65 }
66
setCharSizeInPix(pxs size)67 Error TextFace::setCharSizeInPix(pxs size) {
68 if (!face_) return Error::NotInitialized;
69 if (FT_Set_Pixel_Sizes(*face_, 0, size.count())) {
70 return Error::CharSizeNotSet;
71 }
72 return Error::OK;
73 }
74
getCharIndex(unsigned long codePoint)75 GlyphIndex TextFace::getCharIndex(unsigned long codePoint) {
76 if (!face_) return 0;
77 return FT_Get_Char_Index(*face_, codePoint);
78 }
79
loadGlyph(GlyphIndex index)80 Error TextFace::loadGlyph(GlyphIndex index) {
81 if (!face_) return Error::NotInitialized;
82 if (FT_Load_Glyph(*face_, index, FT_LOAD_DEFAULT)) {
83 return Error::GlyphNotLoaded;
84 }
85 return Error::OK;
86 }
87
renderGlyph()88 Error TextFace::renderGlyph() {
89 if (!face_) return Error::NotInitialized;
90 if (FT_Render_Glyph(face_->glyph, FT_RENDER_MODE_NORMAL)) {
91 return Error::GlyphNotRendered;
92 }
93 return Error::OK;
94 }
95
advance() const96 Vec2d<pxs> TextFace::advance() const {
97 return Vec2d<pxs>(face_->glyph->advance.x / 64.0, face_->glyph->advance.y / 64.0);
98 }
99
kern(GlyphIndex previous) const100 Vec2d<pxs> TextFace::kern(GlyphIndex previous) const {
101 FT_Vector offset = {0, 0};
102 if (hasKerning_ && previous) {
103 if (!FT_Get_Kerning(*face_, previous, face_->glyph->glyph_index, FT_KERNING_DEFAULT,
104 &offset)) {
105 offset = {0, 0};
106 }
107 }
108 return {offset.x / 64.0, offset.y / 64.0};
109 }
110
getGlyphBBox() const111 optional<Box<pxs>> TextFace::getGlyphBBox() const {
112 FT_Glyph glyph;
113 if (!FT_Get_Glyph(face_->glyph, &glyph)) {
114 FT_BBox cbox;
115 FT_Glyph_Get_CBox(glyph, ft_glyph_bbox_pixels, &cbox);
116 FT_Done_Glyph(glyph);
117 return {{cbox.xMin, -cbox.yMax, cbox.xMax - cbox.xMin, cbox.yMax - cbox.yMin}};
118 }
119 return {};
120 }
121
create()122 std::tuple<Error, TextContext> TextContext::create() {
123 std::tuple<Error, TextContext> result;
124 auto& [rc, lib] = result;
125 rc = Error::NotInitialized;
126 FT_Library library = nullptr;
127 auto error = FT_Init_FreeType(&library);
128 if (error) return result;
129 rc = Error::OK;
130 lib.library_ = Handle(library);
131 return result;
132 }
133
134 std::tuple<Error, Box<pxs>, UTF8Range<const char*>>
findLongestWordSequence(TextFace * face,const UTF8Range<const char * > & text,const Box<pxs> & boundingBox)135 findLongestWordSequence(TextFace* face, const UTF8Range<const char*>& text,
136 const Box<pxs>& boundingBox) {
137 std::tuple<Error, Box<pxs>, UTF8Range<const char*>> result;
138 auto& [error, bBox, resultRange] = result;
139
140 Vec2d<pxs> pen = {0, 0};
141 bBox = {pen, {0, 0}};
142
143 GlyphIndex previous = 0;
144 UTF8WordRange<const char*> wordRange(text);
145 auto rangeBegin = wordRange.begin();
146 if (isBreakable((*rangeBegin).codePoint())) {
147 ++rangeBegin;
148 }
149 auto wordEnd = rangeBegin;
150 auto wordStart = wordEnd;
151 ++wordEnd;
152 auto sequenceEnd = *wordStart;
153 auto lastBreakableBegin = *wordStart;
154 Box<pxs> currentBox = bBox;
155 Box<pxs> currentFullWordBox = bBox;
156 while (wordStart != wordRange.end()) {
157 auto codePoint = UTF8Range<const char*>::codePoint(**wordStart);
158 if (isBreakable(codePoint)) {
159 lastBreakableBegin = *wordStart;
160 currentFullWordBox = currentBox;
161 }
162
163 Box<pxs> workingBox = currentBox;
164 auto c = *wordStart;
165 bool exceedsBoundingBox = false;
166 while (c != *wordEnd) {
167 codePoint = c.codePoint();
168 auto gindex = face->getCharIndex(codePoint);
169 if (gindex == 0) {
170 error = Error::GlyphNotLoaded;
171 return result;
172 }
173 error = face->loadGlyph(gindex);
174 if (error != Error::OK) return result;
175 pen += face->kern(previous);
176 if (auto gBox = face->getGlyphBBox()) {
177 TEEUI_LOG << "Glyph Box: " << *gBox << ENDL;
178 workingBox = workingBox.merge(gBox->translateSelf(pen));
179 TEEUI_LOG << "WorkingBox: " << workingBox << ENDL;
180 } else {
181 error = Error::BBoxComputation;
182 return result;
183 }
184 pen += face->advance();
185 previous = gindex;
186 ++c;
187 if (workingBox.fitsInside(boundingBox)) {
188 currentBox = workingBox;
189 sequenceEnd = c;
190 } else {
191 exceedsBoundingBox = true;
192 TEEUI_LOG << "exceeding bbox" << ENDL;
193 break;
194 }
195 }
196 if (exceedsBoundingBox) break;
197 wordStart = wordEnd;
198 ++wordEnd;
199 }
200 if (wordStart == wordRange.end()) {
201 bBox = currentBox;
202 resultRange = {**rangeBegin, *text.end()};
203 TEEUI_LOG << "full range" << ENDL;
204 } else if (*rangeBegin != lastBreakableBegin) {
205 bBox = currentFullWordBox;
206 resultRange = {**rangeBegin, *lastBreakableBegin};
207 TEEUI_LOG << "partial range:" << ENDL;
208 } else {
209 bBox = currentBox;
210 resultRange = {**rangeBegin, *sequenceEnd};
211 TEEUI_LOG << "unbreakable" << ENDL;
212 }
213 error = Error::OK;
214 return result;
215 }
216
drawText(TextFace * face,const UTF8Range<const char * > & text,const PixelDrawer & drawPixel,PxPoint pen)217 Error drawText(TextFace* face, const UTF8Range<const char*>& text, const PixelDrawer& drawPixel,
218 PxPoint pen) {
219 Error error;
220
221 for (auto c : text) {
222 auto codePoint = UTF8Range<const char*>::codePoint(c);
223 auto gindex = face->getCharIndex(codePoint);
224 error = face->loadGlyph(gindex);
225 if (error == Error::OK) error = face->renderGlyph();
226 if (error == Error::OK) error = face->drawGlyph(pen, drawPixel);
227 if (error != Error::OK) return error;
228
229 pen += face->advance();
230 }
231 return Error::OK;
232 }
233
234 } // namespace teeui
235