vbagx/source/gui/gui_button.cpp

360 lines
7.5 KiB
C++
Raw Normal View History

2009-04-07 02:57:49 +00:00
/****************************************************************************
2009-04-10 07:51:55 +00:00
* libwiigui
2009-04-07 02:57:49 +00:00
*
2009-04-10 07:51:55 +00:00
* Tantric 2009
2009-04-07 02:57:49 +00:00
*
* gui_button.cpp
*
* GUI class definitions
***************************************************************************/
#include "gui.h"
/**
* Constructor for the GuiButton class.
*/
GuiButton::GuiButton(int w, int h)
{
width = w;
height = h;
image = NULL;
imageOver = NULL;
imageHold = NULL;
imageClick = NULL;
2009-04-07 02:57:49 +00:00
icon = NULL;
iconOver = NULL;
iconHold = NULL;
iconClick = NULL;
2009-04-07 02:57:49 +00:00
for(int i=0; i < 3; i++)
{
label[i] = NULL;
labelOver[i] = NULL;
labelHold[i] = NULL;
labelClick[i] = NULL;
2009-04-07 02:57:49 +00:00
}
soundOver = NULL;
soundHold = NULL;
2009-04-07 02:57:49 +00:00
soundClick = NULL;
2010-01-25 08:25:28 +00:00
tooltip = NULL;
2009-04-07 02:57:49 +00:00
selectable = true;
holdable = false;
2009-04-07 02:57:49 +00:00
clickable = true;
}
/**
* Destructor for the GuiButton class.
*/
GuiButton::~GuiButton()
{
}
void GuiButton::SetImage(GuiImage* img)
{
image = img;
if(img) img->SetParent(this);
2009-04-07 02:57:49 +00:00
}
void GuiButton::SetImageOver(GuiImage* img)
{
imageOver = img;
if(img) img->SetParent(this);
}
void GuiButton::SetImageHold(GuiImage* img)
{
imageHold = img;
if(img) img->SetParent(this);
}
void GuiButton::SetImageClick(GuiImage* img)
{
imageClick = img;
if(img) img->SetParent(this);
2009-04-07 02:57:49 +00:00
}
void GuiButton::SetIcon(GuiImage* img)
{
icon = img;
if(img) img->SetParent(this);
2009-04-07 02:57:49 +00:00
}
void GuiButton::SetIconOver(GuiImage* img)
{
iconOver = img;
if(img) img->SetParent(this);
2009-04-07 02:57:49 +00:00
}
void GuiButton::SetIconHold(GuiImage* img)
2009-04-07 02:57:49 +00:00
{
iconHold = img;
if(img) img->SetParent(this);
2009-04-07 02:57:49 +00:00
}
void GuiButton::SetIconClick(GuiImage* img)
2009-04-07 02:57:49 +00:00
{
iconClick = img;
if(img) img->SetParent(this);
2009-04-07 02:57:49 +00:00
}
void GuiButton::SetLabel(GuiText* txt, int n)
{
label[n] = txt;
if(txt) txt->SetParent(this);
2009-04-07 02:57:49 +00:00
}
void GuiButton::SetLabelOver(GuiText* txt, int n)
{
labelOver[n] = txt;
if(txt) txt->SetParent(this);
}
void GuiButton::SetLabelHold(GuiText* txt, int n)
{
labelHold[n] = txt;
if(txt) txt->SetParent(this);
}
void GuiButton::SetLabelClick(GuiText* txt, int n)
{
labelClick[n] = txt;
if(txt) txt->SetParent(this);
2009-04-07 02:57:49 +00:00
}
void GuiButton::SetSoundOver(GuiSound * snd)
{
soundOver = snd;
}
void GuiButton::SetSoundHold(GuiSound * snd)
{
soundHold = snd;
}
2009-04-07 02:57:49 +00:00
void GuiButton::SetSoundClick(GuiSound * snd)
{
soundClick = snd;
}
2010-01-25 08:25:28 +00:00
void GuiButton::SetTooltip(GuiTooltip* t)
{
tooltip = t;
if(t)
tooltip->SetParent(this);
}
2009-04-07 02:57:49 +00:00
/**
* Draw the button on screen
*/
void GuiButton::Draw()
{
if(!this->IsVisible())
return;
2010-01-25 07:36:48 +00:00
if(state == STATE_SELECTED || state == STATE_HELD)
2009-04-07 02:57:49 +00:00
{
2010-01-25 07:36:48 +00:00
if(imageOver)
imageOver->Draw();
else if(image) // draw image
image->Draw();
if(iconOver)
iconOver->Draw();
else if(icon) // draw icon
icon->Draw();
// draw text
if(labelOver[0])
labelOver[0]->Draw();
else if(label[0])
label[0]->Draw();
if(labelOver[1])
labelOver[1]->Draw();
else if(label[1])
label[1]->Draw();
if(labelOver[2])
labelOver[2]->Draw();
else if(label[2])
label[2]->Draw();
}
else
{
if(image) // draw image
image->Draw();
if(icon) // draw icon
icon->Draw();
// draw text
if(label[0])
label[0]->Draw();
if(label[1])
label[1]->Draw();
if(label[2])
label[2]->Draw();
2009-04-07 02:57:49 +00:00
}
this->UpdateEffects();
}
2010-01-25 08:25:28 +00:00
void GuiButton::DrawTooltip()
{
if(tooltip)
tooltip->DrawTooltip();
}
void GuiButton::ResetText()
{
for(int i=0; i<3; i++)
{
if(label[i])
label[i]->ResetText();
if(labelOver[i])
labelOver[i]->ResetText();
}
2010-03-15 07:46:11 +00:00
if(tooltip)
tooltip->ResetText();
2010-01-25 08:25:28 +00:00
}
2009-04-07 02:57:49 +00:00
void GuiButton::Update(GuiTrigger * t)
{
if(state == STATE_CLICKED || state == STATE_DISABLED || !t)
return;
else if(parentElement && parentElement->GetState() == STATE_DISABLED)
return;
#ifdef HW_RVL
// cursor
2009-10-13 00:49:08 +00:00
if(t->wpad->ir.valid && t->chan >= 0)
2009-04-07 02:57:49 +00:00
{
2009-10-13 00:49:08 +00:00
if(this->IsInside(t->wpad->ir.x, t->wpad->ir.y))
2009-04-07 02:57:49 +00:00
{
if(state == STATE_DEFAULT) // we weren't on the button before!
{
this->SetState(STATE_SELECTED, t->chan);
2009-04-07 02:57:49 +00:00
if(this->Rumble())
rumbleRequest[t->chan] = 1;
if(soundOver)
soundOver->Play();
if(effectsOver && !effects)
{
// initiate effects
effects = effectsOver;
effectAmount = effectAmountOver;
effectTarget = effectTargetOver;
}
}
}
else
{
if(state == STATE_SELECTED && (stateChan == t->chan || stateChan == -1))
this->ResetState();
2009-04-07 02:57:49 +00:00
if(effectTarget == effectTargetOver && effectAmount == effectAmountOver)
{
// initiate effects (in reverse)
effects = effectsOver;
effectAmount = -effectAmountOver;
effectTarget = 100;
}
}
}
#endif
// button triggers
if(this->IsClickable())
{
s32 wm_btns, wm_btns_trig, cc_btns, cc_btns_trig;
for(int i=0; i<3; i++)
2009-04-07 02:57:49 +00:00
{
if(trigger[i] && (trigger[i]->chan == -1 || trigger[i]->chan == t->chan))
{
// higher 16 bits only (wiimote)
2009-10-13 00:49:08 +00:00
wm_btns = t->wpad->btns_d << 16;
wm_btns_trig = trigger[i]->wpad->btns_d << 16;
2009-04-07 02:57:49 +00:00
// lower 16 bits only (classic controller)
2009-10-13 00:49:08 +00:00
cc_btns = t->wpad->btns_d >> 16;
cc_btns_trig = trigger[i]->wpad->btns_d >> 16;
2009-04-07 02:57:49 +00:00
if(
2009-10-13 00:49:08 +00:00
(t->wpad->btns_d > 0 &&
2009-05-21 04:16:41 +00:00
(wm_btns == wm_btns_trig ||
2009-10-13 00:49:08 +00:00
(cc_btns == cc_btns_trig && t->wpad->exp.type == EXP_CLASSIC))) ||
(t->pad.btns_d == trigger[i]->pad.btns_d && t->pad.btns_d > 0))
2009-04-07 02:57:49 +00:00
{
if(t->chan == stateChan || stateChan == -1)
2009-04-07 02:57:49 +00:00
{
if(state == STATE_SELECTED)
{
2009-10-13 00:49:08 +00:00
if(!t->wpad->ir.valid || this->IsInside(t->wpad->ir.x, t->wpad->ir.y))
2009-05-28 07:50:30 +00:00
{
this->SetState(STATE_CLICKED, t->chan);
2009-05-28 07:50:30 +00:00
if(soundClick)
soundClick->Play();
}
}
else if(trigger[i]->type == TRIGGER_BUTTON_ONLY)
{
this->SetState(STATE_CLICKED, t->chan);
}
else if(trigger[i]->type == TRIGGER_BUTTON_ONLY_IN_FOCUS &&
parentElement->IsFocused())
{
this->SetState(STATE_CLICKED, t->chan);
}
2009-04-07 02:57:49 +00:00
}
}
}
}
}
if(this->IsHoldable())
2009-04-10 07:51:55 +00:00
{
bool held = false;
s32 wm_btns, wm_btns_h, wm_btns_trig, cc_btns, cc_btns_h, cc_btns_trig;
2009-04-10 07:51:55 +00:00
for(int i=0; i<3; i++)
2009-04-10 07:51:55 +00:00
{
if(trigger[i] && (trigger[i]->chan == -1 || trigger[i]->chan == t->chan))
{
// higher 16 bits only (wiimote)
2009-10-13 00:49:08 +00:00
wm_btns = t->wpad->btns_d << 16;
wm_btns_h = t->wpad->btns_h << 16;
wm_btns_trig = trigger[i]->wpad->btns_h << 16;
2009-04-10 07:51:55 +00:00
// lower 16 bits only (classic controller)
2009-10-13 00:49:08 +00:00
cc_btns = t->wpad->btns_d >> 16;
cc_btns_h = t->wpad->btns_h >> 16;
cc_btns_trig = trigger[i]->wpad->btns_h >> 16;
2009-04-10 07:51:55 +00:00
if(
2009-10-13 00:49:08 +00:00
(t->wpad->btns_d > 0 &&
2009-05-21 04:16:41 +00:00
(wm_btns == wm_btns_trig ||
2009-10-13 00:49:08 +00:00
(cc_btns == cc_btns_trig && t->wpad->exp.type == EXP_CLASSIC))) ||
(t->pad.btns_d == trigger[i]->pad.btns_h && t->pad.btns_d > 0))
{
if(trigger[i]->type == TRIGGER_HELD && state == STATE_SELECTED &&
(t->chan == stateChan || stateChan == -1))
this->SetState(STATE_CLICKED, t->chan);
}
if(
2009-10-13 00:49:08 +00:00
(t->wpad->btns_h > 0 &&
2009-05-21 04:16:41 +00:00
(wm_btns_h == wm_btns_trig ||
2009-10-13 00:49:08 +00:00
(cc_btns_h == cc_btns_trig && t->wpad->exp.type == EXP_CLASSIC))) ||
(t->pad.btns_h == trigger[i]->pad.btns_h && t->pad.btns_h > 0))
2009-04-10 07:51:55 +00:00
{
if(trigger[i]->type == TRIGGER_HELD)
held = true;
}
if(!held && state == STATE_HELD && stateChan == t->chan)
{
this->ResetState();
2009-04-10 07:51:55 +00:00
}
else if(held && state == STATE_CLICKED && stateChan == t->chan)
2009-04-10 07:51:55 +00:00
{
this->SetState(STATE_HELD, t->chan);
2009-04-10 07:51:55 +00:00
}
}
}
}
2009-04-07 02:57:49 +00:00
if(updateCB)
updateCB(this);
}