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