• 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/label.h>
19 
20 #include <teeui/error.h>
21 #include <teeui/font_rendering.h>
22 
23 namespace teeui {
24 
draw(const PixelDrawer & drawPixel,const Box<pxs> & bounds,LineInfo * lineInfo)25 Error LabelImpl::draw(const PixelDrawer& drawPixel, const Box<pxs>& bounds, LineInfo* lineInfo) {
26     if (!font_) return Error::NotInitialized;
27 
28     Error error;
29     TextContext context;
30     std::tie(error, context) = TextContext::create();
31     if (error) return error;
32 
33     TextFace face;
34     std::tie(error, face) = context.loadFace(font_);
35     if (error) return error;
36 
37     error = face.setCharSizeInPix(fontSize());
38     if (error) return error;
39 
40     using intpxs = Coordinate<px, int64_t>;
41     Box<intpxs> intBounds(intpxs((int64_t)bounds.x().count()), intpxs((int64_t)bounds.y().count()),
42                           intpxs((int64_t)bounds.w().count()), intpxs((int64_t)bounds.h().count()));
43 
44     auto drawPixelBoundsEnforced =
45         makePixelDrawer([&, this](uint32_t x, uint32_t y, Color color) -> Error {
46             if (!intBounds.contains(Point<intpxs>(x, y))) {
47                 TEEUI_LOG << "Bounds: " << bounds << " Pixel: " << Point<pxs>(x, y) << ENDL;
48                 return Error::OutOfBoundsDrawing;
49             }
50             // combine the given alpha channel with the text color.
51             return drawPixel(x, y, (textColor_ & 0xffffff) | (color & 0xff000000));
52         });
53 
54 #ifdef DRAW_DEBUG_MARKERS
55     auto drawBox = [&](const Box<pxs>& box, Color c) {
56         for (int y = 0; y < box.h().count(); ++y) {
57             for (int x = 0; x < box.w().count(); ++x) {
58                 drawPixel(box.x().count() + x, box.y().count() + y, (c & 0xffffff) | 0x40000000);
59             }
60         }
61     };
62 
63     drawBox(bounds, 0xff);
64 #endif
65 
66     Point<pxs> pen = {0_px, 0_px};
67     auto textBegin = text_.begin();
68     optional<Box<pxs>> boundingBox;
69 
70     auto curLine = lineInfo->begin();
71     while (textBegin != text_.end()) {
72         if (curLine == lineInfo->end()) {
73             TEEUI_LOG << "lineInfo filled: lines=" << lineInfo->size_ << " textId=" << textId_
74                       << ENDL;
75             return Error::OutOfMemory;
76         }
77 
78         auto lineEnd = textBegin;
79 
80         while (!isNewline(lineEnd.codePoint()) && lineEnd != text_.end()) {
81             lineEnd++;
82         }
83 
84         Box<pxs> bBox;
85         std::tie(error, bBox, curLine->lineText) =
86             findLongestWordSequence(&face, text_t(*textBegin, *lineEnd), bounds);
87         if (error) return error;
88 
89         pen = {-bBox.x(), pen.y()};
90 
91         // check horizontal justification to set pen value
92         switch (horizontalTextAlignment_) {
93         case Alignment::LEFT:
94         case Alignment::TOP:
95         case Alignment::BOTTOM:
96             break;
97         case Alignment::CENTER:
98             pen += {(bounds.w() - bBox.w()) / 2.0_px, 0};
99             break;
100         case Alignment::RIGHT:
101             pen += {bounds.w() - bBox.w(), 0};
102             break;
103         }
104 
105         curLine->lineStart = pen;
106         bBox.translateSelf(pen);
107 
108         if (boundingBox)
109             boundingBox = boundingBox->merge(bBox);
110         else
111             boundingBox = bBox;
112 
113         // Set start position for next loop, skipping a newline if found
114         textBegin = curLine->lineText.end();
115         if (isNewline(textBegin.codePoint())) {
116             textBegin++;
117         }
118 
119         pen += {0_px, lineHeight_};
120         ++curLine;
121     }
122 
123     if (!boundingBox) return Error::OK;
124 
125     TEEUI_LOG << "BoundingBox: " << *boundingBox << " Bounds: " << bounds << ENDL;
126     Point<pxs> offset = bounds.topLeft();
127     offset -= {0, boundingBox->y()};
128     TEEUI_LOG << "Offset: " << offset << ENDL;
129 
130     if (verticalTextAlignment_ == Alignment::CENTER)
131         offset += {0, (bounds.h() - boundingBox->h()) / 2.0_px};
132     else if (verticalTextAlignment_ == Alignment::BOTTOM)
133         offset += {0, (bounds.h() - boundingBox->h())};
134 
135     auto lineEnd = curLine;
136     curLine = lineInfo->begin();
137 
138 #ifdef DRAW_DEBUG_MARKERS
139     drawBox(boundingBox->translate(offset), 0xff00);
140     auto p = offset + curLine->lineStart;
141     drawPixel(p.x().count(), p.y().count(), 0xffff0000);
142 #endif
143 
144     while (curLine != lineEnd) {
145         if (auto error = drawText(&face, curLine->lineText, drawPixelBoundsEnforced,
146                                   curLine->lineStart + offset)) {
147             TEEUI_LOG << "drawText returned " << error << ENDL;
148             return error;
149         }
150         ++curLine;
151     }
152     return Error::OK;
153 }
154 
hit(const Event & event,const Box<pxs> & bounds)155 Error LabelImpl::hit(const Event& event, const Box<pxs>& bounds) {
156     using intpxs = Coordinate<px, int64_t>;
157     if (bounds.contains(Point<intpxs>(event.x_, event.y_))) {
158         optional<CallbackEvent> callback = getCB();
159         if (callback) {
160             return callback.value()(event);
161         }
162     }
163     return Error::OK;
164 }
165 }  // namespace teeui
166