2009-09-30 23:10:58 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* libwiigui
|
|
|
|
*
|
|
|
|
* Tantric 2009
|
|
|
|
*
|
|
|
|
* gui_window.cpp
|
|
|
|
*
|
|
|
|
* GUI class definitions
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#include "gui.h"
|
|
|
|
|
2010-02-09 10:59:55 +00:00
|
|
|
GuiWindow::GuiWindow()
|
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
width = 0;
|
|
|
|
height = 0;
|
|
|
|
focus = 0; // allow focus
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
GuiWindow::GuiWindow( int w, int h )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
width = w;
|
|
|
|
height = h;
|
|
|
|
focus = 0; // allow focus
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-02-09 10:59:55 +00:00
|
|
|
GuiWindow::~GuiWindow()
|
|
|
|
{
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::Append( GuiElement* e )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
if ( e == NULL )
|
|
|
|
return;
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
Remove( e );
|
|
|
|
_elements.push_back( e );
|
|
|
|
e->SetParent( this );
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::Insert( GuiElement* e, u32 index )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
if ( e == NULL || index > ( _elements.size() - 1 ) )
|
|
|
|
return;
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
Remove( e );
|
|
|
|
_elements.insert( _elements.begin() + index, e );
|
|
|
|
e->SetParent( this );
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::Remove( GuiElement* e )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
if ( e == NULL )
|
|
|
|
return;
|
|
|
|
|
|
|
|
for ( u8 i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
if ( e == _elements.at( i ) )
|
|
|
|
{
|
|
|
|
_elements.erase( _elements.begin() + i );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-02-09 10:59:55 +00:00
|
|
|
void GuiWindow::RemoveAll()
|
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
_elements.clear();
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
GuiElement* GuiWindow::GetGuiElementAt( u32 index ) const
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
if ( index >= _elements.size() )
|
|
|
|
return NULL;
|
|
|
|
return _elements.at( index );
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-02-09 10:59:55 +00:00
|
|
|
u32 GuiWindow::GetSize()
|
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
return _elements.size();
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-02-09 10:59:55 +00:00
|
|
|
void GuiWindow::Draw()
|
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
if ( _elements.size() == 0 || !this->IsVisible() )
|
|
|
|
return;
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
for ( u8 i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try { _elements.at( i )->Draw(); }
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
this->UpdateEffects();
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
if ( parentElement && state == STATE_DISABLED )
|
|
|
|
//Menu_DrawRectangle(0,0,screenwidth,screenheight,(GXColor){0xbe, 0xca, 0xd5, 0x70},1);
|
|
|
|
Menu_DrawRectangle( 0, 0, screenwidth, screenheight, ( GXColor ) {0, 0, 0, 0x70}, 1 );
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
2010-02-09 10:59:55 +00:00
|
|
|
void GuiWindow::DrawTooltip()
|
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
if ( _elements.size() == 0 || !this->IsVisible() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
for ( u8 i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try { _elements.at( i )->DrawTooltip(); }
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
2010-02-09 10:59:55 +00:00
|
|
|
void GuiWindow::ResetState()
|
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
if ( state != STATE_DISABLED )
|
|
|
|
state = STATE_DEFAULT;
|
|
|
|
|
|
|
|
for ( u8 i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try { _elements.at( i )->ResetState(); }
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::SetState( int s )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
state = s;
|
|
|
|
|
|
|
|
for ( u8 i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try { _elements.at( i )->SetState( s ); }
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::SetVisible( bool v )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
visible = v;
|
|
|
|
|
|
|
|
for ( u8 i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try { _elements.at( i )->SetVisible( v ); }
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::SetFocus( int f )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
focus = f;
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
if ( f == 1 )
|
|
|
|
this->MoveSelectionVert( 1 );
|
|
|
|
else
|
|
|
|
this->ResetState();
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::ChangeFocus( GuiElement* e )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
if ( parentElement )
|
|
|
|
return; // this is only intended for the main window
|
|
|
|
|
|
|
|
for ( u8 i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
if ( e == _elements.at( i ) )
|
|
|
|
_elements.at( i )->SetFocus( 1 );
|
|
|
|
else if ( _elements.at( i )->IsFocused() == 1 )
|
|
|
|
_elements.at( i )->SetFocus( 0 );
|
|
|
|
}
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::ToggleFocus( GuiTrigger * t )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
if ( parentElement )
|
|
|
|
return; // this is only intended for the main window
|
|
|
|
|
|
|
|
int found = -1;
|
|
|
|
int newfocus = -1;
|
|
|
|
u8 i;
|
|
|
|
|
|
|
|
// look for currently in focus element
|
|
|
|
for ( i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->IsFocused() == 1 )
|
|
|
|
{
|
|
|
|
found = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
|
|
|
|
|
|
|
// element with focus not found, try to give focus
|
|
|
|
if ( found == -1 )
|
|
|
|
{
|
|
|
|
for ( i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->IsFocused() == 0 && _elements.at( i )->GetState() != STATE_DISABLED ) // focus is possible (but not set)
|
|
|
|
{
|
|
|
|
_elements.at( i )->SetFocus( 1 ); // give this element focus
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// change focus
|
|
|
|
else if ( t->wpad.btns_d & ( WPAD_BUTTON_1 | WPAD_BUTTON_1 | WPAD_CLASSIC_BUTTON_PLUS )
|
|
|
|
|| t->pad.btns_d & PAD_BUTTON_B )
|
|
|
|
{
|
|
|
|
for ( i = found; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->IsFocused() == 0 && _elements.at( i )->GetState() != STATE_DISABLED ) // focus is possible (but not set)
|
|
|
|
{
|
|
|
|
newfocus = i;
|
|
|
|
_elements.at( i )->SetFocus( 1 ); // give this element focus
|
|
|
|
_elements.at( found )->SetFocus( 0 ); // disable focus on other element
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( newfocus == -1 )
|
|
|
|
{
|
|
|
|
for ( i = 0; i < found; i++ )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->IsFocused() == 0 && _elements.at( i )->GetState() != STATE_DISABLED ) // focus is possible (but not set)
|
|
|
|
{
|
|
|
|
_elements.at( i )->SetFocus( 1 ); // give this element focus
|
|
|
|
_elements.at( found )->SetFocus( 0 ); // disable focus on other element
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-02-09 10:59:55 +00:00
|
|
|
int GuiWindow::GetSelected()
|
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
// find selected element
|
|
|
|
int found = -1;
|
|
|
|
for ( u8 i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->GetState() == STATE_SELECTED )
|
|
|
|
{
|
|
|
|
found = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
|
|
|
return found;
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// set element to left/right as selected
|
|
|
|
// there's probably a more clever way to do this, but this way works
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::MoveSelectionHor( int dir )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
int found = -1;
|
|
|
|
u16 left = 0;
|
|
|
|
u16 top = 0;
|
|
|
|
u8 i = 0;
|
|
|
|
|
|
|
|
int selected = this->GetSelected();
|
|
|
|
|
|
|
|
if ( selected >= 0 )
|
|
|
|
{
|
|
|
|
left = _elements.at( selected )->GetLeft();
|
|
|
|
top = _elements.at( selected )->GetTop();
|
|
|
|
}
|
|
|
|
|
|
|
|
// look for a button on the same row, to the left/right
|
|
|
|
for ( i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->IsSelectable() )
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->GetLeft()*dir > left*dir && _elements.at( i )->GetTop() == top )
|
|
|
|
{
|
|
|
|
if ( found == -1 )
|
|
|
|
found = i;
|
|
|
|
else if ( _elements.at( i )->GetLeft()*dir < _elements.at( found )->GetLeft()*dir )
|
|
|
|
found = i; // this is a better match
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
|
|
|
if ( found >= 0 )
|
|
|
|
goto matchfound;
|
|
|
|
|
|
|
|
// match still not found, let's try the first button in the next row
|
|
|
|
for ( i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->IsSelectable() )
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->GetTop()*dir > top*dir )
|
|
|
|
{
|
|
|
|
if ( found == -1 )
|
|
|
|
found = i;
|
|
|
|
else if ( _elements.at( i )->GetTop()*dir < _elements.at( found )->GetTop()*dir )
|
|
|
|
found = i; // this is a better match
|
|
|
|
else if ( _elements.at( i )->GetTop()*dir == _elements.at( found )->GetTop()*dir
|
|
|
|
&&
|
|
|
|
_elements.at( i )->GetLeft()*dir < _elements.at( found )->GetLeft()*dir )
|
|
|
|
found = i; // this is a better match
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
|
|
|
|
|
|
|
// match found
|
|
|
|
matchfound:
|
|
|
|
if ( found >= 0 )
|
|
|
|
{
|
|
|
|
_elements.at( found )->SetState( STATE_SELECTED );
|
|
|
|
if ( selected >= 0 )
|
|
|
|
_elements.at( selected )->ResetState();
|
|
|
|
}
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::MoveSelectionVert( int dir )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
int found = -1;
|
|
|
|
u16 left = 0;
|
|
|
|
u16 top = 0;
|
|
|
|
u8 i = 0;
|
|
|
|
|
|
|
|
int selected = this->GetSelected();
|
|
|
|
|
|
|
|
if ( selected >= 0 )
|
|
|
|
{
|
|
|
|
left = _elements.at( selected )->GetLeft();
|
|
|
|
top = _elements.at( selected )->GetTop();
|
|
|
|
}
|
|
|
|
|
|
|
|
// look for a button above/below, with the least horizontal difference
|
|
|
|
for ( i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->IsSelectable() )
|
|
|
|
{
|
|
|
|
if ( _elements.at( i )->GetTop()*dir > top*dir )
|
|
|
|
{
|
|
|
|
if ( found == -1 )
|
|
|
|
found = i;
|
|
|
|
else if ( _elements.at( i )->GetTop()*dir < _elements.at( found )->GetTop()*dir )
|
|
|
|
found = i; // this is a better match
|
|
|
|
else if ( _elements.at( i )->GetTop()*dir == _elements.at( found )->GetTop()*dir
|
|
|
|
&&
|
|
|
|
abs( _elements.at( i )->GetLeft() - left ) <
|
|
|
|
abs( _elements.at( found )->GetLeft() - left ) )
|
|
|
|
found = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
|
|
|
if ( found >= 0 )
|
|
|
|
goto matchfound;
|
|
|
|
|
|
|
|
// match found
|
|
|
|
matchfound:
|
|
|
|
if ( found >= 0 )
|
|
|
|
{
|
|
|
|
_elements.at( found )->SetState( STATE_SELECTED );
|
|
|
|
if ( selected >= 0 )
|
|
|
|
_elements.at( selected )->ResetState();
|
|
|
|
}
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2010-09-18 23:16:05 +00:00
|
|
|
void GuiWindow::Update( GuiTrigger * t )
|
2010-02-09 10:59:55 +00:00
|
|
|
{
|
2010-09-18 23:16:05 +00:00
|
|
|
LOCK( this );
|
|
|
|
if ( _elements.size() == 0 || ( state == STATE_DISABLED && parentElement ) )
|
|
|
|
return;
|
|
|
|
|
|
|
|
for ( u8 i = 0; i < _elements.size(); i++ )
|
|
|
|
{
|
|
|
|
try { _elements.at( i )->Update( t ); }
|
|
|
|
catch ( const std::exception& e ) { }
|
|
|
|
}
|
|
|
|
|
|
|
|
this->ToggleFocus( t );
|
|
|
|
|
|
|
|
if ( focus ) // only send actions to this window if it's in focus
|
|
|
|
{
|
|
|
|
// pad/joystick navigation
|
|
|
|
if ( t->Right() )
|
|
|
|
this->MoveSelectionHor( 1 );
|
|
|
|
else if ( t->Left() )
|
|
|
|
this->MoveSelectionHor( -1 );
|
|
|
|
else if ( t->Down() )
|
|
|
|
this->MoveSelectionVert( 1 );
|
|
|
|
else if ( t->Up() )
|
|
|
|
this->MoveSelectionVert( -1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( updateCB )
|
|
|
|
updateCB( this );
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|