mirror of
https://github.com/minetest/irrlicht.git
synced 2025-01-07 13:57:33 +01:00
1032 lines
24 KiB
C++
1032 lines
24 KiB
C++
// Copyright (C) 2002-2012 Nikolaus Gebhardt
|
|
// This file is part of the "Irrlicht Engine".
|
|
// For conditions of distribution and use, see copyright notice in irrlicht.h
|
|
|
|
#include "CGUITabControl.h"
|
|
|
|
#include "CGUIButton.h"
|
|
#include "IGUISkin.h"
|
|
#include "IGUIEnvironment.h"
|
|
#include "IGUIFont.h"
|
|
#include "IVideoDriver.h"
|
|
#include "rect.h"
|
|
#include "os.h"
|
|
|
|
namespace irr
|
|
{
|
|
namespace gui
|
|
{
|
|
|
|
// ------------------------------------------------------------------
|
|
// Tab
|
|
// ------------------------------------------------------------------
|
|
|
|
//! constructor
|
|
CGUITab::CGUITab(IGUIEnvironment* environment,
|
|
IGUIElement* parent, const core::rect<s32>& rectangle,
|
|
s32 id)
|
|
: IGUITab(environment, parent, id, rectangle),
|
|
BackColor(0,0,0,0), OverrideTextColorEnabled(false), TextColor(255,0,0,0),
|
|
DrawBackground(false)
|
|
{
|
|
#ifdef _DEBUG
|
|
setDebugName("CGUITab");
|
|
#endif
|
|
|
|
const IGUISkin* const skin = environment->getSkin();
|
|
if (skin)
|
|
TextColor = skin->getColor(EGDC_BUTTON_TEXT);
|
|
}
|
|
|
|
//! draws the element and its children
|
|
void CGUITab::draw()
|
|
{
|
|
if (!IsVisible)
|
|
return;
|
|
|
|
IGUISkin *skin = Environment->getSkin();
|
|
|
|
if (skin && DrawBackground)
|
|
skin->draw2DRectangle(this, BackColor, AbsoluteRect, &AbsoluteClippingRect);
|
|
|
|
IGUIElement::draw();
|
|
}
|
|
|
|
|
|
//! sets if the tab should draw its background
|
|
void CGUITab::setDrawBackground(bool draw)
|
|
{
|
|
DrawBackground = draw;
|
|
}
|
|
|
|
|
|
//! sets the color of the background, if it should be drawn.
|
|
void CGUITab::setBackgroundColor(video::SColor c)
|
|
{
|
|
BackColor = c;
|
|
}
|
|
|
|
|
|
//! sets the color of the text
|
|
void CGUITab::setTextColor(video::SColor c)
|
|
{
|
|
OverrideTextColorEnabled = true;
|
|
TextColor = c;
|
|
}
|
|
|
|
|
|
video::SColor CGUITab::getTextColor() const
|
|
{
|
|
if ( OverrideTextColorEnabled )
|
|
return TextColor;
|
|
else
|
|
return Environment->getSkin()->getColor(EGDC_BUTTON_TEXT);
|
|
}
|
|
|
|
//! returns true if the tab is drawing its background, false if not
|
|
bool CGUITab::isDrawingBackground() const
|
|
{
|
|
return DrawBackground;
|
|
}
|
|
|
|
|
|
//! returns the color of the background
|
|
video::SColor CGUITab::getBackgroundColor() const
|
|
{
|
|
return BackColor;
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
// Tabcontrol
|
|
// ------------------------------------------------------------------
|
|
|
|
//! constructor
|
|
CGUITabControl::CGUITabControl(IGUIEnvironment* environment,
|
|
IGUIElement* parent, const core::rect<s32>& rectangle,
|
|
bool fillbackground, bool border, s32 id)
|
|
: IGUITabControl(environment, parent, id, rectangle), ActiveTabIndex(-1),
|
|
Border(border), FillBackground(fillbackground), ScrollControl(false), TabHeight(0), VerticalAlignment(EGUIA_UPPERLEFT),
|
|
UpButton(0), DownButton(0), TabMaxWidth(0), CurrentScrollTabIndex(0), TabExtraWidth(20)
|
|
{
|
|
#ifdef _DEBUG
|
|
setDebugName("CGUITabControl");
|
|
#endif
|
|
|
|
IGUISkin* skin = Environment->getSkin();
|
|
IGUISpriteBank* sprites = 0;
|
|
|
|
TabHeight = 32;
|
|
|
|
if (skin)
|
|
{
|
|
sprites = skin->getSpriteBank();
|
|
TabHeight = skin->getSize(gui::EGDS_BUTTON_HEIGHT) + 2;
|
|
}
|
|
|
|
UpButton = Environment->addButton(core::rect<s32>(0,0,10,10), this);
|
|
|
|
if (UpButton)
|
|
{
|
|
UpButton->setSpriteBank(sprites);
|
|
UpButton->setVisible(false);
|
|
UpButton->setSubElement(true);
|
|
UpButton->setAlignment(EGUIA_LOWERRIGHT, EGUIA_LOWERRIGHT, EGUIA_UPPERLEFT, EGUIA_UPPERLEFT);
|
|
UpButton->setOverrideFont(Environment->getBuiltInFont());
|
|
UpButton->grab();
|
|
}
|
|
|
|
DownButton = Environment->addButton(core::rect<s32>(0,0,10,10), this);
|
|
|
|
if (DownButton)
|
|
{
|
|
DownButton->setSpriteBank(sprites);
|
|
DownButton->setVisible(false);
|
|
DownButton->setSubElement(true);
|
|
DownButton->setAlignment(EGUIA_LOWERRIGHT, EGUIA_LOWERRIGHT, EGUIA_UPPERLEFT, EGUIA_UPPERLEFT);
|
|
DownButton->setOverrideFont(Environment->getBuiltInFont());
|
|
DownButton->grab();
|
|
}
|
|
|
|
setTabVerticalAlignment(EGUIA_UPPERLEFT);
|
|
refreshSprites();
|
|
}
|
|
|
|
//! destructor
|
|
CGUITabControl::~CGUITabControl()
|
|
{
|
|
for (u32 i=0; i<Tabs.size(); ++i)
|
|
{
|
|
if (Tabs[i])
|
|
Tabs[i]->drop();
|
|
}
|
|
|
|
if (UpButton)
|
|
UpButton->drop();
|
|
|
|
if (DownButton)
|
|
DownButton->drop();
|
|
}
|
|
|
|
void CGUITabControl::refreshSprites()
|
|
{
|
|
video::SColor color(255,255,255,255);
|
|
IGUISkin* skin = Environment->getSkin();
|
|
if (skin)
|
|
{
|
|
color = skin->getColor(isEnabled() ? EGDC_WINDOW_SYMBOL : EGDC_GRAY_WINDOW_SYMBOL);
|
|
|
|
if (UpButton)
|
|
{
|
|
UpButton->setSprite(EGBS_BUTTON_UP, skin->getIcon(EGDI_CURSOR_LEFT), color);
|
|
UpButton->setSprite(EGBS_BUTTON_DOWN, skin->getIcon(EGDI_CURSOR_LEFT), color);
|
|
}
|
|
|
|
if (DownButton)
|
|
{
|
|
DownButton->setSprite(EGBS_BUTTON_UP, skin->getIcon(EGDI_CURSOR_RIGHT), color);
|
|
DownButton->setSprite(EGBS_BUTTON_DOWN, skin->getIcon(EGDI_CURSOR_RIGHT), color);
|
|
}
|
|
}
|
|
}
|
|
|
|
//! Adds a tab
|
|
IGUITab* CGUITabControl::addTab(const wchar_t* caption, s32 id)
|
|
{
|
|
CGUITab* tab = new CGUITab(Environment, this, calcTabPos(), id);
|
|
|
|
tab->setText(caption);
|
|
tab->setAlignment(EGUIA_UPPERLEFT, EGUIA_LOWERRIGHT, EGUIA_UPPERLEFT, EGUIA_LOWERRIGHT);
|
|
tab->setVisible(false);
|
|
Tabs.push_back(tab); // no grab as new already creates a reference
|
|
|
|
if (ActiveTabIndex == -1)
|
|
{
|
|
ActiveTabIndex = Tabs.size()-1;
|
|
tab->setVisible(true);
|
|
}
|
|
|
|
recalculateScrollBar();
|
|
|
|
return tab;
|
|
}
|
|
|
|
|
|
//! adds a tab which has been created elsewhere
|
|
s32 CGUITabControl::addTab(IGUITab* tab)
|
|
{
|
|
return insertTab( Tabs.size(), tab, false);
|
|
}
|
|
|
|
//! Insert the tab at the given index
|
|
IGUITab* CGUITabControl::insertTab(s32 idx, const wchar_t* caption, s32 id)
|
|
{
|
|
if ( idx < 0 || idx > (s32)Tabs.size() ) // idx == Tabs.size() is indeed OK here as core::array can handle that
|
|
return NULL;
|
|
|
|
CGUITab* tab = new CGUITab(Environment, this, calcTabPos(), id);
|
|
|
|
tab->setText(caption);
|
|
tab->setAlignment(EGUIA_UPPERLEFT, EGUIA_LOWERRIGHT, EGUIA_UPPERLEFT, EGUIA_LOWERRIGHT);
|
|
tab->setVisible(false);
|
|
Tabs.insert(tab, (u32)idx);
|
|
|
|
if (ActiveTabIndex == -1)
|
|
{
|
|
ActiveTabIndex = (u32)idx;
|
|
tab->setVisible(true);
|
|
}
|
|
else if ( idx <= ActiveTabIndex )
|
|
{
|
|
++ActiveTabIndex;
|
|
setVisibleTab(ActiveTabIndex);
|
|
}
|
|
|
|
recalculateScrollBar();
|
|
|
|
return tab;
|
|
}
|
|
|
|
s32 CGUITabControl::insertTab(s32 idx, IGUITab* tab, bool serializationMode)
|
|
{
|
|
if (!tab)
|
|
return -1;
|
|
if ( idx > (s32)Tabs.size() && !serializationMode ) // idx == Tabs.size() is indeed OK here as core::array can handle that
|
|
return -1;
|
|
// Not allowing to add same tab twice as it would make things complicated (serialization or setting active visible)
|
|
if ( getTabIndex(tab) >= 0 )
|
|
return -1;
|
|
|
|
if ( idx < 0 )
|
|
idx = (s32)Tabs.size();
|
|
|
|
if ( tab->getParent() != this )
|
|
this->addChildToEnd(tab);
|
|
|
|
tab->setVisible(false);
|
|
|
|
tab->grab();
|
|
|
|
if ( serializationMode)
|
|
{
|
|
while ( idx >= (s32)Tabs.size() )
|
|
{
|
|
Tabs.push_back(0);
|
|
}
|
|
Tabs[idx] = tab;
|
|
|
|
if ( idx == ActiveTabIndex) // in serialization that can happen for any index
|
|
{
|
|
setVisibleTab(ActiveTabIndex);
|
|
tab->setVisible(true);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Tabs.insert(tab, (u32)idx);
|
|
|
|
if (ActiveTabIndex == -1)
|
|
{
|
|
ActiveTabIndex = idx;
|
|
setVisibleTab(ActiveTabIndex);
|
|
}
|
|
else if ( idx <= ActiveTabIndex)
|
|
{
|
|
++ActiveTabIndex;
|
|
setVisibleTab(ActiveTabIndex);
|
|
}
|
|
}
|
|
|
|
recalculateScrollBar();
|
|
|
|
return idx;
|
|
}
|
|
|
|
//! Removes a child.
|
|
void CGUITabControl::removeChild(IGUIElement* child)
|
|
{
|
|
s32 idx = getTabIndex(child);
|
|
if ( idx >= 0 )
|
|
removeTabButNotChild(idx);
|
|
|
|
// remove real element
|
|
IGUIElement::removeChild(child);
|
|
|
|
recalculateScrollBar();
|
|
}
|
|
|
|
|
|
//! Removes a tab from the tabcontrol
|
|
void CGUITabControl::removeTab(s32 idx)
|
|
{
|
|
if ( idx < 0 || idx >= (s32)Tabs.size() )
|
|
return;
|
|
|
|
removeChild(Tabs[(u32)idx]);
|
|
}
|
|
|
|
void CGUITabControl::removeTabButNotChild(s32 idx)
|
|
{
|
|
if ( idx < 0 || idx >= (s32)Tabs.size() )
|
|
return;
|
|
|
|
Tabs[(u32)idx]->drop();
|
|
Tabs.erase((u32)idx);
|
|
|
|
if ( idx < ActiveTabIndex )
|
|
{
|
|
--ActiveTabIndex;
|
|
setVisibleTab(ActiveTabIndex);
|
|
}
|
|
else if ( idx == ActiveTabIndex )
|
|
{
|
|
if ( (u32)idx == Tabs.size() )
|
|
--ActiveTabIndex;
|
|
setVisibleTab(ActiveTabIndex);
|
|
}
|
|
}
|
|
|
|
//! Clears the tabcontrol removing all tabs
|
|
void CGUITabControl::clear()
|
|
{
|
|
for (u32 i=0; i<Tabs.size(); ++i)
|
|
{
|
|
if (Tabs[i])
|
|
{
|
|
IGUIElement::removeChild(Tabs[i]);
|
|
Tabs[i]->drop();
|
|
}
|
|
}
|
|
Tabs.clear();
|
|
|
|
recalculateScrollBar();
|
|
}
|
|
|
|
//! Returns amount of tabs in the tabcontrol
|
|
s32 CGUITabControl::getTabCount() const
|
|
{
|
|
return Tabs.size();
|
|
}
|
|
|
|
|
|
//! Returns a tab based on zero based index
|
|
IGUITab* CGUITabControl::getTab(s32 idx) const
|
|
{
|
|
if (idx < 0 || (u32)idx >= Tabs.size())
|
|
return 0;
|
|
|
|
return Tabs[idx];
|
|
}
|
|
|
|
|
|
//! called if an event happened.
|
|
bool CGUITabControl::OnEvent(const SEvent& event)
|
|
{
|
|
if (isEnabled())
|
|
{
|
|
switch(event.EventType)
|
|
{
|
|
case EET_GUI_EVENT:
|
|
switch(event.GUIEvent.EventType)
|
|
{
|
|
case EGET_BUTTON_CLICKED:
|
|
if (event.GUIEvent.Caller == UpButton)
|
|
{
|
|
scrollLeft();
|
|
return true;
|
|
}
|
|
else if (event.GUIEvent.Caller == DownButton)
|
|
{
|
|
scrollRight();
|
|
return true;
|
|
}
|
|
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
case EET_MOUSE_INPUT_EVENT:
|
|
switch(event.MouseInput.Event)
|
|
{
|
|
//case EMIE_LMOUSE_PRESSED_DOWN:
|
|
// // todo: dragging tabs around
|
|
// return true;
|
|
case EMIE_LMOUSE_LEFT_UP:
|
|
{
|
|
s32 idx = getTabAt(event.MouseInput.X, event.MouseInput.Y);
|
|
if ( idx >= 0 )
|
|
{
|
|
setActiveTab(idx);
|
|
return true;
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return IGUIElement::OnEvent(event);
|
|
}
|
|
|
|
|
|
void CGUITabControl::scrollLeft()
|
|
{
|
|
if ( CurrentScrollTabIndex > 0 )
|
|
--CurrentScrollTabIndex;
|
|
recalculateScrollBar();
|
|
}
|
|
|
|
|
|
void CGUITabControl::scrollRight()
|
|
{
|
|
if ( CurrentScrollTabIndex < (s32)(Tabs.size()) - 1 )
|
|
{
|
|
if ( needScrollControl(CurrentScrollTabIndex, true) )
|
|
++CurrentScrollTabIndex;
|
|
}
|
|
recalculateScrollBar();
|
|
}
|
|
|
|
s32 CGUITabControl::calcTabWidth(IGUIFont* font, const wchar_t* text) const
|
|
{
|
|
if ( !font )
|
|
return 0;
|
|
|
|
s32 len = font->getDimension(text).Width + TabExtraWidth;
|
|
if ( TabMaxWidth > 0 && len > TabMaxWidth )
|
|
len = TabMaxWidth;
|
|
|
|
return len;
|
|
}
|
|
|
|
bool CGUITabControl::needScrollControl(s32 startIndex, bool withScrollControl, s32 *pos_rightmost)
|
|
{
|
|
if ( startIndex < 0 )
|
|
startIndex = 0;
|
|
|
|
IGUISkin* skin = Environment->getSkin();
|
|
if (!skin)
|
|
return false;
|
|
|
|
IGUIFont* font = skin->getFont();
|
|
|
|
if (Tabs.empty())
|
|
return false;
|
|
|
|
if (!font)
|
|
return false;
|
|
|
|
s32 pos = AbsoluteRect.UpperLeftCorner.X + 2;
|
|
const s32 pos_right = withScrollControl ?
|
|
UpButton->getAbsolutePosition().UpperLeftCorner.X - 2 :
|
|
AbsoluteRect.LowerRightCorner.X;
|
|
|
|
for (s32 i = startIndex; i < (s32)Tabs.size(); ++i)
|
|
{
|
|
// get Text
|
|
const wchar_t* text = 0;
|
|
if (Tabs[i])
|
|
{
|
|
text = Tabs[i]->getText();
|
|
|
|
// get text length
|
|
s32 len = calcTabWidth(font, text); // always without withScrollControl here or len would be shortened
|
|
pos += len;
|
|
}
|
|
|
|
if (pos > pos_right)
|
|
return true;
|
|
}
|
|
|
|
if (pos_rightmost)
|
|
*pos_rightmost = pos;
|
|
return false;
|
|
}
|
|
|
|
|
|
s32 CGUITabControl::calculateScrollIndexFromActive()
|
|
{
|
|
if (!ScrollControl || Tabs.empty())
|
|
return 0;
|
|
|
|
IGUISkin *skin = Environment->getSkin();
|
|
if (!skin)
|
|
return false;
|
|
|
|
IGUIFont *font = skin->getFont();
|
|
if (!font)
|
|
return false;
|
|
|
|
const s32 pos_left = AbsoluteRect.UpperLeftCorner.X + 2;
|
|
const s32 pos_right = UpButton->getAbsolutePosition().UpperLeftCorner.X - 2;
|
|
|
|
// Move from center to the left border left until it is reached
|
|
s32 pos_cl = (pos_left + pos_right) / 2;
|
|
s32 i = ActiveTabIndex;
|
|
for (; i > 0; --i) {
|
|
if (!Tabs[i])
|
|
continue;
|
|
|
|
s32 len = calcTabWidth(font, Tabs[i]->getText());
|
|
if (i == ActiveTabIndex)
|
|
len /= 2;
|
|
if (pos_cl - len < pos_left)
|
|
break;
|
|
|
|
pos_cl -= len;
|
|
}
|
|
if (i == 0)
|
|
return i;
|
|
|
|
// Is scrolling to right still possible?
|
|
s32 pos_rr = 0;
|
|
if (needScrollControl(i, true, &pos_rr))
|
|
return i; // Yes? -> OK
|
|
|
|
// No? -> Decrease "i" more. Append tabs until scrolling becomes necessary
|
|
for (--i; i > 0; --i) {
|
|
if (!Tabs[i])
|
|
continue;
|
|
|
|
pos_rr += calcTabWidth(font, Tabs[i]->getText());
|
|
if (pos_rr > pos_right)
|
|
break;
|
|
}
|
|
return i + 1;
|
|
}
|
|
|
|
core::rect<s32> CGUITabControl::calcTabPos()
|
|
{
|
|
core::rect<s32> r;
|
|
r.UpperLeftCorner.X = 0;
|
|
r.LowerRightCorner.X = AbsoluteRect.getWidth();
|
|
if ( Border )
|
|
{
|
|
++r.UpperLeftCorner.X;
|
|
--r.LowerRightCorner.X;
|
|
}
|
|
|
|
if ( VerticalAlignment == EGUIA_UPPERLEFT )
|
|
{
|
|
r.UpperLeftCorner.Y = TabHeight+2;
|
|
r.LowerRightCorner.Y = AbsoluteRect.getHeight()-1;
|
|
if ( Border )
|
|
{
|
|
--r.LowerRightCorner.Y;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
r.UpperLeftCorner.Y = 0;
|
|
r.LowerRightCorner.Y = AbsoluteRect.getHeight()-(TabHeight+2);
|
|
if ( Border )
|
|
{
|
|
++r.UpperLeftCorner.Y;
|
|
}
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
|
|
//! draws the element and its children
|
|
void CGUITabControl::draw()
|
|
{
|
|
if (!IsVisible)
|
|
return;
|
|
|
|
IGUISkin* skin = Environment->getSkin();
|
|
if (!skin)
|
|
return;
|
|
|
|
IGUIFont* font = skin->getFont();
|
|
video::IVideoDriver* driver = Environment->getVideoDriver();
|
|
|
|
core::rect<s32> frameRect(AbsoluteRect);
|
|
|
|
// some empty background as placeholder when there are no tabs
|
|
if (Tabs.empty())
|
|
driver->draw2DRectangle(skin->getColor(EGDC_3D_HIGH_LIGHT), frameRect, &AbsoluteClippingRect);
|
|
|
|
if (!font)
|
|
return;
|
|
|
|
// tab button bar can be above or below the tabs
|
|
if ( VerticalAlignment == EGUIA_UPPERLEFT )
|
|
{
|
|
frameRect.UpperLeftCorner.Y += 2;
|
|
frameRect.LowerRightCorner.Y = frameRect.UpperLeftCorner.Y + TabHeight;
|
|
}
|
|
else
|
|
{
|
|
frameRect.UpperLeftCorner.Y = frameRect.LowerRightCorner.Y - TabHeight - 1;
|
|
frameRect.LowerRightCorner.Y -= 2;
|
|
}
|
|
|
|
core::rect<s32> tr;
|
|
s32 pos = frameRect.UpperLeftCorner.X + 2;
|
|
|
|
bool needLeftScroll = CurrentScrollTabIndex > 0;
|
|
bool needRightScroll = false;
|
|
|
|
// left and right pos of the active tab
|
|
s32 left = 0;
|
|
s32 right = 0;
|
|
|
|
//const wchar_t* activetext = 0;
|
|
IGUITab *activeTab = 0;
|
|
|
|
// Draw all tab-buttons except the active one
|
|
for (u32 i = CurrentScrollTabIndex; i < Tabs.size() && !needRightScroll; ++i)
|
|
{
|
|
// get Text
|
|
const wchar_t* text = 0;
|
|
if (Tabs[i])
|
|
text = Tabs[i]->getText();
|
|
|
|
// get text length
|
|
s32 len = calcTabWidth(font, text);
|
|
if (ScrollControl) {
|
|
s32 space = UpButton->getAbsolutePosition().UpperLeftCorner.X - 2 - pos;
|
|
if (space < len) {
|
|
needRightScroll = true;
|
|
len = space;
|
|
}
|
|
}
|
|
|
|
frameRect.LowerRightCorner.X += len;
|
|
frameRect.UpperLeftCorner.X = pos;
|
|
frameRect.LowerRightCorner.X = frameRect.UpperLeftCorner.X + len;
|
|
|
|
pos += len;
|
|
|
|
if ((s32)i == ActiveTabIndex)
|
|
{
|
|
// for active button just remember values
|
|
left = frameRect.UpperLeftCorner.X;
|
|
right = frameRect.LowerRightCorner.X;
|
|
//activetext = text;
|
|
activeTab = Tabs[i];
|
|
}
|
|
else
|
|
{
|
|
skin->draw3DTabButton(this, false, frameRect, &AbsoluteClippingRect, VerticalAlignment);
|
|
|
|
// draw text
|
|
core::rect<s32> textClipRect(frameRect); // TODO: exact size depends on borders in draw3DTabButton which we don't get with current interface
|
|
textClipRect.clipAgainst(AbsoluteClippingRect);
|
|
font->draw(text, frameRect, Tabs[i]->getTextColor(),
|
|
true, true, &textClipRect);
|
|
}
|
|
}
|
|
|
|
// Draw active tab button
|
|
// Drawn later than other buttons because it draw over the buttons before/after it.
|
|
if (left != 0 && right != 0 && activeTab != 0)
|
|
{
|
|
// draw upper highlight frame
|
|
if ( VerticalAlignment == EGUIA_UPPERLEFT )
|
|
{
|
|
frameRect.UpperLeftCorner.X = left-2;
|
|
frameRect.LowerRightCorner.X = right+2;
|
|
frameRect.UpperLeftCorner.Y -= 2;
|
|
|
|
skin->draw3DTabButton(this, true, frameRect, &AbsoluteClippingRect, VerticalAlignment);
|
|
|
|
// draw text
|
|
core::rect<s32> textClipRect(frameRect); // TODO: exact size depends on borders in draw3DTabButton which we don't get with current interface
|
|
textClipRect.clipAgainst(AbsoluteClippingRect);
|
|
font->draw(activeTab->getText(), frameRect, activeTab->getTextColor(),
|
|
true, true, &textClipRect);
|
|
|
|
tr.UpperLeftCorner.X = AbsoluteRect.UpperLeftCorner.X;
|
|
tr.LowerRightCorner.X = left - 1;
|
|
tr.UpperLeftCorner.Y = frameRect.LowerRightCorner.Y - 1;
|
|
tr.LowerRightCorner.Y = frameRect.LowerRightCorner.Y;
|
|
driver->draw2DRectangle(skin->getColor(EGDC_3D_HIGH_LIGHT), tr, &AbsoluteClippingRect);
|
|
|
|
tr.UpperLeftCorner.X = right;
|
|
tr.LowerRightCorner.X = AbsoluteRect.LowerRightCorner.X;
|
|
driver->draw2DRectangle(skin->getColor(EGDC_3D_HIGH_LIGHT), tr, &AbsoluteClippingRect);
|
|
}
|
|
else
|
|
{
|
|
frameRect.UpperLeftCorner.X = left-2;
|
|
frameRect.LowerRightCorner.X = right+2;
|
|
frameRect.LowerRightCorner.Y += 2;
|
|
|
|
skin->draw3DTabButton(this, true, frameRect, &AbsoluteClippingRect, VerticalAlignment);
|
|
|
|
// draw text
|
|
font->draw(activeTab->getText(), frameRect, activeTab->getTextColor(),
|
|
true, true, &frameRect);
|
|
|
|
tr.UpperLeftCorner.X = AbsoluteRect.UpperLeftCorner.X;
|
|
tr.LowerRightCorner.X = left - 1;
|
|
tr.UpperLeftCorner.Y = frameRect.UpperLeftCorner.Y - 1;
|
|
tr.LowerRightCorner.Y = frameRect.UpperLeftCorner.Y;
|
|
driver->draw2DRectangle(skin->getColor(EGDC_3D_DARK_SHADOW), tr, &AbsoluteClippingRect);
|
|
|
|
tr.UpperLeftCorner.X = right;
|
|
tr.LowerRightCorner.X = AbsoluteRect.LowerRightCorner.X;
|
|
driver->draw2DRectangle(skin->getColor(EGDC_3D_DARK_SHADOW), tr, &AbsoluteClippingRect);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// No active tab
|
|
// Draw a line separating button bar from tab area
|
|
tr.UpperLeftCorner.X = AbsoluteRect.UpperLeftCorner.X;
|
|
tr.LowerRightCorner.X = AbsoluteRect.LowerRightCorner.X;
|
|
tr.UpperLeftCorner.Y = frameRect.LowerRightCorner.Y - 1;
|
|
tr.LowerRightCorner.Y = frameRect.LowerRightCorner.Y;
|
|
|
|
if ( VerticalAlignment == EGUIA_UPPERLEFT )
|
|
{
|
|
driver->draw2DRectangle(skin->getColor(EGDC_3D_HIGH_LIGHT), tr, &AbsoluteClippingRect);
|
|
}
|
|
else
|
|
{
|
|
tr.UpperLeftCorner.Y = frameRect.UpperLeftCorner.Y - 1;
|
|
tr.LowerRightCorner.Y = frameRect.UpperLeftCorner.Y;
|
|
driver->draw2DRectangle(skin->getColor(EGDC_3D_DARK_SHADOW), tr, &AbsoluteClippingRect);
|
|
}
|
|
}
|
|
|
|
// drawing some border and background for the tab-area.
|
|
skin->draw3DTabBody(this, Border, FillBackground, AbsoluteRect, &AbsoluteClippingRect, TabHeight, VerticalAlignment);
|
|
|
|
// enable scrollcontrols on need
|
|
if ( UpButton )
|
|
UpButton->setEnabled(needLeftScroll);
|
|
if ( DownButton )
|
|
DownButton->setEnabled(needRightScroll);
|
|
refreshSprites();
|
|
|
|
IGUIElement::draw();
|
|
}
|
|
|
|
|
|
//! Set the height of the tabs
|
|
void CGUITabControl::setTabHeight( s32 height )
|
|
{
|
|
if ( height < 0 )
|
|
height = 0;
|
|
|
|
TabHeight = height;
|
|
|
|
recalculateScrollButtonPlacement();
|
|
recalculateScrollBar();
|
|
}
|
|
|
|
|
|
//! Get the height of the tabs
|
|
s32 CGUITabControl::getTabHeight() const
|
|
{
|
|
return TabHeight;
|
|
}
|
|
|
|
//! set the maximal width of a tab. Per default width is 0 which means "no width restriction".
|
|
void CGUITabControl::setTabMaxWidth(s32 width )
|
|
{
|
|
TabMaxWidth = width;
|
|
}
|
|
|
|
//! get the maximal width of a tab
|
|
s32 CGUITabControl::getTabMaxWidth() const
|
|
{
|
|
return TabMaxWidth;
|
|
}
|
|
|
|
|
|
//! Set the extra width added to tabs on each side of the text
|
|
void CGUITabControl::setTabExtraWidth( s32 extraWidth )
|
|
{
|
|
if ( extraWidth < 0 )
|
|
extraWidth = 0;
|
|
|
|
TabExtraWidth = extraWidth;
|
|
|
|
recalculateScrollBar();
|
|
}
|
|
|
|
|
|
//! Get the extra width added to tabs on each side of the text
|
|
s32 CGUITabControl::getTabExtraWidth() const
|
|
{
|
|
return TabExtraWidth;
|
|
}
|
|
|
|
|
|
void CGUITabControl::recalculateScrollBar()
|
|
{
|
|
// Down: to right, Up: to left
|
|
if (!UpButton || !DownButton)
|
|
return;
|
|
|
|
ScrollControl = needScrollControl() || CurrentScrollTabIndex > 0;
|
|
|
|
if (ScrollControl)
|
|
{
|
|
UpButton->setVisible( true );
|
|
DownButton->setVisible( true );
|
|
}
|
|
else
|
|
{
|
|
UpButton->setVisible( false );
|
|
DownButton->setVisible( false );
|
|
}
|
|
|
|
bringToFront( UpButton );
|
|
bringToFront( DownButton );
|
|
}
|
|
|
|
//! Set the alignment of the tabs
|
|
void CGUITabControl::setTabVerticalAlignment( EGUI_ALIGNMENT alignment )
|
|
{
|
|
VerticalAlignment = alignment;
|
|
|
|
recalculateScrollButtonPlacement();
|
|
recalculateScrollBar();
|
|
|
|
core::rect<s32> r(calcTabPos());
|
|
for ( u32 i=0; i<Tabs.size(); ++i )
|
|
{
|
|
Tabs[i]->setRelativePosition(r);
|
|
}
|
|
}
|
|
|
|
void CGUITabControl::recalculateScrollButtonPlacement()
|
|
{
|
|
IGUISkin* skin = Environment->getSkin();
|
|
s32 ButtonSize = 16;
|
|
s32 ButtonHeight = TabHeight - 2;
|
|
if ( ButtonHeight < 0 )
|
|
ButtonHeight = TabHeight;
|
|
if (skin)
|
|
{
|
|
ButtonSize = skin->getSize(EGDS_WINDOW_BUTTON_WIDTH);
|
|
if (ButtonSize > TabHeight)
|
|
ButtonSize = TabHeight;
|
|
}
|
|
|
|
s32 ButtonX = RelativeRect.getWidth() - (s32)(2.5f*(f32)ButtonSize) - 1;
|
|
s32 ButtonY = 0;
|
|
|
|
if (VerticalAlignment == EGUIA_UPPERLEFT)
|
|
{
|
|
ButtonY = 2 + (TabHeight / 2) - (ButtonHeight / 2);
|
|
UpButton->setAlignment(EGUIA_LOWERRIGHT, EGUIA_LOWERRIGHT, EGUIA_UPPERLEFT, EGUIA_UPPERLEFT);
|
|
DownButton->setAlignment(EGUIA_LOWERRIGHT, EGUIA_LOWERRIGHT, EGUIA_UPPERLEFT, EGUIA_UPPERLEFT);
|
|
}
|
|
else
|
|
{
|
|
ButtonY = RelativeRect.getHeight() - (TabHeight / 2) - (ButtonHeight / 2) - 2;
|
|
UpButton->setAlignment(EGUIA_LOWERRIGHT, EGUIA_LOWERRIGHT, EGUIA_LOWERRIGHT, EGUIA_LOWERRIGHT);
|
|
DownButton->setAlignment(EGUIA_LOWERRIGHT, EGUIA_LOWERRIGHT, EGUIA_LOWERRIGHT, EGUIA_LOWERRIGHT);
|
|
}
|
|
|
|
UpButton->setRelativePosition(core::rect<s32>(ButtonX, ButtonY, ButtonX+ButtonSize, ButtonY+ButtonHeight));
|
|
ButtonX += ButtonSize + 1;
|
|
DownButton->setRelativePosition(core::rect<s32>(ButtonX, ButtonY, ButtonX+ButtonSize, ButtonY+ButtonHeight));
|
|
}
|
|
|
|
//! Get the alignment of the tabs
|
|
EGUI_ALIGNMENT CGUITabControl::getTabVerticalAlignment() const
|
|
{
|
|
return VerticalAlignment;
|
|
}
|
|
|
|
|
|
s32 CGUITabControl::getTabAt(s32 xpos, s32 ypos) const
|
|
{
|
|
core::position2di p(xpos, ypos);
|
|
IGUISkin* skin = Environment->getSkin();
|
|
IGUIFont* font = skin->getFont();
|
|
|
|
core::rect<s32> frameRect(AbsoluteRect);
|
|
|
|
if ( VerticalAlignment == EGUIA_UPPERLEFT )
|
|
{
|
|
frameRect.UpperLeftCorner.Y += 2;
|
|
frameRect.LowerRightCorner.Y = frameRect.UpperLeftCorner.Y + TabHeight;
|
|
}
|
|
else
|
|
{
|
|
frameRect.UpperLeftCorner.Y = frameRect.LowerRightCorner.Y - TabHeight;
|
|
}
|
|
|
|
s32 pos = frameRect.UpperLeftCorner.X + 2;
|
|
|
|
if (!frameRect.isPointInside(p))
|
|
return -1;
|
|
|
|
bool abort = false;
|
|
for (s32 i = CurrentScrollTabIndex; i < (s32)Tabs.size() && !abort; ++i)
|
|
{
|
|
// get Text
|
|
const wchar_t* text = 0;
|
|
if (Tabs[i])
|
|
text = Tabs[i]->getText();
|
|
|
|
// get text length
|
|
s32 len = calcTabWidth(font, text);
|
|
if (ScrollControl) {
|
|
// TODO: merge this with draw() ?
|
|
s32 space = UpButton->getAbsolutePosition().UpperLeftCorner.X - 2 - pos;
|
|
if (space < len) {
|
|
abort = true;
|
|
len = space;
|
|
}
|
|
}
|
|
|
|
frameRect.UpperLeftCorner.X = pos;
|
|
frameRect.LowerRightCorner.X = frameRect.UpperLeftCorner.X + len;
|
|
|
|
pos += len;
|
|
|
|
if (frameRect.isPointInside(p))
|
|
{
|
|
return i;
|
|
}
|
|
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
//! Returns which tab is currently active
|
|
s32 CGUITabControl::getActiveTab() const
|
|
{
|
|
return ActiveTabIndex;
|
|
}
|
|
|
|
|
|
//! Brings a tab to front.
|
|
bool CGUITabControl::setActiveTab(s32 idx)
|
|
{
|
|
if ((u32)idx >= Tabs.size())
|
|
return false;
|
|
|
|
bool changed = (ActiveTabIndex != idx);
|
|
|
|
ActiveTabIndex = idx;
|
|
|
|
setVisibleTab(ActiveTabIndex);
|
|
|
|
if (changed && Parent)
|
|
{
|
|
SEvent event;
|
|
event.EventType = EET_GUI_EVENT;
|
|
event.GUIEvent.Caller = this;
|
|
event.GUIEvent.Element = 0;
|
|
event.GUIEvent.EventType = EGET_TAB_CHANGED;
|
|
Parent->OnEvent(event);
|
|
}
|
|
|
|
if (ScrollControl) {
|
|
CurrentScrollTabIndex = calculateScrollIndexFromActive();
|
|
recalculateScrollBar();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void CGUITabControl::setVisibleTab(s32 idx)
|
|
{
|
|
for (u32 i=0; i<Tabs.size(); ++i)
|
|
if (Tabs[i])
|
|
Tabs[i]->setVisible( (s32)i == idx );
|
|
}
|
|
|
|
|
|
bool CGUITabControl::setActiveTab(IGUITab *tab)
|
|
{
|
|
return setActiveTab(getTabIndex(tab));
|
|
}
|
|
|
|
s32 CGUITabControl::getTabIndex(const IGUIElement *tab) const
|
|
{
|
|
for (u32 i=0; i<Tabs.size(); ++i)
|
|
if (Tabs[i] == tab)
|
|
return (s32)i;
|
|
|
|
return -1;
|
|
}
|
|
|
|
//! Update the position of the element, decides scroll button status
|
|
void CGUITabControl::updateAbsolutePosition()
|
|
{
|
|
IGUIElement::updateAbsolutePosition();
|
|
recalculateScrollBar();
|
|
}
|
|
|
|
|
|
} // end namespace irr
|
|
} // end namespace gui
|