blob: f0d76c646dc077190e41a729201240a9a1c07c30 [file] [log] [blame]
Dees_Troy51a0e822012-09-05 15:24:24 -04001// console.cpp - GUIConsole object
2
3#include <stdarg.h>
4#include <stdio.h>
5#include <stdlib.h>
6#include <string.h>
7#include <fcntl.h>
8#include <sys/reboot.h>
9#include <sys/stat.h>
10#include <sys/time.h>
11#include <sys/mman.h>
12#include <sys/types.h>
13#include <sys/ioctl.h>
14#include <time.h>
15#include <unistd.h>
16#include <stdlib.h>
17
18#include <string>
19
20extern "C" {
Dees_Troy2673cec2013-04-02 20:22:16 +000021#include "../twcommon.h"
Dees_Troy51a0e822012-09-05 15:24:24 -040022#include "../minuitwrp/minui.h"
Dees_Troy51a0e822012-09-05 15:24:24 -040023}
24
25#include "rapidxml.hpp"
26#include "objects.hpp"
27
28
29static std::vector<std::string> gConsole;
30
31extern "C" void gui_print(const char *fmt, ...)
32{
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020033 char buf[512]; // We're going to limit a single request to 512 bytes
Dees_Troy51a0e822012-09-05 15:24:24 -040034
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020035 va_list ap;
36 va_start(ap, fmt);
37 vsnprintf(buf, 512, fmt, ap);
38 va_end(ap);
Dees_Troy51a0e822012-09-05 15:24:24 -040039
Dees_Troy32c8eb82012-09-11 15:28:06 -040040 fputs(buf, stdout);
41
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020042 char *start, *next;
Dees_Troy51a0e822012-09-05 15:24:24 -040043
44 if (buf[0] == '\n' && strlen(buf) < 2) {
45 // This prevents the double lines bug seen in the console during zip installs
46 return;
47 }
48
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020049 for (start = next = buf; *next != '\0'; next++)
50 {
51 if (*next == '\n')
52 {
53 *next = '\0';
54 next++;
Dees_Troy51a0e822012-09-05 15:24:24 -040055
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020056 std::string line = start;
57 gConsole.push_back(line);
58 start = next;
Dees_Troy51a0e822012-09-05 15:24:24 -040059
60 // Handle the normal \n\0 case
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020061 if (*next == '\0')
Dees_Troy51a0e822012-09-05 15:24:24 -040062 return;
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020063 }
64 }
65 std::string line = start;
66 gConsole.push_back(line);
67 return;
Dees_Troy51a0e822012-09-05 15:24:24 -040068}
69
70extern "C" void gui_print_overwrite(const char *fmt, ...)
71{
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020072 char buf[512]; // We're going to limit a single request to 512 bytes
Dees_Troy51a0e822012-09-05 15:24:24 -040073
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020074 va_list ap;
75 va_start(ap, fmt);
76 vsnprintf(buf, 512, fmt, ap);
77 va_end(ap);
Dees_Troy51a0e822012-09-05 15:24:24 -040078
Dees_Troy32c8eb82012-09-11 15:28:06 -040079 fputs(buf, stdout);
80
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020081 // Pop the last line, and we can continue
82 if (!gConsole.empty()) gConsole.pop_back();
Dees_Troy51a0e822012-09-05 15:24:24 -040083
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020084 char *start, *next;
85 for (start = next = buf; *next != '\0'; next++)
86 {
87 if (*next == '\n')
88 {
89 *next = '\0';
90 next++;
91
92 std::string line = start;
93 gConsole.push_back(line);
94 start = next;
Dees_Troy51a0e822012-09-05 15:24:24 -040095
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020096 // Handle the normal \n\0 case
97 if (*next == '\0')
Dees_Troy51a0e822012-09-05 15:24:24 -040098 return;
Vojtech Bocekfafb0c52013-07-25 22:53:02 +020099 }
100 }
101 std::string line = start;
102 gConsole.push_back(line);
103 return;
Dees_Troy51a0e822012-09-05 15:24:24 -0400104}
105
Vojtech Bocekede51c52014-02-07 23:58:09 +0100106GUIConsole::GUIConsole(xml_node<>* node) : GUIObject(node)
Dees_Troy51a0e822012-09-05 15:24:24 -0400107{
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200108 xml_attribute<>* attr;
109 xml_node<>* child;
Dees_Troy51a0e822012-09-05 15:24:24 -0400110
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200111 mFont = NULL;
112 mCurrentLine = -1;
113 memset(&mForegroundColor, 255, sizeof(COLOR));
114 memset(&mBackgroundColor, 0, sizeof(COLOR));
115 mBackgroundColor.alpha = 255;
116 memset(&mScrollColor, 0x08, sizeof(COLOR));
117 mScrollColor.alpha = 255;
118 mLastCount = 0;
119 mSlideout = 0;
Dees Troy31218ec2014-02-25 20:35:56 +0000120 RenderCount = 0;
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200121 mSlideoutState = hidden;
Dees Troy31218ec2014-02-25 20:35:56 +0000122 mRender = true;
Dees_Troy51a0e822012-09-05 15:24:24 -0400123
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200124 mRenderX = 0; mRenderY = 0; mRenderW = gr_fb_width(); mRenderH = gr_fb_height();
Dees_Troy51a0e822012-09-05 15:24:24 -0400125
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200126 if (!node)
127 {
128 mSlideoutX = 0; mSlideoutY = 0; mSlideoutW = 0; mSlideoutH = 0;
129 mConsoleX = 0; mConsoleY = 0; mConsoleW = gr_fb_width(); mConsoleH = gr_fb_height();
130 }
131 else
132 {
133 child = node->first_node("font");
134 if (child)
135 {
136 attr = child->first_attribute("resource");
137 if (attr)
138 mFont = PageManager::FindResource(attr->value());
139 }
Dees_Troy51a0e822012-09-05 15:24:24 -0400140
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200141 child = node->first_node("color");
142 if (child)
143 {
144 attr = child->first_attribute("foreground");
145 if (attr)
146 {
147 std::string color = attr->value();
148 ConvertStrToColor(color, &mForegroundColor);
149 }
150 attr = child->first_attribute("background");
151 if (attr)
152 {
153 std::string color = attr->value();
154 ConvertStrToColor(color, &mBackgroundColor);
155 }
156 attr = child->first_attribute("scroll");
157 if (attr)
158 {
159 std::string color = attr->value();
160 ConvertStrToColor(color, &mScrollColor);
161 }
162 }
Dees_Troy51a0e822012-09-05 15:24:24 -0400163
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200164 // Load the placement
165 LoadPlacement(node->first_node("placement"), &mConsoleX, &mConsoleY, &mConsoleW, &mConsoleH);
Dees_Troy51a0e822012-09-05 15:24:24 -0400166
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200167 child = node->first_node("slideout");
168 if (child)
169 {
170 mSlideout = 1;
171 LoadPlacement(child, &mSlideoutX, &mSlideoutY);
Dees_Troy51a0e822012-09-05 15:24:24 -0400172
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200173 attr = child->first_attribute("resource");
174 if (attr) mSlideoutImage = PageManager::FindResource(attr->value());
Dees_Troy51a0e822012-09-05 15:24:24 -0400175
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200176 if (mSlideoutImage && mSlideoutImage->GetResource())
177 {
178 mSlideoutW = gr_get_width(mSlideoutImage->GetResource());
179 mSlideoutH = gr_get_height(mSlideoutImage->GetResource());
180 }
181 }
182 }
Dees_Troy51a0e822012-09-05 15:24:24 -0400183
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200184 gr_getFontDetails(mFont, &mFontHeight, NULL);
185 SetActionPos(mRenderX, mRenderY, mRenderW, mRenderH);
186 SetRenderPos(mConsoleX, mConsoleY);
187 return;
Dees_Troy51a0e822012-09-05 15:24:24 -0400188}
189
190int GUIConsole::RenderSlideout(void)
191{
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200192 if (!mSlideoutImage || !mSlideoutImage->GetResource())
193 return -1;
Dees_Troy51a0e822012-09-05 15:24:24 -0400194
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200195 gr_blit(mSlideoutImage->GetResource(), 0, 0, mSlideoutW, mSlideoutH, mSlideoutX, mSlideoutY);
196 return 0;
Dees_Troy51a0e822012-09-05 15:24:24 -0400197}
198
199int GUIConsole::RenderConsole(void)
200{
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200201 void* fontResource = NULL;
202 if (mFont)
203 fontResource = mFont->GetResource();
Dees_Troy51a0e822012-09-05 15:24:24 -0400204
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200205 // We fill the background
206 gr_color(mBackgroundColor.red, mBackgroundColor.green, mBackgroundColor.blue, 255);
207 gr_fill(mConsoleX, mConsoleY, mConsoleW, mConsoleH);
Dees_Troy51a0e822012-09-05 15:24:24 -0400208
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200209 gr_color(mScrollColor.red, mScrollColor.green, mScrollColor.blue, mScrollColor.alpha);
210 gr_fill(mConsoleX + (mConsoleW * 9 / 10), mConsoleY, (mConsoleW / 10), mConsoleH);
Dees_Troy51a0e822012-09-05 15:24:24 -0400211
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200212 // Render the lines
213 gr_color(mForegroundColor.red, mForegroundColor.green, mForegroundColor.blue, mForegroundColor.alpha);
Dees_Troy51a0e822012-09-05 15:24:24 -0400214
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200215 // Don't try to continue to render without data
Dees Troy31218ec2014-02-25 20:35:56 +0000216 int prevCount = mLastCount;
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200217 mLastCount = gConsole.size();
Dees Troy31218ec2014-02-25 20:35:56 +0000218 mRender = false;
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200219 if (mLastCount == 0)
220 return (mSlideout ? RenderSlideout() : 0);
Dees_Troy51a0e822012-09-05 15:24:24 -0400221
Dees Troy31218ec2014-02-25 20:35:56 +0000222 // Due to word wrap, figure out what / how the newly added text needs to be added to the render vector that is word wrapped
223 // Note, that multiple consoles on different GUI pages may be different widths or use different fonts, so the word wrapping
224 // may different in different console windows
225 for (int i = prevCount; i < mLastCount; i++) {
226 string curr_line = gConsole[i];
227 int line_char_width;
228 for(;;) {
229 line_char_width = gr_maxExW(curr_line.c_str(), fontResource, mConsoleW);
230 if (line_char_width < curr_line.size()) {
231 rConsole.push_back(curr_line.substr(0, line_char_width));
Vojtech Bocekc58846b2014-03-04 19:57:45 +0100232 curr_line = curr_line.substr(line_char_width, curr_line.size() - line_char_width);
Dees Troy31218ec2014-02-25 20:35:56 +0000233 } else {
234 rConsole.push_back(curr_line);
235 break;
236 }
237 }
238 }
239 RenderCount = rConsole.size();
240
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200241 // Find the start point
242 int start;
243 int curLine = mCurrentLine; // Thread-safing (Another thread updates this value)
244 if (curLine == -1)
245 {
Dees Troy31218ec2014-02-25 20:35:56 +0000246 start = RenderCount - mMaxRows;
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200247 }
248 else
249 {
Dees Troy31218ec2014-02-25 20:35:56 +0000250 if (curLine > (int) RenderCount)
251 curLine = (int) RenderCount;
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200252 if ((int) mMaxRows > curLine)
253 curLine = (int) mMaxRows;
254 start = curLine - mMaxRows;
255 }
Dees_Troy51a0e822012-09-05 15:24:24 -0400256
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200257 unsigned int line;
258 for (line = 0; line < mMaxRows; line++)
259 {
Dees Troy31218ec2014-02-25 20:35:56 +0000260 if ((start + (int) line) >= 0 && (start + (int) line) < (int) RenderCount)
261 gr_textExW(mConsoleX, mStartY + (line * mFontHeight), rConsole[start + line].c_str(), fontResource, mConsoleW + mConsoleX);
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200262 }
263 return (mSlideout ? RenderSlideout() : 0);
Dees_Troy51a0e822012-09-05 15:24:24 -0400264}
265
266int GUIConsole::Render(void)
267{
Vojtech Bocekede51c52014-02-07 23:58:09 +0100268 if(!isConditionTrue())
269 return 0;
270
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200271 if (mSlideout && mSlideoutState == hidden)
272 return RenderSlideout();
273
274 return RenderConsole();
Dees_Troy51a0e822012-09-05 15:24:24 -0400275}
276
277int GUIConsole::Update(void)
278{
Vojtech Bocekede51c52014-02-07 23:58:09 +0100279 if(!isConditionTrue())
280 return 0;
281
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200282 if (mSlideout && mSlideoutState != visible)
283 {
284 if (mSlideoutState == hidden)
285 return 0;
Dees_Troy51a0e822012-09-05 15:24:24 -0400286
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200287 if (mSlideoutState == request_hide)
288 mSlideoutState = hidden;
Dees_Troy51a0e822012-09-05 15:24:24 -0400289
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200290 if (mSlideoutState == request_show)
291 mSlideoutState = visible;
Dees_Troy51a0e822012-09-05 15:24:24 -0400292
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200293 // Any time we activate the slider, we reset the position
294 mCurrentLine = -1;
295 return 2;
296 }
Dees_Troy51a0e822012-09-05 15:24:24 -0400297
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200298 if (mCurrentLine == -1 && mLastCount != gConsole.size())
299 {
300 // We can use Render, and return for just a flip
301 Render();
302 return 2;
303 }
Dees Troy31218ec2014-02-25 20:35:56 +0000304 else if (mRender)
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200305 {
306 // They're still touching, so re-render
307 Render();
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200308 return 2;
309 }
310 return 0;
Dees_Troy51a0e822012-09-05 15:24:24 -0400311}
312
313int GUIConsole::SetRenderPos(int x, int y, int w, int h)
314{
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200315 // Adjust the stub position accordingly
316 mSlideoutX += (x - mConsoleX);
317 mSlideoutY += (y - mConsoleY);
Dees_Troy51a0e822012-09-05 15:24:24 -0400318
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200319 mConsoleX = x;
320 mConsoleY = y;
321 if (w || h)
322 {
323 mConsoleW = w;
324 mConsoleH = h;
325 }
Dees_Troy51a0e822012-09-05 15:24:24 -0400326
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200327 // Calculate the max rows
328 mMaxRows = mConsoleH / mFontHeight;
Dees_Troy51a0e822012-09-05 15:24:24 -0400329
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200330 // Adjust so we always fit to bottom
331 mStartY = mConsoleY + (mConsoleH % mFontHeight);
332 return 0;
Dees_Troy51a0e822012-09-05 15:24:24 -0400333}
334
335// IsInRegion - Checks if the request is handled by this object
336// Return 0 if this object handles the request, 1 if not
337int GUIConsole::IsInRegion(int x, int y)
338{
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200339 if (mSlideout)
340 {
341 // Check if they tapped the slideout button
342 if (x >= mSlideoutX && x <= mSlideoutX + mSlideoutW && y >= mSlideoutY && y < mSlideoutY + mSlideoutH)
343 return 1;
Dees_Troy51a0e822012-09-05 15:24:24 -0400344
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200345 // If we're only rendering the slideout, bail now
346 if (mSlideoutState == hidden)
347 return 0;
348 }
Dees_Troy51a0e822012-09-05 15:24:24 -0400349
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200350 return (x < mConsoleX || x > mConsoleX + mConsoleW || y < mConsoleY || y > mConsoleY + mConsoleH) ? 0 : 1;
Dees_Troy51a0e822012-09-05 15:24:24 -0400351}
352
353// NotifyTouch - Notify of a touch event
354// Return 0 on success, >0 to ignore remainder of touch, and <0 on error
355int GUIConsole::NotifyTouch(TOUCH_STATE state, int x, int y)
356{
Vojtech Bocekede51c52014-02-07 23:58:09 +0100357 if(!isConditionTrue())
358 return -1;
359
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200360 if (mSlideout && mSlideoutState == hidden)
361 {
362 if (state == TOUCH_START)
363 {
364 mSlideoutState = request_show;
365 return 1;
366 }
367 }
368 else if (mSlideout && mSlideoutState == visible)
369 {
370 // Are we sliding it back in?
371 if (state == TOUCH_START && x > mSlideoutX && x < (mSlideoutX + mSlideoutW) && y > mSlideoutY && y < (mSlideoutY + mSlideoutH))
372 {
373 mSlideoutState = request_hide;
374 return 1;
375 }
376 }
Dees_Troy51a0e822012-09-05 15:24:24 -0400377
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200378 // If we don't have enough lines to scroll, throw this away.
Dees Troy31218ec2014-02-25 20:35:56 +0000379 if (RenderCount < mMaxRows) return 1;
Dees_Troy51a0e822012-09-05 15:24:24 -0400380
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200381 // We are scrolling!!!
382 switch (state)
383 {
384 case TOUCH_START:
385 mLastTouchX = x;
386 mLastTouchY = y;
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200387 break;
Dees_Troy51a0e822012-09-05 15:24:24 -0400388
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200389 case TOUCH_DRAG:
Dees Troy31218ec2014-02-25 20:35:56 +0000390 if (x < mConsoleX || x > mConsoleX + mConsoleW || y < mConsoleY || y > mConsoleY + mConsoleH)
391 break; // touch is outside of the console area -- do nothing
392 if (y > mLastTouchY + mFontHeight) {
393 while (y > mLastTouchY + mFontHeight) {
394 if (mCurrentLine == -1)
395 mCurrentLine = RenderCount - 1;
396 else if (mCurrentLine > mMaxRows)
397 mCurrentLine--;
398 mLastTouchY += mFontHeight;
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200399 }
Dees Troy31218ec2014-02-25 20:35:56 +0000400 mRender = true;
401 } else if (y < mLastTouchY - mFontHeight) {
402 while (y < mLastTouchY - mFontHeight) {
403 if (mCurrentLine >= 0)
404 mCurrentLine++;
405 mLastTouchY -= mFontHeight;
406 }
407 if (mCurrentLine >= (int) RenderCount)
408 mCurrentLine = -1;
409 mRender = true;
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200410 }
411 break;
Dees_Troy51a0e822012-09-05 15:24:24 -0400412
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200413 case TOUCH_RELEASE:
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200414 mLastTouchY = -1;
Dees_Troy51a0e822012-09-05 15:24:24 -0400415 case TOUCH_REPEAT:
416 case TOUCH_HOLD:
Vojtech Bocekfafb0c52013-07-25 22:53:02 +0200417 break;
418 }
419 return 0;
Dees_Troy51a0e822012-09-05 15:24:24 -0400420}