mirror of
https://github.com/tuxbox-fork-migrations/recycled-ni-neutrino.git
synced 2025-08-30 17:01:08 +02:00
Member initVarHeader() now used with parameters, so it's possible
to remove multiple code in constructors, but some class must be
renamed to CComponentsHeaderLocalized.
Origin commit data
------------------
Commit: 5ea7ec48df
Author: Thilo Graf <dbt@novatux.de>
Date: 2014-01-18 (Sat, 18 Jan 2014)
Origin message was:
------------------
CComponentsHeader: rework constructors, add CComponentsHeaderLocalized
Member initVarHeader() now used with parameters, so it's possible
to remove multiple code in constructors, but some class must be
renamed to CComponentsHeaderLocalized.
2113 lines
60 KiB
C++
2113 lines
60 KiB
C++
/*
|
|
Neutrino-GUI - DBoxII-Project
|
|
|
|
Copyright (C) 2001 Steffen Hehn 'McClean'
|
|
Homepage: http://dbox.cyberphoria.org/
|
|
|
|
(C) 2008, 2009 Stefan Seyfried
|
|
Copyright (C) 2012 CoolStream International Ltd
|
|
|
|
License: GPLv2
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the
|
|
Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
|
|
#include <global.h>
|
|
#include <neutrino.h>
|
|
#include <gui/widget/menue.h>
|
|
|
|
#include <driver/fontrenderer.h>
|
|
#include <driver/screen_max.h>
|
|
#include <gui/widget/stringinput.h>
|
|
|
|
#include <neutrino_menue.h>
|
|
#include <driver/fade.h>
|
|
#include <system/helpers.h>
|
|
|
|
#include <cctype>
|
|
|
|
/* the following generic menu items are integrated into multiple menus at the same time */
|
|
CMenuSeparator CGenericMenuSeparator(0, NONEXISTANT_LOCALE, true);
|
|
CMenuSeparator CGenericMenuSeparatorLine(CMenuSeparator::LINE, NONEXISTANT_LOCALE, true);
|
|
CMenuForwarder CGenericMenuBack(LOCALE_MENU_BACK, true, NULL, NULL, NULL, CRCInput::RC_nokey, NEUTRINO_ICON_BUTTON_LEFT, NULL, true);
|
|
CMenuForwarder CGenericMenuCancel(LOCALE_MENU_CANCEL, true, NULL, NULL, NULL, CRCInput::RC_nokey, NEUTRINO_ICON_BUTTON_HOME, NULL, true);
|
|
CMenuForwarder CGenericMenuNext(LOCALE_MENU_NEXT, true, NULL, NULL, NULL, CRCInput::RC_nokey, NEUTRINO_ICON_BUTTON_HOME, NULL, true);
|
|
CMenuSeparator * const GenericMenuSeparator = &CGenericMenuSeparator;
|
|
CMenuSeparator * const GenericMenuSeparatorLine = &CGenericMenuSeparatorLine;
|
|
CMenuForwarder * const GenericMenuBack = &CGenericMenuBack;
|
|
CMenuForwarder * const GenericMenuCancel = &CGenericMenuCancel;
|
|
CMenuForwarder * const GenericMenuNext = &CGenericMenuNext;
|
|
|
|
CMenuItem::CMenuItem()
|
|
{
|
|
x = -1;
|
|
directKey = CRCInput::RC_nokey;
|
|
iconName = "";
|
|
iconName_Info_right = "";
|
|
used = false;
|
|
icon_frame_w = 10;
|
|
hint = NONEXISTANT_LOCALE;
|
|
name = NONEXISTANT_LOCALE;
|
|
nameString = "";
|
|
isStatic = false;
|
|
marked = false;
|
|
inert = false;
|
|
}
|
|
|
|
void CMenuItem::init(const int X, const int Y, const int DX, const int OFFX)
|
|
{
|
|
x = X;
|
|
y = Y;
|
|
dx = DX;
|
|
offx = OFFX;
|
|
name_start_x = x + offx + icon_frame_w;
|
|
item_color = COL_MENUCONTENT_TEXT;
|
|
item_bgcolor = COL_MENUCONTENT_PLUS_0;
|
|
}
|
|
|
|
void CMenuItem::setActive(const bool Active)
|
|
{
|
|
active = Active;
|
|
/* used gets set by the addItem() function. This is for disabling
|
|
machine-specific options by just not calling the addItem() function.
|
|
Without this, the changeNotifiers would become machine-dependent. */
|
|
if (used && x != -1)
|
|
paint();
|
|
}
|
|
|
|
void CMenuItem::setMarked(const bool Marked)
|
|
{
|
|
marked = Marked;
|
|
if (used && x != -1)
|
|
paint();
|
|
}
|
|
|
|
void CMenuItem::setInert(const bool Inert)
|
|
{
|
|
inert = Inert;
|
|
if (used && x != -1)
|
|
paint();
|
|
}
|
|
|
|
void CMenuItem::setItemButton(const std::string& icon_Name, const bool is_select_button)
|
|
{
|
|
if (is_select_button)
|
|
selected_iconName = icon_Name;
|
|
else
|
|
iconName = icon_Name;
|
|
}
|
|
|
|
void CMenuItem::initItemColors(const bool select_mode)
|
|
{
|
|
if (select_mode)
|
|
{
|
|
item_color = COL_MENUCONTENTSELECTED_TEXT;
|
|
item_bgcolor = COL_MENUCONTENTSELECTED_PLUS_0;
|
|
}
|
|
else if (!active || inert)
|
|
{
|
|
item_color = COL_MENUCONTENTINACTIVE_TEXT;
|
|
item_bgcolor = COL_MENUCONTENTINACTIVE_PLUS_0;
|
|
}
|
|
else if (marked)
|
|
{
|
|
item_color = COL_MENUCONTENT_TEXT;
|
|
item_bgcolor = COL_MENUCONTENT_PLUS_1;
|
|
}
|
|
else
|
|
{
|
|
item_color = COL_MENUCONTENT_TEXT;
|
|
item_bgcolor = COL_MENUCONTENT_PLUS_0;
|
|
}
|
|
}
|
|
|
|
void CMenuItem::paintItemBackground (const bool select_mode, const int &item_height)
|
|
{
|
|
CFrameBuffer *frameBuffer = CFrameBuffer::getInstance();
|
|
//FIXME what select_mode change here ??
|
|
if(select_mode)
|
|
frameBuffer->paintBoxRel(x, y, dx, item_height, item_bgcolor, RADIUS_LARGE);
|
|
else
|
|
frameBuffer->paintBoxRel(x, y, dx, item_height, item_bgcolor, RADIUS_LARGE);
|
|
}
|
|
|
|
void CMenuItem::paintItemCaption(const bool select_mode, const int &item_height, const char * left_text, const char * right_text, const fb_pixel_t right_bgcol)
|
|
{
|
|
if (select_mode)
|
|
{
|
|
char str[256];
|
|
|
|
if (right_text != NULL)
|
|
{
|
|
snprintf(str, 255, "%s %s", left_text, right_text);
|
|
CVFD::getInstance()->showMenuText(0, str, -1, true);
|
|
}
|
|
else
|
|
CVFD::getInstance()->showMenuText(0, left_text, -1, true);
|
|
}
|
|
|
|
//left text
|
|
int _dx = dx;
|
|
int icon_w = 0;
|
|
int icon_h = 0;
|
|
if (!iconName_Info_right.empty()) {
|
|
CFrameBuffer::getInstance()->getIconSize(iconName_Info_right.c_str(), &icon_w, &icon_h);
|
|
if (icon_w)
|
|
_dx -= icon_frame_w + icon_w;
|
|
}
|
|
g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->RenderString(name_start_x, y+ item_height, _dx- (name_start_x - x), left_text, item_color, 0, true); // UTF-8
|
|
|
|
//right text
|
|
if (right_text && (*right_text || right_bgcol))
|
|
{
|
|
int stringwidth = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(right_text, true);
|
|
int stringstartposOption = std::max(name_start_x + g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(left_text, true) + icon_frame_w, x + dx - stringwidth - icon_frame_w); //+ offx
|
|
if (right_bgcol) {
|
|
if (!*right_text)
|
|
stringstartposOption -= 60;
|
|
fb_pixel_t right_frame_col, right_bg_col;
|
|
if (active) {
|
|
right_bg_col = right_bgcol;
|
|
right_frame_col = COL_MENUCONTENT_PLUS_6;
|
|
}
|
|
else {
|
|
right_bg_col = COL_MENUCONTENTINACTIVE_TEXT;
|
|
right_frame_col = COL_MENUCONTENTINACTIVE_TEXT;
|
|
}
|
|
CComponentsShapeSquare col(stringstartposOption, y + 2, dx - stringstartposOption + x - 2, item_height - 4, false, right_frame_col, right_bg_col);
|
|
col.setFrameThickness(3);
|
|
col.setCorner(RADIUS_LARGE);
|
|
col.paint(false);
|
|
}
|
|
if (*right_text) {
|
|
stringstartposOption -= (icon_w == 0 ? 0 : icon_w + icon_frame_w);
|
|
g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->RenderString(stringstartposOption, y+item_height,dx- (stringstartposOption- x), right_text, item_color, 0, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CMenuItem::prepareItem(const bool select_mode, const int &item_height)
|
|
{
|
|
//set colors
|
|
initItemColors(select_mode);
|
|
|
|
//paint item background
|
|
paintItemBackground(select_mode, item_height);
|
|
}
|
|
|
|
void CMenuItem::paintItemSlider( const bool select_mode, const int &item_height, const int &optionvalue, const int &factor, const char * left_text, const char * right_text)
|
|
{
|
|
CFrameBuffer *frameBuffer = CFrameBuffer::getInstance();
|
|
int slider_lenght = 0, h = 0;
|
|
frameBuffer->getIconSize(NEUTRINO_ICON_VOLUMEBODY, &slider_lenght, &h);
|
|
if(slider_lenght == 0 || factor < optionvalue )
|
|
return;
|
|
int stringwidth = 0;
|
|
if (right_text != NULL) {
|
|
stringwidth = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth("U999", true) ;
|
|
}
|
|
int stringwidth2 = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(left_text, true);
|
|
|
|
int maxspace = dx - stringwidth - icon_frame_w - stringwidth2 - 10;
|
|
if(maxspace < slider_lenght){
|
|
return ;
|
|
}
|
|
|
|
int stringstartposOption = x + dx - stringwidth - slider_lenght;
|
|
int optionV = (optionvalue < 0) ? 0 : optionvalue;
|
|
frameBuffer->paintBoxRel(stringstartposOption, y, slider_lenght, item_height, item_bgcolor);
|
|
frameBuffer->paintIcon(NEUTRINO_ICON_VOLUMEBODY, stringstartposOption, y+2+item_height/4);
|
|
frameBuffer->paintIcon(select_mode ? NEUTRINO_ICON_VOLUMESLIDER2BLUE : NEUTRINO_ICON_VOLUMESLIDER2, (stringstartposOption + (optionV * 100 / factor)), y+item_height/4);
|
|
}
|
|
|
|
void CMenuItem::paintItemButton(const bool select_mode, const int &item_height, const std::string &icon_Name)
|
|
{
|
|
CFrameBuffer *frameBuffer = CFrameBuffer::getInstance();
|
|
bool selected = select_mode;
|
|
bool icon_painted = false;
|
|
|
|
std::string icon_name = iconName;
|
|
int icon_w = 0;
|
|
int icon_h = 0;
|
|
|
|
//define icon name depends of numeric value
|
|
if (icon_name.empty() && CRCInput::isNumeric(directKey))
|
|
{
|
|
char i_name[6]; /* X +'\0' */
|
|
snprintf(i_name, 6, "%d", CRCInput::getNumericValue(directKey));
|
|
i_name[5] = '\0'; /* even if snprintf truncated the string, ensure termination */
|
|
icon_name = i_name;
|
|
}
|
|
|
|
//define select icon
|
|
if (selected && offx > 0)
|
|
{
|
|
if (!selected_iconName.empty())
|
|
icon_name = selected_iconName;
|
|
else if (icon_name.empty() && !CRCInput::isNumeric(directKey))
|
|
icon_name = icon_Name;
|
|
}
|
|
|
|
int icon_x = 0;
|
|
int icon_start_x = x+icon_frame_w; //start of icon space
|
|
int icon_space_x = name_start_x - icon_frame_w - icon_start_x; //size of space where to paint icon
|
|
int icon_space_mid = icon_start_x + icon_space_x/2;
|
|
|
|
//get data of number icon and paint
|
|
if (!icon_name.empty())
|
|
{
|
|
frameBuffer->getIconSize(icon_name.c_str(), &icon_w, &icon_h);
|
|
|
|
if (active && icon_w>0 && icon_h>0 && icon_space_x >= icon_w)
|
|
{
|
|
icon_x = icon_space_mid - (icon_w/2);
|
|
|
|
icon_painted = frameBuffer->paintIcon(icon_name, icon_x, y+ ((item_height/2- icon_h/2)) );
|
|
}
|
|
}
|
|
|
|
//paint only number if no icon was painted and keyval is numeric
|
|
if (active && CRCInput::isNumeric(directKey) && !icon_painted)
|
|
{
|
|
int number_w = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(CRCInput::getKeyName(directKey));
|
|
|
|
int number_x = icon_space_mid - (number_w/2);
|
|
|
|
g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->RenderString(number_x, y+ item_height, item_height, CRCInput::getKeyName(directKey), item_color, item_height);
|
|
}
|
|
|
|
//get data of number right info icon and paint
|
|
if (!iconName_Info_right.empty())
|
|
{
|
|
frameBuffer->getIconSize(iconName_Info_right.c_str(), &icon_w, &icon_h);
|
|
|
|
if (active && icon_w>0 && icon_h>0)
|
|
icon_painted = frameBuffer->paintIcon(iconName_Info_right, dx + icon_start_x - (icon_w + 20), y+ ((item_height/2- icon_h/2)) );
|
|
}
|
|
}
|
|
|
|
const char *CMenuItem::getName(void)
|
|
{
|
|
if (name != NONEXISTANT_LOCALE)
|
|
return g_Locale->getText(name);
|
|
return nameString.c_str();
|
|
}
|
|
|
|
//small helper class to manage values e.g.: handling needed but deallocated widget objects
|
|
CMenuGlobal::CMenuGlobal()
|
|
{
|
|
//creates needed select values with default value NO_WIDGET_ID = -1
|
|
for (uint i=0; i<MN_WIDGET_ID_MAX; ++i)
|
|
v_selected.push_back(NO_WIDGET_ID);
|
|
}
|
|
|
|
CMenuGlobal::~CMenuGlobal()
|
|
{
|
|
v_selected.clear();
|
|
}
|
|
|
|
//Note: use only singleton to create an instance in the constructor or init handler of menu widget
|
|
CMenuGlobal* CMenuGlobal::getInstance()
|
|
{
|
|
static CMenuGlobal* m = NULL;
|
|
|
|
if(!m)
|
|
m = new CMenuGlobal();
|
|
return m;
|
|
}
|
|
//****************************************************************************************
|
|
|
|
CMenuWidget::CMenuWidget()
|
|
{
|
|
nameString = g_Locale->getText(NONEXISTANT_LOCALE);
|
|
name = NONEXISTANT_LOCALE;
|
|
iconfile = "";
|
|
selected = -1;
|
|
iconOffset = 0;
|
|
offx = offy = 0;
|
|
from_wizard = false;
|
|
fade = true;
|
|
sb_width = 0;
|
|
savescreen = false;
|
|
background = NULL;
|
|
preselected = -1;
|
|
details_line = NULL;
|
|
info_box = NULL;
|
|
}
|
|
|
|
CMenuWidget::CMenuWidget(const neutrino_locale_t Name, const std::string & Icon, const int mwidth, const mn_widget_id_t &w_index)
|
|
{
|
|
name = Name;
|
|
nameString = g_Locale->getText(Name);
|
|
preselected = -1;
|
|
Init(Icon, mwidth, w_index);
|
|
}
|
|
|
|
CMenuWidget::CMenuWidget(const char* Name, const std::string & Icon, const int mwidth, const mn_widget_id_t &w_index)
|
|
{
|
|
name = NONEXISTANT_LOCALE;
|
|
nameString = Name;
|
|
preselected = -1;
|
|
Init(Icon, mwidth, w_index);
|
|
}
|
|
|
|
void CMenuWidget::Init(const std::string & Icon, const int mwidth, const mn_widget_id_t &w_index)
|
|
{
|
|
mglobal = CMenuGlobal::getInstance(); //create CMenuGlobal instance only here
|
|
frameBuffer = CFrameBuffer::getInstance();
|
|
iconfile = Icon;
|
|
details_line = new CComponentsDetailLine();
|
|
info_box = new CComponentsInfoBox();
|
|
|
|
//handle select values
|
|
if(w_index > MN_WIDGET_ID_MAX){
|
|
//error
|
|
fprintf(stderr, "Warning: %s Index ID value (%i) is bigger than MN_WIDGET_ID_MAX (%i) \n", __FUNCTION__,w_index,MN_WIDGET_ID_MAX );
|
|
widget_index = NO_WIDGET_ID;
|
|
}
|
|
else{
|
|
//ok
|
|
widget_index = w_index;
|
|
}
|
|
|
|
//set default preselected if value = -1 (no selected). If value has changed, e.g. with setSelected(), then use current value
|
|
preselected = preselected != -1 ? preselected : -1;
|
|
|
|
//overwrite preselected value with global select value
|
|
selected = (widget_index == NO_WIDGET_ID ? preselected : mglobal->v_selected[widget_index]);
|
|
|
|
|
|
min_width = 0;
|
|
width = 0; /* is set in paint() */
|
|
|
|
if (mwidth > 100)
|
|
{
|
|
/* warn about abuse until we found all offenders... */
|
|
fprintf(stderr, "Warning: %s (%s) (%s) mwidth over 100%%: %d\n", __FUNCTION__, nameString.c_str(), Icon.c_str(), mwidth);
|
|
}
|
|
else
|
|
{
|
|
min_width = frameBuffer->getScreenWidth(true) * mwidth / 100;
|
|
if(min_width > (int) frameBuffer->getScreenWidth())
|
|
min_width = frameBuffer->getScreenWidth();
|
|
}
|
|
|
|
current_page = 0;
|
|
offx = offy = 0;
|
|
from_wizard = false;
|
|
fade = true;
|
|
savescreen = false;
|
|
background = NULL;
|
|
has_hints = false;
|
|
hint_painted = false;
|
|
hint_height = 0;
|
|
}
|
|
|
|
void CMenuWidget::move(int xoff, int yoff)
|
|
{
|
|
offx = xoff;
|
|
offy = yoff;
|
|
}
|
|
|
|
CMenuWidget::~CMenuWidget()
|
|
{
|
|
resetWidget(true);
|
|
delete details_line;
|
|
delete info_box;
|
|
}
|
|
|
|
void CMenuWidget::addItem(CMenuItem* menuItem, const bool defaultselected)
|
|
{
|
|
if (menuItem->isSelectable())
|
|
{
|
|
bool isSelected = defaultselected;
|
|
|
|
if (preselected != -1)
|
|
isSelected = (preselected == (int)items.size());
|
|
|
|
if (isSelected)
|
|
selected = items.size();
|
|
}
|
|
|
|
menuItem->isUsed();
|
|
items.push_back(menuItem);
|
|
}
|
|
|
|
void CMenuWidget::resetWidget(bool delete_items)
|
|
{
|
|
for(unsigned int count=0;count<items.size();count++) {
|
|
CMenuItem * item = items[count];
|
|
if (!item->isStatic && delete_items) {
|
|
delete item;
|
|
item = NULL;
|
|
}
|
|
}
|
|
|
|
items.clear();
|
|
page_start.clear();
|
|
selected=-1;
|
|
}
|
|
|
|
void CMenuWidget::insertItem(const uint& item_id, CMenuItem* menuItem)
|
|
{
|
|
items.insert(items.begin()+item_id, menuItem);
|
|
}
|
|
|
|
void CMenuWidget::removeItem(const uint& item_id)
|
|
{
|
|
items.erase(items.begin()+item_id);
|
|
}
|
|
|
|
bool CMenuWidget::hasItem()
|
|
{
|
|
return !items.empty();
|
|
}
|
|
|
|
int CMenuWidget::getItemId(CMenuItem* menuItem)
|
|
{
|
|
for (uint i= 0; i< items.size(); i++)
|
|
{
|
|
if (items[i] == menuItem)
|
|
return i;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
CMenuItem* CMenuWidget::getItem(const uint& item_id)
|
|
{
|
|
for (uint i= 0; i< items.size(); i++)
|
|
{
|
|
if (i == item_id)
|
|
return items[i];
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
std::string CMenuWidget::getName()
|
|
{
|
|
if (name != NONEXISTANT_LOCALE)
|
|
nameString = g_Locale->getText(name);
|
|
return nameString;
|
|
}
|
|
|
|
int CMenuWidget::exec(CMenuTarget* parent, const std::string &)
|
|
{
|
|
neutrino_msg_t msg;
|
|
neutrino_msg_data_t data;
|
|
bool bAllowRepeatLR = false;
|
|
CVFD::MODES oldLcdMode = CVFD::getInstance()->getMode();
|
|
|
|
int pos = 0;
|
|
if (selected > 0 && selected < (int)items.size())
|
|
pos = selected;
|
|
exit_pressed = false;
|
|
|
|
frameBuffer->Lock();
|
|
|
|
if (parent)
|
|
parent->hide();
|
|
|
|
COSDFader fader(g_settings.menu_Content_alpha);
|
|
if(fade)
|
|
fader.StartFadeIn();
|
|
|
|
if(from_wizard) {
|
|
for (unsigned int count = 0; count < items.size(); count++) {
|
|
if(items[count] == GenericMenuBack) {
|
|
items[count] = GenericMenuNext;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
/* make sure we start with a selectable item... */
|
|
while (pos < (int)items.size()) {
|
|
if (items[pos]->isSelectable())
|
|
break;
|
|
pos++;
|
|
}
|
|
checkHints();
|
|
|
|
if(savescreen) {
|
|
calcSize();
|
|
saveScreen();
|
|
}
|
|
paint();
|
|
int retval = menu_return::RETURN_REPAINT;
|
|
uint64_t timeoutEnd = CRCInput::calcTimeoutEnd(g_settings.timing[SNeutrinoSettings::TIMING_MENU] == 0 ? 0xFFFF : g_settings.timing[SNeutrinoSettings::TIMING_MENU]);
|
|
|
|
do {
|
|
if(hasItem() && selected >= 0 && (int)items.size() > selected )
|
|
bAllowRepeatLR = items[selected]->isMenueOptionChooser();
|
|
|
|
g_RCInput->getMsgAbsoluteTimeout(&msg, &data, &timeoutEnd, bAllowRepeatLR);
|
|
|
|
int handled= false;
|
|
if ( msg <= CRCInput::RC_MaxRC ) {
|
|
timeoutEnd = CRCInput::calcTimeoutEnd(g_settings.timing[SNeutrinoSettings::TIMING_MENU] == 0 ? 0xFFFF : g_settings.timing[SNeutrinoSettings::TIMING_MENU]);
|
|
|
|
std::map<neutrino_msg_t, keyAction>::iterator it = keyActionMap.find(msg);
|
|
if (it != keyActionMap.end()) {
|
|
fader.Stop();
|
|
int rv = it->second.menue->exec(this, it->second.action);
|
|
switch ( rv ) {
|
|
case menu_return::RETURN_EXIT_ALL:
|
|
retval = menu_return::RETURN_EXIT_ALL;
|
|
case menu_return::RETURN_EXIT:
|
|
msg = CRCInput::RC_timeout;
|
|
break;
|
|
case menu_return::RETURN_REPAINT:
|
|
case menu_return::RETURN_EXIT_REPAINT:
|
|
if (fade && washidden)
|
|
fader.StartFadeIn();
|
|
checkHints();
|
|
paint();
|
|
break;
|
|
}
|
|
frameBuffer->blit();
|
|
continue;
|
|
}
|
|
for (unsigned int i= 0; i< items.size(); i++) {
|
|
CMenuItem* titem = items[i];
|
|
if ((titem->directKey != CRCInput::RC_nokey) && (titem->directKey == msg)) {
|
|
if (titem->isSelectable()) {
|
|
items[selected]->paint( false );
|
|
selected= i;
|
|
paintHint(selected);
|
|
pos = selected;
|
|
msg= CRCInput::RC_ok;
|
|
} else {
|
|
// swallow-key...
|
|
handled= true;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
#if 0
|
|
if (msg == (uint32_t) g_settings.key_channelList_pageup)
|
|
msg = CRCInput::RC_page_up;
|
|
else if (msg == (uint32_t) g_settings.key_channelList_pagedown)
|
|
msg = CRCInput::RC_page_down;
|
|
#endif
|
|
}
|
|
|
|
if (handled)
|
|
continue;
|
|
|
|
switch (msg) {
|
|
case (NeutrinoMessages::EVT_TIMER):
|
|
if(data == fader.GetTimer()) {
|
|
if(fader.Fade())
|
|
msg = CRCInput::RC_timeout;
|
|
} else {
|
|
if ( CNeutrinoApp::getInstance()->handleMsg( msg, data ) & messages_return::cancel_all ) {
|
|
retval = menu_return::RETURN_EXIT_ALL;
|
|
msg = CRCInput::RC_timeout;
|
|
}
|
|
}
|
|
break;
|
|
case CRCInput::RC_page_up:
|
|
case CRCInput::RC_page_down:
|
|
case CRCInput::RC_up:
|
|
case CRCInput::RC_down:
|
|
{
|
|
/* dir and wrap are used when searching for a selectable item:
|
|
* if the item is not selectable, try the previous (dir = -1) or
|
|
* next (dir = 1) item, until a selectale item is found.
|
|
* if wrap = true, allows to wrap around while searching */
|
|
int dir = 1;
|
|
bool wrap = false;
|
|
switch (msg) {
|
|
case CRCInput::RC_page_up:
|
|
if (current_page) {
|
|
pos = page_start[current_page] - 1;
|
|
dir = -1; /* pg_up: search upwards */
|
|
} else
|
|
pos = 0; /* ...but not if already at top */
|
|
break;
|
|
case CRCInput::RC_page_down:
|
|
pos = page_start[current_page + 1];
|
|
if (pos >= (int)items.size()) {
|
|
pos = items.size() - 1;
|
|
dir = -1; /* if last item is not selectable, go up */
|
|
}
|
|
break;
|
|
case CRCInput::RC_up:
|
|
dir = -1;
|
|
default: /* fallthrough or RC_down => dir = 1 */
|
|
pos += dir;
|
|
if (pos < 0 || pos >= (int)items.size())
|
|
pos -= dir * items.size();
|
|
wrap = true;
|
|
}
|
|
do {
|
|
CMenuItem* item = items[pos];
|
|
if (item->isSelectable()) {
|
|
if (pos < page_start[current_page + 1] && pos >= page_start[current_page]) {
|
|
/* in current page */
|
|
items[selected]->paint(false);
|
|
item->paint(true);
|
|
paintHint(pos);
|
|
selected = pos;
|
|
} else {
|
|
/* different page */
|
|
selected = pos;
|
|
paintItems();
|
|
}
|
|
break;
|
|
}
|
|
pos += dir;
|
|
if (wrap && (pos >= (int)items.size() || pos < 0)) {
|
|
pos -= dir * items.size();
|
|
wrap = false; /* wrap only once, avoids endless loop */
|
|
}
|
|
} while (pos >= 0 && pos < (int)items.size());
|
|
break;
|
|
}
|
|
case (CRCInput::RC_left):
|
|
{
|
|
if(hasItem() && selected > -1 && (int)items.size() > selected) {
|
|
CMenuItem* itemX = items[selected];
|
|
if (!itemX->isMenueOptionChooser()) {
|
|
if (g_settings.menu_left_exit)
|
|
msg = CRCInput::RC_timeout;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
case (CRCInput::RC_right):
|
|
case (CRCInput::RC_ok):
|
|
{
|
|
if(hasItem() && selected > -1 && (int)items.size() > selected) {
|
|
//exec this item...
|
|
CMenuItem* item = items[selected];
|
|
if (!item->isSelectable())
|
|
break;
|
|
item->msg = msg;
|
|
fader.Stop();
|
|
int rv = item->exec( this );
|
|
switch ( rv ) {
|
|
case menu_return::RETURN_EXIT_ALL:
|
|
retval = menu_return::RETURN_EXIT_ALL;
|
|
case menu_return::RETURN_EXIT:
|
|
msg = CRCInput::RC_timeout;
|
|
break;
|
|
case menu_return::RETURN_REPAINT:
|
|
case menu_return::RETURN_EXIT_REPAINT:
|
|
if (fade && washidden)
|
|
fader.StartFadeIn();
|
|
checkHints();
|
|
pos = selected;
|
|
paint();
|
|
break;
|
|
}
|
|
} else
|
|
msg = CRCInput::RC_timeout;
|
|
}
|
|
break;
|
|
|
|
case (CRCInput::RC_home):
|
|
exit_pressed = true;
|
|
msg = CRCInput::RC_timeout;
|
|
break;
|
|
case (CRCInput::RC_timeout):
|
|
break;
|
|
|
|
case (CRCInput::RC_sat):
|
|
case (CRCInput::RC_favorites):
|
|
g_RCInput->postMsg (msg, 0);
|
|
//close any menue on dbox-key
|
|
case (CRCInput::RC_setup):
|
|
{
|
|
msg = CRCInput::RC_timeout;
|
|
retval = menu_return::RETURN_EXIT_ALL;
|
|
}
|
|
break;
|
|
case (CRCInput::RC_help):
|
|
// FIXME should we switch hints in menu without hints ?
|
|
checkHints();
|
|
if (has_hints)
|
|
hide();
|
|
g_settings.show_menu_hints = !g_settings.show_menu_hints;
|
|
if (has_hints)
|
|
paint();
|
|
break;
|
|
|
|
default:
|
|
if ( CNeutrinoApp::getInstance()->handleMsg( msg, data ) & messages_return::cancel_all ) {
|
|
retval = menu_return::RETURN_EXIT_ALL;
|
|
msg = CRCInput::RC_timeout;
|
|
}
|
|
}
|
|
if(msg == CRCInput::RC_timeout) {
|
|
if(fade && fader.StartFadeOut()) {
|
|
timeoutEnd = CRCInput::calcTimeoutEnd( 1 );
|
|
msg = 0;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if ( msg <= CRCInput::RC_MaxRC )
|
|
{
|
|
// recalculate timeout for RC-keys
|
|
timeoutEnd = CRCInput::calcTimeoutEnd(g_settings.timing[SNeutrinoSettings::TIMING_MENU] == 0 ? 0xFFFF : g_settings.timing[SNeutrinoSettings::TIMING_MENU]);
|
|
}
|
|
}
|
|
while ( msg!=CRCInput::RC_timeout );
|
|
hide();
|
|
delete[] background;
|
|
background = NULL;
|
|
|
|
fader.Stop();
|
|
|
|
if(!parent)
|
|
if(oldLcdMode != CVFD::getInstance()->getMode())
|
|
CVFD::getInstance()->setMode(CVFD::MODE_TVRADIO);
|
|
|
|
for (unsigned int count = 0; count < items.size(); count++)
|
|
{
|
|
if(items[count] == GenericMenuNext)
|
|
items[count] = GenericMenuBack;
|
|
else if (items[count] == GenericMenuCancel)
|
|
items[count] = GenericMenuBack;
|
|
}
|
|
|
|
if (widget_index > -1)
|
|
mglobal->v_selected[widget_index] = selected;
|
|
|
|
frameBuffer->Unlock();
|
|
return retval;
|
|
}
|
|
|
|
void CMenuWidget::hide()
|
|
{
|
|
if(savescreen && background)
|
|
restoreScreen();//FIXME
|
|
else {
|
|
frameBuffer->paintBackgroundBoxRel(x, y, full_width, full_height);
|
|
//paintHint(-1);
|
|
}
|
|
paintHint(-1);
|
|
|
|
/* setActive() paints item for hidden parent menu, if called from child menu */
|
|
for (unsigned int count = 0; count < items.size(); count++)
|
|
items[count]->init(-1, 0, 0, 0);
|
|
hint_painted = false;
|
|
washidden = true;
|
|
}
|
|
|
|
void CMenuWidget::checkHints()
|
|
{
|
|
GenericMenuBack->setHint("", NONEXISTANT_LOCALE);
|
|
for (unsigned int i= 0; i< items.size(); i++) {
|
|
if(!items[i]->hintIcon.empty() || items[i]->hint != NONEXISTANT_LOCALE) {
|
|
has_hints = true;
|
|
break;
|
|
}
|
|
}
|
|
if (has_hints)
|
|
GenericMenuBack->setHint(NEUTRINO_ICON_HINT_BACK, LOCALE_MENU_HINT_BACK);
|
|
}
|
|
|
|
void CMenuWidget::calcSize()
|
|
{
|
|
if (name != NONEXISTANT_LOCALE)
|
|
nameString = g_Locale->getText(name);
|
|
|
|
width = min_width;
|
|
|
|
int wi, hi;
|
|
for (unsigned int i= 0; i< items.size(); i++) {
|
|
wi = 0;
|
|
if (!items[i]->iconName_Info_right.empty()) {
|
|
frameBuffer->getIconSize(items[i]->iconName_Info_right.c_str(), &wi, &hi);
|
|
if ((wi > 0) && (hi > 0))
|
|
wi += 10;
|
|
else
|
|
wi = 0;
|
|
}
|
|
int tmpw = items[i]->getWidth() + 10 + 10 + wi; /* 10 pixels to the left and right of the text */
|
|
if (tmpw > width)
|
|
width = tmpw;
|
|
}
|
|
hint_height = 0;
|
|
if(g_settings.show_menu_hints && has_hints) {
|
|
hint_height = 60; //TODO: rework calculation of hint_height
|
|
int fheight = g_Font[SNeutrinoSettings::FONT_TYPE_MENU_HINT]->getHeight();
|
|
int h_tmp = 16 + 2*fheight;
|
|
/* assuming all hint icons has the same size ! */
|
|
int iw, ih;
|
|
frameBuffer->getIconSize(NEUTRINO_ICON_HINT_TVMODE, &iw, &ih);
|
|
h_tmp = std::max(h_tmp, ih+10);
|
|
hint_height = std::max(h_tmp, hint_height);
|
|
}
|
|
/* set the max height to 9/10 of usable screen height
|
|
debatable, if the callers need a possibility to set this */
|
|
height = (frameBuffer->getScreenHeight() - hint_height) / 20 * 18; /* make sure its a multiple of 2 */
|
|
|
|
if(height > ((int)frameBuffer->getScreenHeight() - 10))
|
|
height = frameBuffer->getScreenHeight() - 10;
|
|
|
|
int neededWidth = g_Font[SNeutrinoSettings::FONT_TYPE_MENU_TITLE]->getRenderWidth(nameString.c_str(), true); // UTF-8
|
|
if (neededWidth > width-48) {
|
|
width= neededWidth+ 49;
|
|
}
|
|
hheight = g_Font[SNeutrinoSettings::FONT_TYPE_MENU_TITLE]->getHeight();
|
|
|
|
int itemHeightTotal=0;
|
|
int heightCurrPage=0;
|
|
page_start.clear();
|
|
page_start.push_back(0);
|
|
total_pages=1;
|
|
for (unsigned int i= 0; i< items.size(); i++) {
|
|
int item_height=items[i]->getHeight();
|
|
heightCurrPage+=item_height;
|
|
if(heightCurrPage > (height-hheight)) {
|
|
page_start.push_back(i);
|
|
total_pages++;
|
|
heightCurrPage=item_height;
|
|
}
|
|
if (heightCurrPage > itemHeightTotal)
|
|
itemHeightTotal = heightCurrPage;
|
|
}
|
|
page_start.push_back(items.size());
|
|
|
|
iconOffset= 0;
|
|
for (unsigned int i= 0; i< items.size(); i++) {
|
|
if ((!(items[i]->iconName.empty())) || CRCInput::isNumeric(items[i]->directKey))
|
|
{
|
|
const char *tmp;
|
|
if (items[i]->iconName.empty())
|
|
tmp = CRCInput::getKeyName(items[i]->directKey).c_str();
|
|
else
|
|
tmp = items[i]->iconName.c_str();
|
|
int w, h;
|
|
frameBuffer->getIconSize(tmp, &w, &h);
|
|
if (w > iconOffset)
|
|
iconOffset = w;
|
|
}
|
|
}
|
|
|
|
iconOffset += 10;
|
|
width += iconOffset;
|
|
if (width > (int)frameBuffer->getScreenWidth())
|
|
width = frameBuffer->getScreenWidth();
|
|
|
|
// shrink menu if less items
|
|
if(hheight+itemHeightTotal < height)
|
|
height=hheight+itemHeightTotal;
|
|
|
|
//scrollbar width
|
|
sb_width=0;
|
|
if(total_pages > 1)
|
|
sb_width=15;
|
|
|
|
full_width = /*ConnectLineBox_Width+*/width+sb_width+SHADOW_OFFSET;
|
|
full_height = height+RADIUS_LARGE+SHADOW_OFFSET*2 /*+hint_height+INFO_BOX_Y_OFFSET*/;
|
|
|
|
setMenuPos(width - sb_width);
|
|
}
|
|
|
|
void CMenuWidget::paint()
|
|
{
|
|
calcSize();
|
|
CVFD::getInstance()->setMode(CVFD::MODE_MENU_UTF8 /*, nameString.c_str()*/);
|
|
|
|
// paint head
|
|
CComponentsHeader header(x, y, width + sb_width, hheight, nameString, iconfile);
|
|
header.setShadowOnOff(CC_SHADOW_ON);
|
|
header.setOffset(10);
|
|
header.paint(CC_SAVE_SCREEN_NO);
|
|
|
|
// paint body shadow
|
|
frameBuffer->paintBoxRel(x+SHADOW_OFFSET, y + hheight + SHADOW_OFFSET, width + sb_width, height - hheight + RADIUS_LARGE, COL_MENUCONTENTDARK_PLUS_0, RADIUS_LARGE, CORNER_BOTTOM);
|
|
// paint body background
|
|
frameBuffer->paintBoxRel(x ,y+hheight, width + sb_width, height-hheight + RADIUS_LARGE ,COL_MENUCONTENT_PLUS_0 ,RADIUS_LARGE, CORNER_BOTTOM);
|
|
|
|
item_start_y = y+hheight;
|
|
paintItems();
|
|
washidden = false;
|
|
}
|
|
|
|
void CMenuWidget::setMenuPos(const int& menu_width)
|
|
{
|
|
int scr_x = frameBuffer->getScreenX();
|
|
int scr_y = frameBuffer->getScreenY();
|
|
int scr_w = frameBuffer->getScreenWidth();
|
|
int scr_h = frameBuffer->getScreenHeight();
|
|
|
|
//configured positions
|
|
switch(g_settings.menu_pos)
|
|
{
|
|
case MENU_POS_CENTER:
|
|
x = offx + scr_x + ((scr_w - menu_width ) >> 1 );
|
|
y = offy + scr_y + ((scr_h - height - hint_height) >> 1 );
|
|
break;
|
|
|
|
case MENU_POS_TOP_LEFT:
|
|
y = offy + scr_y + 10;
|
|
x = offx + scr_x + 10;
|
|
break;
|
|
|
|
case MENU_POS_TOP_RIGHT:
|
|
y = offy + scr_y + 10;
|
|
x = offx + scr_x + scr_w - menu_width - 10;
|
|
break;
|
|
|
|
case MENU_POS_BOTTOM_LEFT:
|
|
y = offy + scr_y + scr_h - height - hint_height - 10;
|
|
x = offx + scr_x + 10;
|
|
break;
|
|
|
|
case MENU_POS_BOTTOM_RIGHT:
|
|
y = offy + scr_y + scr_h - height - hint_height - 10;
|
|
x = offx + scr_x + scr_w - menu_width - 10;
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CMenuWidget::paintItems()
|
|
{
|
|
//Item not currently on screen
|
|
if (selected >= 0)
|
|
{
|
|
while (selected < page_start[current_page])
|
|
current_page--;
|
|
while (selected >= page_start[current_page + 1])
|
|
current_page++;
|
|
}
|
|
|
|
// Scrollbar
|
|
if(total_pages>1)
|
|
{
|
|
int item_height=height-(item_start_y-y);
|
|
frameBuffer->paintBoxRel(x+ width,item_start_y, 15, item_height, COL_MENUCONTENT_PLUS_1, RADIUS_MIN);
|
|
frameBuffer->paintBoxRel(x+ width +2, item_start_y+ 2+ current_page*(item_height-4)/total_pages, 11, (item_height-4)/total_pages, COL_MENUCONTENT_PLUS_3, RADIUS_MIN);
|
|
/* background of menu items, paint every time because different items can have
|
|
* different height and this might leave artifacts otherwise after changing pages */
|
|
frameBuffer->paintBoxRel(x,item_start_y, width,item_height, COL_MENUCONTENT_PLUS_0);
|
|
}
|
|
int ypos=item_start_y;
|
|
for (int count = 0; count < (int)items.size(); count++)
|
|
{
|
|
CMenuItem* item = items[count];
|
|
|
|
if ((count >= page_start[current_page]) &&
|
|
(count < page_start[current_page + 1]))
|
|
{
|
|
item->init(x, ypos, width, iconOffset);
|
|
if( (item->isSelectable()) && (selected==-1) )
|
|
{
|
|
paintHint(count);
|
|
ypos = item->paint(true);
|
|
selected = count;
|
|
}
|
|
else
|
|
{
|
|
bool sel = selected==((signed int) count) ;
|
|
if(sel)
|
|
paintHint(count);
|
|
ypos = item->paint(sel);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* x = -1 is a marker which prevents the item from being painted on setActive changes */
|
|
item->init(-1, 0, 0, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*adds the typical menu intro with optional subhead, separator, back or cancel button and separatorline to menu*/
|
|
void CMenuWidget::addIntroItems(neutrino_locale_t subhead_text, neutrino_locale_t section_text, int buttontype)
|
|
{
|
|
if (subhead_text != NONEXISTANT_LOCALE)
|
|
addItem(new CMenuSeparator(CMenuSeparator::ALIGN_LEFT | CMenuSeparator::SUB_HEAD | CMenuSeparator::STRING, subhead_text));
|
|
|
|
addItem(GenericMenuSeparator);
|
|
|
|
if (buttontype != BTN_TYPE_NO)
|
|
{
|
|
switch (buttontype)
|
|
{
|
|
case BTN_TYPE_BACK:
|
|
GenericMenuBack->setItemButton(!g_settings.menu_left_exit ? NEUTRINO_ICON_BUTTON_HOME : NEUTRINO_ICON_BUTTON_LEFT);
|
|
addItem(GenericMenuBack);
|
|
break;
|
|
case BTN_TYPE_CANCEL:
|
|
addItem(GenericMenuCancel);
|
|
break;
|
|
case BTN_TYPE_NEXT:
|
|
addItem(GenericMenuNext);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (section_text != NONEXISTANT_LOCALE)
|
|
addItem(new CMenuSeparator(CMenuSeparator::LINE | CMenuSeparator::STRING, section_text));
|
|
else if (buttontype != BTN_TYPE_NO)
|
|
addItem(GenericMenuSeparatorLine);
|
|
}
|
|
|
|
void CMenuWidget::saveScreen()
|
|
{
|
|
if(!savescreen)
|
|
return;
|
|
|
|
delete[] background;
|
|
|
|
background = new fb_pixel_t [full_width * full_height];
|
|
if(background)
|
|
frameBuffer->SaveScreen(x /*-ConnectLineBox_Width*/, y, full_width, full_height, background);
|
|
}
|
|
|
|
void CMenuWidget::restoreScreen()
|
|
{
|
|
if(background) {
|
|
if(savescreen)
|
|
frameBuffer->RestoreScreen(x /*-ConnectLineBox_Width*/, y, full_width, full_height, background);
|
|
}
|
|
}
|
|
|
|
void CMenuWidget::enableSaveScreen(bool enable)
|
|
{
|
|
savescreen = enable;
|
|
if (!enable && background) {
|
|
delete[] background;
|
|
background = NULL;
|
|
}
|
|
}
|
|
|
|
void CMenuWidget::paintHint(int pos)
|
|
{
|
|
if (!g_settings.show_menu_hints)
|
|
return;
|
|
|
|
if (pos < 0 && !hint_painted)
|
|
return;
|
|
|
|
if (hint_painted) {
|
|
/* clear detailsline line */
|
|
if (details_line)
|
|
savescreen ? details_line->hide() : details_line->kill();
|
|
/* clear info box */
|
|
if ((info_box) && (pos < 0))
|
|
savescreen ? info_box->hide(true) : info_box->kill();
|
|
hint_painted = false;
|
|
}
|
|
if (pos < 0)
|
|
return;
|
|
|
|
CMenuItem* item = items[pos];
|
|
|
|
if (item->hintIcon.empty() && item->hint == NONEXISTANT_LOCALE) {
|
|
if (info_box) {
|
|
savescreen ? info_box->hide(false) : info_box->kill();
|
|
hint_painted = false;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (item->hint == NONEXISTANT_LOCALE)
|
|
return;
|
|
|
|
int iheight = item->getHeight();
|
|
int rad = RADIUS_LARGE;
|
|
int xpos = x - ConnectLineBox_Width;
|
|
int ypos2 = y + height + rad + SHADOW_OFFSET + INFO_BOX_Y_OFFSET;
|
|
int iwidth = width+sb_width;
|
|
|
|
//init details line and infobox dimensions
|
|
int ypos1 = item->getYPosition();
|
|
int ypos1a = ypos1 + (iheight/2)-2;
|
|
int ypos2a = ypos2 + (hint_height/2) - INFO_BOX_Y_OFFSET;
|
|
int markh = hint_height > rad*2 ? hint_height - rad*2 : hint_height;
|
|
int imarkh = iheight/2+1;
|
|
|
|
//init details line
|
|
if (details_line){
|
|
details_line->setXPos(xpos);
|
|
details_line->setYPos(ypos1a);
|
|
details_line->setYPosDown(ypos2a);
|
|
details_line->setHMarkTop(imarkh);
|
|
details_line->setHMarkDown(markh);
|
|
details_line->syncSysColors();
|
|
}
|
|
|
|
//init infobox
|
|
std::string str = g_Locale->getText(item->hint);
|
|
if (info_box){
|
|
info_box->setDimensionsAll(x, ypos2, iwidth, hint_height);
|
|
info_box->setFrameThickness(2);
|
|
info_box->removeLineBreaks(str);
|
|
info_box->setText(str, CTextBox::AUTO_WIDTH, g_Font[SNeutrinoSettings::FONT_TYPE_MENU_HINT]);
|
|
info_box->setCorner(RADIUS_LARGE);
|
|
info_box->syncSysColors();
|
|
info_box->setColorBody(COL_MENUCONTENTDARK_PLUS_0);
|
|
info_box->setShadowOnOff(CC_SHADOW_ON);
|
|
info_box->setPicture(item->hintIcon);
|
|
}
|
|
|
|
//paint result
|
|
details_line->paint(savescreen);
|
|
info_box->paint(savescreen);
|
|
|
|
hint_painted = true;
|
|
}
|
|
|
|
void CMenuWidget::addKey(neutrino_msg_t key, CMenuTarget *menue, const std::string & action)
|
|
{
|
|
keyActionMap[key].menue = menue;
|
|
keyActionMap[key].action = action;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------------------------------------------------------------
|
|
CMenuOptionNumberChooser::CMenuOptionNumberChooser(const neutrino_locale_t Name, int * const OptionValue, const bool Active, const int min_value, const int max_value, CChangeObserver * const Observ, const int print_offset, const int special_value, const neutrino_locale_t special_value_name, bool sliderOn)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
name = Name;
|
|
active = Active;
|
|
optionValue = OptionValue;
|
|
|
|
lower_bound = min_value;
|
|
upper_bound = max_value;
|
|
|
|
display_offset = print_offset;
|
|
|
|
localized_value = special_value;
|
|
localized_value_name = special_value_name;
|
|
|
|
nameString = "";
|
|
numberFormat = "%d";
|
|
numberFormatFunction = NULL;
|
|
observ = Observ;
|
|
slider_on = sliderOn;
|
|
|
|
numeric_input = false;
|
|
}
|
|
|
|
CMenuOptionNumberChooser::CMenuOptionNumberChooser(const std::string &Name, int * const OptionValue, const bool Active, const int min_value, const int max_value, CChangeObserver * const Observ, const int print_offset, const int special_value, const neutrino_locale_t special_value_name, bool sliderOn)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
name = NONEXISTANT_LOCALE;
|
|
active = Active;
|
|
optionValue = OptionValue;
|
|
|
|
lower_bound = min_value;
|
|
upper_bound = max_value;
|
|
|
|
display_offset = print_offset;
|
|
|
|
localized_value = special_value;
|
|
localized_value_name = special_value_name;
|
|
|
|
nameString = Name;
|
|
numberFormat = "%d";
|
|
numberFormatFunction = NULL;
|
|
observ = Observ;
|
|
slider_on = sliderOn;
|
|
|
|
numeric_input = false;
|
|
}
|
|
|
|
int CMenuOptionNumberChooser::exec(CMenuTarget*)
|
|
{
|
|
int res = menu_return::RETURN_NONE;
|
|
|
|
if(msg == CRCInput::RC_left) {
|
|
if (((*optionValue) > upper_bound) || ((*optionValue) <= lower_bound))
|
|
*optionValue = upper_bound;
|
|
else
|
|
(*optionValue)--;
|
|
} else if (numeric_input && msg == CRCInput::RC_ok) {
|
|
int size = 0;
|
|
int b = lower_bound;
|
|
if (b < 0) {
|
|
size++,
|
|
b = -b;
|
|
}
|
|
if (b < upper_bound)
|
|
b = upper_bound;
|
|
for (; b; b /= 10, size++);
|
|
CIntInput cii(name, optionValue, size, LOCALE_IPSETUP_HINT_1, LOCALE_IPSETUP_HINT_2);
|
|
cii.exec(NULL, "");
|
|
if (*optionValue > upper_bound)
|
|
*optionValue = upper_bound;
|
|
else if (*optionValue < lower_bound)
|
|
*optionValue = lower_bound;
|
|
res = menu_return::RETURN_REPAINT;
|
|
} else {
|
|
if (((*optionValue) >= upper_bound) || ((*optionValue) < lower_bound))
|
|
*optionValue = lower_bound;
|
|
else
|
|
(*optionValue)++;
|
|
}
|
|
#if ENABLE_LUA
|
|
if(observ && !luaAction.empty()) {
|
|
// optionValue is int*
|
|
observ->changeNotify(luaState, luaAction, luaId, (void *) to_string(*optionValue).c_str());
|
|
} else
|
|
#endif
|
|
if(observ)
|
|
observ->changeNotify(name, optionValue);
|
|
|
|
// give the observer a chance to modify the value
|
|
paint(true);
|
|
|
|
return res;
|
|
}
|
|
|
|
int CMenuOptionNumberChooser::paint(bool selected)
|
|
{
|
|
const char * l_option;
|
|
char option_value[40];
|
|
|
|
if ((localized_value_name == NONEXISTANT_LOCALE) || ((*optionValue) != localized_value))
|
|
{
|
|
if (numberFormatFunction) {
|
|
std::string s = numberFormatFunction(*optionValue + display_offset);
|
|
strncpy(option_value, s.c_str(), s.length());
|
|
} else
|
|
sprintf(option_value, numberFormat.c_str(), *optionValue + display_offset);
|
|
l_option = option_value;
|
|
}
|
|
else
|
|
l_option = g_Locale->getText(localized_value_name);
|
|
|
|
//paint item
|
|
prepareItem(selected, height);
|
|
|
|
//paint item icon
|
|
paintItemButton(selected, height, NEUTRINO_ICON_BUTTON_OKAY);
|
|
if(slider_on)
|
|
paintItemSlider( selected, height, *optionValue, (upper_bound - lower_bound) , getName(), l_option);
|
|
//paint text
|
|
paintItemCaption(selected, height , getName(), l_option);
|
|
|
|
return y+height;
|
|
}
|
|
|
|
int CMenuOptionNumberChooser::getWidth(void)
|
|
{
|
|
int width = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(getName(), true);
|
|
int _lower_bound = lower_bound;
|
|
int _upper_bound = upper_bound;
|
|
int m = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getMaxDigitWidth();
|
|
|
|
int w1 = 0;
|
|
if (_lower_bound < 0)
|
|
w1 += g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth("-", true);
|
|
while (_lower_bound) {
|
|
w1 += m;
|
|
_lower_bound /= 10;
|
|
}
|
|
|
|
int w2 = 0;
|
|
if (_upper_bound < 0)
|
|
w2 += g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth("-", true);
|
|
while (_upper_bound) {
|
|
w1 += m;
|
|
_upper_bound /= 10;
|
|
}
|
|
|
|
width += (w1 > w2) ? w1 : w2;
|
|
|
|
if (numberFormatFunction) {
|
|
std::string s = numberFormatFunction(0);
|
|
width += g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(s.c_str(), true) - g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth("0", true); // arbitrary
|
|
} else if (numberFormat != "%d") {
|
|
char format[numberFormat.length()];
|
|
snprintf(format, numberFormat.length(), numberFormat.c_str(), 0);
|
|
width += g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(format, true) - g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth("0", true);
|
|
}
|
|
|
|
return width + 10; /* min 10 pixels between option name and value. enough? */
|
|
}
|
|
|
|
//-------------------------------------------------------------------------------------------------------------------------------
|
|
|
|
CMenuOptionChooser::CMenuOptionChooser(const neutrino_locale_t OptionName, int * const OptionValue, const struct keyval * const Options, const unsigned Number_Of_Options, const bool Active, CChangeObserver * const Observ, const neutrino_msg_t DirectKey, const std::string & IconName, bool Pulldown, bool OptionsSort)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
nameString = "";
|
|
name = OptionName;
|
|
active = Active;
|
|
optionValue = OptionValue;
|
|
number_of_options = Number_Of_Options;
|
|
observ = Observ;
|
|
directKey = DirectKey;
|
|
iconName = IconName;
|
|
pulldown = Pulldown;
|
|
optionsSort = OptionsSort;
|
|
for (unsigned int i = 0; i < number_of_options; i++)
|
|
{
|
|
struct keyval_ext opt;
|
|
opt.key = Options[i].key;
|
|
opt.value = Options[i].value;
|
|
opt.valname = NULL;
|
|
options.push_back(opt);
|
|
}
|
|
}
|
|
|
|
CMenuOptionChooser::CMenuOptionChooser(const std::string &OptionName, int * const OptionValue, const struct keyval * const Options, const unsigned Number_Of_Options, const bool Active, CChangeObserver * const Observ, const neutrino_msg_t DirectKey, const std::string & IconName, bool Pulldown, bool OptionsSort)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
nameString = OptionName;
|
|
name = NONEXISTANT_LOCALE;
|
|
active = Active;
|
|
optionValue = OptionValue;
|
|
number_of_options = Number_Of_Options;
|
|
observ = Observ;
|
|
directKey = DirectKey;
|
|
iconName = IconName;
|
|
pulldown = Pulldown;
|
|
optionsSort = OptionsSort;
|
|
for (unsigned int i = 0; i < number_of_options; i++)
|
|
{
|
|
struct keyval_ext opt;
|
|
opt.key = Options[i].key;
|
|
opt.value = Options[i].value;
|
|
opt.valname = NULL;
|
|
options.push_back(opt);
|
|
}
|
|
}
|
|
|
|
CMenuOptionChooser::CMenuOptionChooser(const neutrino_locale_t OptionName, int * const OptionValue, const struct keyval_ext * const Options,
|
|
const unsigned Number_Of_Options, const bool Active, CChangeObserver * const Observ,
|
|
const neutrino_msg_t DirectKey, const std::string & IconName, bool Pulldown, bool OptionsSort)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
nameString = "";
|
|
name = OptionName;
|
|
active = Active;
|
|
optionValue = OptionValue;
|
|
number_of_options = Number_Of_Options;
|
|
observ = Observ;
|
|
directKey = DirectKey;
|
|
iconName = IconName;
|
|
pulldown = Pulldown;
|
|
optionsSort = OptionsSort;
|
|
for (unsigned int i = 0; i < number_of_options; i++)
|
|
options.push_back(Options[i]);
|
|
}
|
|
|
|
CMenuOptionChooser::CMenuOptionChooser(const std::string &OptionName, int * const OptionValue, const struct keyval_ext * const Options,
|
|
const unsigned Number_Of_Options, const bool Active, CChangeObserver * const Observ,
|
|
const neutrino_msg_t DirectKey, const std::string & IconName, bool Pulldown, bool OptionsSort)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
nameString = OptionName;
|
|
name = NONEXISTANT_LOCALE;
|
|
active = Active;
|
|
optionValue = OptionValue;
|
|
number_of_options = Number_Of_Options;
|
|
observ = Observ;
|
|
directKey = DirectKey;
|
|
iconName = IconName;
|
|
pulldown = Pulldown;
|
|
optionsSort = OptionsSort;
|
|
for (unsigned int i = 0; i < number_of_options; i++)
|
|
options.push_back(Options[i]);
|
|
}
|
|
|
|
CMenuOptionChooser::CMenuOptionChooser(const neutrino_locale_t OptionName, int * const OptionValue, std::vector<keyval_ext> &Options, const bool Active, CChangeObserver * const Observ, const neutrino_msg_t DirectKey, const std::string & IconName, bool Pulldown, bool OptionsSort)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
nameString = "";
|
|
name = OptionName;
|
|
active = Active;
|
|
optionValue = OptionValue;
|
|
options = Options;
|
|
observ = Observ;
|
|
directKey = DirectKey;
|
|
iconName = IconName;
|
|
pulldown = Pulldown;
|
|
optionsSort = OptionsSort;
|
|
}
|
|
|
|
CMenuOptionChooser::CMenuOptionChooser(const std::string &OptionName, int * const OptionValue, std::vector<keyval_ext> &Options, const bool Active, CChangeObserver * const Observ, const neutrino_msg_t DirectKey, const std::string & IconName, bool Pulldown, bool OptionsSort)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
nameString = OptionName;
|
|
name = NONEXISTANT_LOCALE;
|
|
active = Active;
|
|
optionValue = OptionValue;
|
|
options = Options;
|
|
observ = Observ;
|
|
directKey = DirectKey;
|
|
iconName = IconName;
|
|
pulldown = Pulldown;
|
|
optionsSort = OptionsSort;
|
|
}
|
|
|
|
CMenuOptionChooser::~CMenuOptionChooser()
|
|
{
|
|
options.clear();
|
|
clearChooserOptions();
|
|
}
|
|
|
|
void CMenuOptionChooser::setOption(const int newvalue)
|
|
{
|
|
*optionValue = newvalue;
|
|
}
|
|
|
|
int CMenuOptionChooser::getOption(void) const
|
|
{
|
|
return *optionValue;
|
|
}
|
|
|
|
void CMenuOptionChooser::clearChooserOptions()
|
|
{
|
|
for (std::vector<CMenuOptionChooserOptions*>::iterator it = option_chooser_options_v.begin(); it != option_chooser_options_v.end(); ++it)
|
|
delete *it;
|
|
|
|
option_chooser_options_v.clear();
|
|
}
|
|
|
|
int CMenuOptionChooser::exec(CMenuTarget*)
|
|
{
|
|
bool wantsRepaint = false;
|
|
int ret = menu_return::RETURN_NONE;
|
|
#if ENABLE_LUA
|
|
char *optionValname = NULL;
|
|
#endif
|
|
if (optionsSort) {
|
|
optionsSort = false;
|
|
clearChooserOptions();
|
|
unsigned int i1;
|
|
for (i1 = 0; i1 < number_of_options; i1++)
|
|
{
|
|
CMenuOptionChooserOptions* co = new CMenuOptionChooserOptions();
|
|
co->key = options[i1].key;
|
|
co->valname = (options[i1].valname != 0) ? options[i1].valname : g_Locale->getText(options[i1].value);
|
|
option_chooser_options_v.push_back(co);
|
|
}
|
|
|
|
sort(option_chooser_options_v.begin(), option_chooser_options_v.end(), CMenuOptionChooserCompareItem());
|
|
|
|
options.clear();
|
|
for (std::vector<CMenuOptionChooserOptions*>::iterator it = option_chooser_options_v.begin(); it != option_chooser_options_v.end(); ++it) {
|
|
struct keyval_ext opt;
|
|
opt.key = (*it)->key;
|
|
opt.value = NONEXISTANT_LOCALE;
|
|
opt.valname = (*it)->valname.c_str();
|
|
options.push_back(opt);
|
|
}
|
|
}
|
|
|
|
if((msg == CRCInput::RC_ok) && pulldown) {
|
|
int select = -1;
|
|
CMenuWidget* menu = new CMenuWidget(getName(), NEUTRINO_ICON_SETTINGS);
|
|
/* FIXME: BACK button with hints enabled - parent menu getting holes, possible solution
|
|
* to hide parent, or add hints ? */
|
|
menu->addIntroItems(NONEXISTANT_LOCALE, NONEXISTANT_LOCALE, CMenuWidget::BTN_TYPE_CANCEL);
|
|
// menu->move(20, 0);
|
|
CMenuSelectorTarget * selector = new CMenuSelectorTarget(&select);
|
|
for(unsigned int count = 0; count < number_of_options; count++)
|
|
{
|
|
bool selected = false;
|
|
const char * l_option;
|
|
if (options[count].key == (*optionValue))
|
|
selected = true;
|
|
|
|
if(options[count].valname != 0)
|
|
l_option = options[count].valname;
|
|
else
|
|
l_option = g_Locale->getText(options[count].value);
|
|
CMenuForwarder *mn_option = new CMenuForwarder(l_option, true, NULL, selector, to_string(count).c_str());
|
|
mn_option->setItemButton(NEUTRINO_ICON_BUTTON_OKAY, true /*for selected item*/);
|
|
menu->addItem(mn_option, selected);
|
|
}
|
|
menu->exec(NULL, "");
|
|
ret = menu_return::RETURN_REPAINT;;
|
|
if(select >= 0)
|
|
{
|
|
*optionValue = options[select].key;
|
|
#if ENABLE_LUA
|
|
optionValname = (char *) options[select].valname;
|
|
#endif
|
|
}
|
|
delete menu;
|
|
delete selector;
|
|
} else {
|
|
for(unsigned int count = 0; count < number_of_options; count++) {
|
|
if (options[count].key == (*optionValue)) {
|
|
if(msg == CRCInput::RC_left) {
|
|
if(count > 0)
|
|
#if ENABLE_LUA
|
|
optionValname = (char *) options[(count-1) % number_of_options].valname,
|
|
#endif
|
|
*optionValue = options[(count-1) % number_of_options].key;
|
|
else
|
|
#if ENABLE_LUA
|
|
optionValname = (char *) options[number_of_options-1].valname,
|
|
#endif
|
|
*optionValue = options[number_of_options-1].key;
|
|
} else
|
|
#if ENABLE_LUA
|
|
optionValname = (char *) options[(count+1) % number_of_options].valname,
|
|
#endif
|
|
*optionValue = options[(count+1) % number_of_options].key;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
paint(true);
|
|
#if ENABLE_LUA
|
|
if(observ && !luaAction.empty()) {
|
|
if (optionValname)
|
|
wantsRepaint = observ->changeNotify(luaState, luaAction, luaId, optionValname);
|
|
} else
|
|
#endif
|
|
if(observ)
|
|
wantsRepaint = observ->changeNotify(name, optionValue);
|
|
|
|
if ( wantsRepaint )
|
|
ret = menu_return::RETURN_REPAINT;
|
|
|
|
return ret;
|
|
}
|
|
|
|
int CMenuOptionChooser::paint( bool selected)
|
|
{
|
|
neutrino_locale_t option = NONEXISTANT_LOCALE;
|
|
const char * l_option = NULL;
|
|
|
|
for(unsigned int count = 0 ; count < number_of_options; count++) {
|
|
if (options[count].key == *optionValue) {
|
|
option = options[count].value;
|
|
if(options[count].valname != 0)
|
|
l_option = options[count].valname;
|
|
else
|
|
l_option = g_Locale->getText(option);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(l_option == NULL)
|
|
{
|
|
*optionValue = options[0].key;
|
|
option = options[0].value;
|
|
if(options[0].valname != 0)
|
|
l_option = options[0].valname;
|
|
else
|
|
l_option = g_Locale->getText(option);
|
|
}
|
|
|
|
//paint item
|
|
prepareItem(selected, height);
|
|
|
|
//paint item icon
|
|
paintItemButton(selected, height, NEUTRINO_ICON_BUTTON_OKAY);
|
|
|
|
//paint text
|
|
paintItemCaption(selected, height , getName(), l_option);
|
|
|
|
return y+height;
|
|
}
|
|
|
|
int CMenuOptionChooser::getWidth(void)
|
|
{
|
|
int ow = 0;
|
|
int tw = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(getName(), true);
|
|
int width = tw;
|
|
|
|
for(unsigned int count = 0; count < options.size(); count++) {
|
|
ow = 0;
|
|
if (options[count].valname)
|
|
ow = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]
|
|
->getRenderWidth(options[count].valname, true);
|
|
else
|
|
ow = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]
|
|
->getRenderWidth(g_Locale->getText(options[count].value), true);
|
|
|
|
if (tw + ow > width)
|
|
width = tw + ow;
|
|
}
|
|
|
|
return width + 10; /* min 10 pixels between option name and value. enough? */
|
|
}
|
|
|
|
//-------------------------------------------------------------------------------------------------------------------------------
|
|
|
|
CMenuOptionStringChooser::CMenuOptionStringChooser(const neutrino_locale_t OptionName, std::string* OptionValue, bool Active, CChangeObserver* Observ, const neutrino_msg_t DirectKey, const std::string & IconName, bool Pulldown)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
nameString = "";
|
|
name = OptionName;
|
|
active = Active;
|
|
optionValueString = OptionValue;
|
|
observ = Observ;
|
|
directKey = DirectKey;
|
|
iconName = IconName;
|
|
pulldown = Pulldown;
|
|
}
|
|
|
|
CMenuOptionStringChooser::CMenuOptionStringChooser(const std::string &OptionName, std::string* OptionValue, bool Active, CChangeObserver* Observ, const neutrino_msg_t DirectKey, const std::string & IconName, bool Pulldown)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
nameString = OptionName;
|
|
name = NONEXISTANT_LOCALE;
|
|
active = Active;
|
|
optionValueString = OptionValue;
|
|
observ = Observ;
|
|
directKey = DirectKey;
|
|
iconName = IconName;
|
|
pulldown = Pulldown;
|
|
}
|
|
|
|
|
|
CMenuOptionStringChooser::~CMenuOptionStringChooser()
|
|
{
|
|
options.clear();
|
|
}
|
|
|
|
void CMenuOptionStringChooser::addOption(const std::string &value)
|
|
{
|
|
options.push_back(value);
|
|
}
|
|
|
|
void CMenuOptionStringChooser::sortOptions()
|
|
{
|
|
sort(options.begin(), options.end());
|
|
}
|
|
|
|
int CMenuOptionStringChooser::exec(CMenuTarget* parent)
|
|
{
|
|
bool wantsRepaint = false;
|
|
int ret = menu_return::RETURN_NONE;
|
|
|
|
if((!parent || msg == CRCInput::RC_ok) && pulldown) {
|
|
int select = -1;
|
|
|
|
if (parent)
|
|
parent->hide();
|
|
|
|
const char *l_name = (name == NONEXISTANT_LOCALE) ? nameString.c_str() : g_Locale->getText(name);
|
|
CMenuWidget* menu = new CMenuWidget(l_name, NEUTRINO_ICON_SETTINGS);
|
|
menu->addIntroItems(NONEXISTANT_LOCALE, NONEXISTANT_LOCALE, CMenuWidget::BTN_TYPE_CANCEL);
|
|
//if(parent) menu->move(20, 0);
|
|
CMenuSelectorTarget * selector = new CMenuSelectorTarget(&select);
|
|
for(unsigned int count = 0; count < options.size(); count++)
|
|
{
|
|
bool selected = optionValueString && (options[count] == *optionValueString);
|
|
CMenuForwarder *mn_option = new CMenuForwarder(options[count], true, NULL, selector, to_string(count).c_str());
|
|
mn_option->setItemButton(NEUTRINO_ICON_BUTTON_OKAY, true /*for selected item*/);
|
|
menu->addItem(mn_option, selected);
|
|
}
|
|
menu->exec(NULL, "");
|
|
ret = menu_return::RETURN_REPAINT;
|
|
if(select >= 0 && optionValueString)
|
|
*optionValueString = options[select];
|
|
delete menu;
|
|
delete selector;
|
|
} else {
|
|
//select next value
|
|
for(unsigned int count = 0; count < options.size(); count++) {
|
|
if (optionValueString && (options[count] == *optionValueString)) {
|
|
if(msg == CRCInput::RC_left) {
|
|
if(count > 0)
|
|
*optionValueString = options[(count - 1) % options.size()];
|
|
else
|
|
*optionValueString = options[options.size() - 1];
|
|
} else
|
|
*optionValueString = options[(count + 1) % options.size()];
|
|
//wantsRepaint = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
paint(true);
|
|
}
|
|
#if ENABLE_LUA
|
|
if(observ && !luaAction.empty())
|
|
wantsRepaint = observ->changeNotify(luaState, luaAction, luaId, (void *)(optionValueString ? optionValueString->c_str() : ""));
|
|
else
|
|
#endif
|
|
if(observ) {
|
|
wantsRepaint = observ->changeNotify(name, (void *)(optionValueString ? optionValueString->c_str() : ""));
|
|
}
|
|
if (wantsRepaint)
|
|
ret = menu_return::RETURN_REPAINT;
|
|
|
|
return ret;
|
|
}
|
|
|
|
int CMenuOptionStringChooser::paint( bool selected )
|
|
{
|
|
const char *l_name = (name == NONEXISTANT_LOCALE) ? nameString.c_str() : g_Locale->getText(name);
|
|
|
|
//paint item
|
|
prepareItem(selected, height);
|
|
|
|
//paint item icon
|
|
paintItemButton(selected, height, NEUTRINO_ICON_BUTTON_OKAY);
|
|
|
|
//paint text
|
|
paintItemCaption(selected, height , l_name, optionValueString->c_str());
|
|
|
|
return y+height;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------------------------------------------------------------
|
|
|
|
CMenuOptionLanguageChooser::CMenuOptionLanguageChooser(char* OptionValue, CChangeObserver* Observ, const char * const IconName)
|
|
{
|
|
height = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
optionValue = OptionValue;
|
|
observ = Observ;
|
|
|
|
directKey = CRCInput::RC_nokey;
|
|
iconName = IconName ? IconName : "";
|
|
}
|
|
|
|
|
|
CMenuOptionLanguageChooser::~CMenuOptionLanguageChooser()
|
|
{
|
|
}
|
|
|
|
int CMenuOptionLanguageChooser::exec(CMenuTarget*)
|
|
{
|
|
g_settings.language = optionValue;
|
|
if(observ)
|
|
observ->changeNotify(LOCALE_LANGUAGESETUP_SELECT, (void *) optionValue.c_str());
|
|
return menu_return::RETURN_EXIT;
|
|
}
|
|
|
|
int CMenuOptionLanguageChooser::paint( bool selected )
|
|
{
|
|
active = true;
|
|
|
|
//paint item
|
|
prepareItem(selected, height);
|
|
|
|
paintItemButton(selected, height, iconName);
|
|
|
|
//convert first letter to large
|
|
std::string s_optionValue = optionValue;
|
|
if(!s_optionValue.empty())
|
|
s_optionValue[0] = (char)toupper(s_optionValue[0]);
|
|
//paint text
|
|
paintItemCaption(selected, height , s_optionValue.c_str());
|
|
|
|
return y+height;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------------------------------------------------------------
|
|
CMenuForwarder::CMenuForwarder(const neutrino_locale_t Text, const bool Active, const std::string &Option, CMenuTarget* Target, const char * const ActionKey, neutrino_msg_t DirectKey, const char * const IconName, const char * const IconName_Info_right, bool IsStatic)
|
|
{
|
|
option_string_ptr = &Option;
|
|
name = Text;
|
|
nameString = "";
|
|
active = Active;
|
|
jumpTarget = Target;
|
|
actionKey = ActionKey ? ActionKey : "";
|
|
directKey = DirectKey;
|
|
iconName = IconName ? IconName : "";
|
|
iconName_Info_right = IconName_Info_right ? IconName_Info_right : "";
|
|
isStatic = IsStatic;
|
|
}
|
|
|
|
CMenuForwarder::CMenuForwarder(const std::string& Text, const bool Active, const std::string &Option, CMenuTarget* Target, const char * const ActionKey, neutrino_msg_t DirectKey, const char * const IconName, const char * const IconName_Info_right, bool IsStatic)
|
|
{
|
|
option_string_ptr = &Option;
|
|
name = NONEXISTANT_LOCALE;
|
|
nameString = Text;
|
|
active = Active;
|
|
jumpTarget = Target;
|
|
actionKey = ActionKey ? ActionKey : "";
|
|
directKey = DirectKey;
|
|
iconName = IconName ? IconName : "";
|
|
iconName_Info_right = IconName_Info_right ? IconName_Info_right : "";
|
|
isStatic = IsStatic;
|
|
}
|
|
|
|
CMenuForwarder::CMenuForwarder(const neutrino_locale_t Text, const bool Active, const char * const Option, CMenuTarget* Target, const char * const ActionKey, neutrino_msg_t DirectKey, const char * const IconName, const char * const IconName_Info_right, bool IsStatic)
|
|
{
|
|
option_string = Option ? Option : "";
|
|
option_string_ptr = &option_string;
|
|
name = Text;
|
|
nameString = "";
|
|
active = Active;
|
|
jumpTarget = Target;
|
|
actionKey = ActionKey ? ActionKey : "";
|
|
directKey = DirectKey;
|
|
iconName = IconName ? IconName : "";
|
|
iconName_Info_right = IconName_Info_right ? IconName_Info_right : "";
|
|
isStatic = IsStatic;
|
|
}
|
|
|
|
CMenuForwarder::CMenuForwarder(const std::string& Text, const bool Active, const char * const Option, CMenuTarget* Target, const char * const ActionKey, neutrino_msg_t DirectKey, const char * const IconName, const char * const IconName_Info_right, bool IsStatic)
|
|
{
|
|
option_string = Option ? Option : "";
|
|
option_string_ptr = &option_string;
|
|
name = NONEXISTANT_LOCALE;
|
|
nameString = Text;
|
|
active = Active;
|
|
jumpTarget = Target;
|
|
actionKey = ActionKey ? ActionKey : "";
|
|
directKey = DirectKey;
|
|
iconName = IconName ? IconName : "";
|
|
iconName_Info_right = IconName_Info_right ? IconName_Info_right : "";
|
|
isStatic = IsStatic;
|
|
}
|
|
|
|
void CMenuForwarder::setName(const std::string& t)
|
|
{
|
|
name = NONEXISTANT_LOCALE;
|
|
nameString = t;
|
|
}
|
|
|
|
void CMenuForwarder::setName(const neutrino_locale_t t)
|
|
{
|
|
name = t;
|
|
nameString = "";
|
|
}
|
|
|
|
void CMenuForwarder::setOption(const std::string &Option)
|
|
{
|
|
option_string = Option;
|
|
}
|
|
|
|
int CMenuForwarder::getHeight(void) const
|
|
{
|
|
return g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
}
|
|
|
|
int CMenuForwarder::getWidth(void)
|
|
{
|
|
const char *_name = (name == NONEXISTANT_LOCALE) ? nameString.c_str() : g_Locale->getText(name);
|
|
int tw = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(_name, true);
|
|
|
|
fb_pixel_t bgcol = 0;
|
|
std::string option_name = getOption();
|
|
if (jumpTarget)
|
|
bgcol = jumpTarget->getColor();
|
|
|
|
if (!option_name.empty())
|
|
tw += 10 + g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(option_name.c_str(), true);
|
|
else if (bgcol)
|
|
tw += 10 + 60;
|
|
|
|
return tw;
|
|
}
|
|
|
|
int CMenuForwarder::exec(CMenuTarget* parent)
|
|
{
|
|
if(jumpTarget)
|
|
{
|
|
return jumpTarget->exec(parent, actionKey);
|
|
}
|
|
else
|
|
{
|
|
return menu_return::RETURN_EXIT;
|
|
}
|
|
}
|
|
|
|
std::string CMenuForwarder::getOption(void)
|
|
{
|
|
if (!option_string_ptr->empty())
|
|
return *option_string_ptr;
|
|
if (jumpTarget)
|
|
return jumpTarget->getValue();
|
|
return "";
|
|
}
|
|
|
|
int CMenuForwarder::paint(bool selected)
|
|
{
|
|
int height = getHeight();
|
|
const char * l_name = getName();
|
|
|
|
std::string option_name = getOption();
|
|
fb_pixel_t bgcol = 0;
|
|
if (jumpTarget)
|
|
bgcol = jumpTarget->getColor();
|
|
|
|
//paint item
|
|
prepareItem(selected, height);
|
|
|
|
//paint icon
|
|
paintItemButton(selected, height);
|
|
|
|
//caption
|
|
paintItemCaption(selected, height, l_name, option_name.c_str(), bgcol);
|
|
|
|
return y+ height;
|
|
}
|
|
|
|
//-------------------------------------------------------------------------------------------------------------------------------
|
|
CMenuSeparator::CMenuSeparator(const int Type, const neutrino_locale_t Text, bool IsStatic)
|
|
{
|
|
directKey = CRCInput::RC_nokey;
|
|
iconName = "";
|
|
type = Type;
|
|
name = Text;
|
|
nameString = "";
|
|
isStatic = IsStatic;
|
|
}
|
|
|
|
CMenuSeparator::CMenuSeparator(const int Type, const std::string Text, bool IsStatic)
|
|
{
|
|
directKey = CRCInput::RC_nokey;
|
|
iconName = "";
|
|
type = Type;
|
|
name = NONEXISTANT_LOCALE;
|
|
nameString = Text;
|
|
isStatic = IsStatic;
|
|
}
|
|
|
|
int CMenuSeparator::getHeight(void) const
|
|
{
|
|
if (nameString.empty() && name == NONEXISTANT_LOCALE)
|
|
return 10;
|
|
return g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getHeight();
|
|
}
|
|
|
|
void CMenuSeparator::setName(const std::string& t)
|
|
{
|
|
name = NONEXISTANT_LOCALE;
|
|
nameString = t;
|
|
}
|
|
|
|
void CMenuSeparator::setName(const neutrino_locale_t t)
|
|
{
|
|
name = t;
|
|
nameString = "";
|
|
}
|
|
|
|
int CMenuSeparator::getWidth(void)
|
|
{
|
|
int w = 0;
|
|
if (type & LINE)
|
|
w = 30; /* 15 pixel left and right */
|
|
const char *l_name = getName();
|
|
if ((type & STRING) && *l_name)
|
|
w += g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(l_name, true);
|
|
return w;
|
|
}
|
|
|
|
int CMenuSeparator::paint(bool selected)
|
|
{
|
|
int height = getHeight();
|
|
CFrameBuffer * frameBuffer = CFrameBuffer::getInstance();
|
|
|
|
if ((type & SUB_HEAD))
|
|
{
|
|
item_color = COL_MENUHEAD_TEXT;
|
|
item_bgcolor = COL_MENUHEAD_PLUS_0;
|
|
}
|
|
else
|
|
{
|
|
item_color = COL_MENUCONTENTINACTIVE_TEXT;
|
|
item_bgcolor = COL_MENUCONTENT_PLUS_0;
|
|
}
|
|
|
|
frameBuffer->paintBoxRel(x,y, dx, height, item_bgcolor);
|
|
if ((type & LINE))
|
|
{
|
|
frameBuffer->paintHLineRel(x+10,dx-20,y+(height>>1), COL_MENUCONTENT_PLUS_3);
|
|
frameBuffer->paintHLineRel(x+10,dx-20,y+(height>>1)+1, COL_MENUCONTENT_PLUS_1);
|
|
}
|
|
if ((type & STRING))
|
|
{
|
|
const char * l_name = getName();
|
|
|
|
if (*l_name)
|
|
{
|
|
int stringwidth = g_Font[SNeutrinoSettings::FONT_TYPE_MENU]->getRenderWidth(l_name, true); // UTF-8
|
|
|
|
/* if no alignment is specified, align centered */
|
|
if (type & ALIGN_LEFT)
|
|
name_start_x = x + (!(type & SUB_HEAD) ? name_start_x : 20 + 24 /*std icon_width is 24px - this should be determinated from NEUTRINO_ICON_BUTTON_HOME or so*/);
|
|
else if (type & ALIGN_RIGHT)
|
|
name_start_x = x + dx - stringwidth - 20;
|
|
else /* ALIGN_CENTER */
|
|
name_start_x = x + (dx >> 1) - (stringwidth >> 1);
|
|
|
|
frameBuffer->paintBoxRel(name_start_x-5, y, stringwidth+10, height, item_bgcolor);
|
|
|
|
paintItemCaption(selected, height, l_name);
|
|
}
|
|
}
|
|
return y+ height;
|
|
}
|
|
|
|
bool CPINProtection::check()
|
|
{
|
|
hint = NONEXISTANT_LOCALE;
|
|
std::string cPIN;
|
|
do
|
|
{
|
|
cPIN = "";
|
|
CPINInput* PINInput = new CPINInput(title, &cPIN, 4, hint);
|
|
PINInput->exec( getParent(), "");
|
|
delete PINInput;
|
|
hint = LOCALE_PINPROTECTION_WRONGCODE;
|
|
} while ((cPIN != *validPIN) && !cPIN.empty());
|
|
return (cPIN == *validPIN);
|
|
}
|
|
|
|
bool CZapProtection::check()
|
|
{
|
|
hint = NONEXISTANT_LOCALE;
|
|
int res;
|
|
std::string cPIN;
|
|
do
|
|
{
|
|
cPIN = "";
|
|
|
|
CPLPINInput* PINInput = new CPLPINInput(title, &cPIN, 4, hint, fsk);
|
|
|
|
res = PINInput->exec(getParent(), "");
|
|
delete PINInput;
|
|
|
|
hint = LOCALE_PINPROTECTION_WRONGCODE;
|
|
} while ( (cPIN != *validPIN) && !cPIN.empty() &&
|
|
( res == menu_return::RETURN_REPAINT ) &&
|
|
( fsk >= g_settings.parentallock_lockage ) );
|
|
return ( (cPIN == *validPIN) ||
|
|
( fsk < g_settings.parentallock_lockage ) );
|
|
}
|
|
|
|
int CLockedMenuForwarder::exec(CMenuTarget* parent)
|
|
{
|
|
Parent = parent;
|
|
|
|
if (Ask && !check())
|
|
{
|
|
Parent = NULL;
|
|
return menu_return::RETURN_REPAINT;
|
|
}
|
|
|
|
Parent = NULL;
|
|
return CMenuForwarder::exec(parent);
|
|
}
|
|
|
|
int CMenuSelectorTarget::exec(CMenuTarget* /*parent*/, const std::string & actionKey)
|
|
{
|
|
if (actionKey != "")
|
|
*m_select = atoi(actionKey.c_str());
|
|
else
|
|
*m_select = -1;
|
|
return menu_return::RETURN_EXIT;
|
|
}
|
|
|