starting to implement better skinning. currently, only plain buttons are skinned (you can see one in the help screens if you wish to see). I made a theme as proof of concept, if people don't like it, it can easily be replaced ;)

git-svn-id: svn+ssh://svn.code.sf.net/p/supertuxkart/code/main/branches/irrlicht@3374 178a84e3-b1eb-0310-8ba1-8eac791a3b58
This commit is contained in:
auria 2009-04-11 16:07:41 +00:00
parent de53cd22c1
commit b8033334d6
15 changed files with 215 additions and 10 deletions

Binary file not shown.

After

Width:  |  Height:  |  Size: 75 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 57 KiB

BIN
data/gui/glassbutton.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 24 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 27 KiB

BIN
data/gui/glassspinner.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 25 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 28 KiB

BIN
data/gui/glasstab.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB

BIN
data/gui/glasstab_focus.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 12 KiB

View File

@ -38,4 +38,8 @@
<label align="center" text="* Current key bindings can be seen/changed in menu Options"/> <label align="center" text="* Current key bindings can be seen/changed in menu Options"/>
<spacer width="50" height="45" />
</div> </div>
<button id="back" x="20" y="-40" width="250" height="35" align="left" text="Back to main menu"/>

View File

@ -50,5 +50,7 @@
<label proportion="1" height="100%" align="left" word_wrap="true" text="Anchor - slows down greatly the kart in the first position"/> <label proportion="1" height="100%" align="left" word_wrap="true" text="Anchor - slows down greatly the kart in the first position"/>
</div> </div>
<spacer width="50" height="45" />
</div> </div>
<button id="back" x="20" y="-40" width="250" height="35" align="left" text="Back to main menu"/>

View File

@ -1,3 +1,4 @@
<div x="2%" y="2%" width="96%" height="96%" layout="vertical-row" > <div x="2%" y="2%" width="96%" height="96%" layout="vertical-row" >
<label align="center" text="SuperTuxKart Options"/> <label align="center" text="SuperTuxKart Options"/>
@ -42,7 +43,7 @@
<label proportion="1" width="100%" align="left" word_wrap="true" text="* Most of these game modes can also be played in a Grand Prix fashion : instead of playing a single race, you play many in a row. The better you rank, the more points you get. In the end, the player with the most points wins the cup."/> <label proportion="1" width="100%" align="left" word_wrap="true" text="* Most of these game modes can also be played in a Grand Prix fashion : instead of playing a single race, you play many in a row. The better you rank, the more points you get. In the end, the player with the most points wins the cup."/>
<spacer width="50" height="45" />
</div> </div>
<button id="back" x="20" y="-40" width="250" height="35" align="left" text="Back to main menu"/>

View File

@ -2,6 +2,7 @@
#include "gui/engine.hpp" #include "gui/engine.hpp"
#include "gui/screen.hpp" #include "gui/screen.hpp"
#include "gui/widget.hpp" #include "gui/widget.hpp"
#include "io/file_manager.hpp"
#include <cassert> #include <cassert>
#include <iostream> #include <iostream>
@ -12,6 +13,17 @@ Skin::Skin(IGUISkin* fallback_skin)
m_fallback_skin = fallback_skin; m_fallback_skin = fallback_skin;
m_fallback_skin->grab(); m_fallback_skin->grab();
assert(fallback_skin != NULL); assert(fallback_skin != NULL);
m_tex_button = GUIEngine::getDriver()->getTexture( (file_manager->getGUIDir() + "/glassbutton.png").c_str() );
m_tex_fbutton = GUIEngine::getDriver()->getTexture( (file_manager->getGUIDir() + "/glassbutton_focused.png").c_str() );
m_tex_spinner = GUIEngine::getDriver()->getTexture( (file_manager->getGUIDir() + "/glassspinner.png").c_str() );
m_tex_fspinner = GUIEngine::getDriver()->getTexture( (file_manager->getGUIDir() + "/glassspinner_focus.png").c_str() );
m_tex_tab = GUIEngine::getDriver()->getTexture( (file_manager->getGUIDir() + "/glasstab.png").c_str() );
m_tex_ftab = GUIEngine::getDriver()->getTexture( (file_manager->getGUIDir() + "/glasstab_focus.png").c_str() );
m_tex_iconhighlight = GUIEngine::getDriver()->getTexture( (file_manager->getGUIDir() + "/glass_iconhighlight.png").c_str() );
m_tex_ficonhighlight = GUIEngine::getDriver()->getTexture( (file_manager->getGUIDir() + "/glass_iconhighlight_focus.png").c_str() );
} }
Skin::~Skin() Skin::~Skin()
@ -33,12 +45,173 @@ void Skin::draw2DRectangle (IGUIElement *element, const video::SColor &color, co
GUIEngine::getDriver()->draw2DRectangle( SColor(255, 0, 150, 150), pos ); GUIEngine::getDriver()->draw2DRectangle( SColor(255, 0, 150, 150), pos );
} }
void Skin::drawBoxFromStretchableTexture(const core::rect< s32 > &dest, ITexture* source,
const int left_border, const int right_border,
const int top_border, const int bottom_border)
{
// FIXME? - lots of things here will be re-calculated every frame, which is useless since
// widgets won't move, so they'd only need to be calculated once.
const int texture_w = source->getSize().Width;
const int texture_h = source->getSize().Height;
/*
The source texture is split this way to allow for a stretchable center and borders that don't stretch :
+----+--------------------+----+
| | | |
+----a--------------------b----+ <-- top_border
| | | |
| | | |
| | | |
+----c--------------------d----+ <-- height - bottom-border
| | | |
+----+--------------------+----+
*/
const int ax = left_border;
const int ay = top_border;
const int bx = texture_w - right_border;
const int by = top_border;
const int cx = left_border;
const int cy = texture_h - bottom_border;
const int dx = texture_w - right_border;
const int dy = texture_h - bottom_border;
core::rect<s32> source_area_left = core::rect<s32>(0, ay, cx, cy);
core::rect<s32> source_area_center = core::rect<s32>(ax, ay, dx, dy);
core::rect<s32> source_area_right = core::rect<s32>(bx, top_border, texture_w, dy);
core::rect<s32> source_area_top = core::rect<s32>(ax, 0, bx, by);
core::rect<s32> source_area_bottom = core::rect<s32>(cx, cy, dx, texture_h);
core::rect<s32> source_area_top_left = core::rect<s32>(0, 0, ax, ay);
core::rect<s32> source_area_top_right = core::rect<s32>(bx, 0, texture_w, top_border);
core::rect<s32> source_area_bottom_left = core::rect<s32>(0, cy, cx, texture_h);
core::rect<s32> source_area_bottom_right = core::rect<s32>(dx, dy, texture_w, texture_h);
/*
The dest area is split this way. Borders will go a bit beyond the
given area so components inside don't go over the borders (by half their size)
a----b--------------------c----+
| | | |
d----e--------------------f----g <-- top_border
| | | |
| | | |
| | | |
h----i--------------------j----k <-- height - bottom-border
| | | |
+----l--------------------m----n
*/
{
const int dest_x = dest.UpperLeftCorner.X;
const int dest_y = dest.UpperLeftCorner.Y;
const int dest_x2 = dest.LowerRightCorner.X;
const int dest_y2 = dest.LowerRightCorner.Y;
const float scale = (float)(dest_y2-dest_y)/texture_h;
const int dest_left_border_half = left_border*scale*0.5f;
const int dest_right_border_half = right_border*scale*0.5f;
const int dest_top_border_half = top_border*scale*0.5f;
const int dest_bottom_border_half = bottom_border*scale*0.5f;
const int ax = dest_x - dest_left_border_half;
const int ay = dest_y - dest_top_border_half;
const int bx = dest_x + dest_left_border_half;
const int by = ay;
const int cx = dest_x2 - dest_right_border_half;
const int cy = ay;
const int dx = ax;
const int dy = dest_y + dest_top_border_half;
const int ex = bx;
const int ey = dy;
const int fx = cx;
const int fy = dy;
const int gx = dest_x2 + dest_right_border_half;
const int gy = dy;
const int hx = ax;
const int hy = dest_y2 - dest_bottom_border_half;
const int ix = bx;
const int iy = hy;
const int jx = cx;
const int jy = hy;
const int kx = gx;
const int ky = hy;
const int lx = bx;
const int ly = dest_y2 + dest_bottom_border_half;
const int mx = cx;
const int my = ly;
const int nx = gx;
const int ny = ly;
core::rect<s32> dest_area_left = core::rect<s32>(dx, dy, ix, iy);
core::rect<s32> dest_area_center = core::rect<s32>(ex, ey, jx, jy);
core::rect<s32> dest_area_right = core::rect<s32>(fx, fy, kx, ky);
core::rect<s32> dest_area_top = core::rect<s32>(bx, by, fx, fy);
core::rect<s32> dest_area_bottom = core::rect<s32>(ix, iy, mx, my);
core::rect<s32> dest_area_top_left = core::rect<s32>(ax, ay, ex, ey);
core::rect<s32> dest_area_top_right = core::rect<s32>(cx, cy, gx, gy);
core::rect<s32> dest_area_bottom_left = core::rect<s32>(hx, hy, lx, ly);
core::rect<s32> dest_area_bottom_right = core::rect<s32>(jx, jy, nx, ny);
GUIEngine::getDriver()->draw2DImage(source, dest_area_left, source_area_left,
0 /* no clipping */, 0, true /* alpha */);
GUIEngine::getDriver()->draw2DImage(source, dest_area_center, source_area_center,
0 /* no clipping */, 0, true /* alpha */);
GUIEngine::getDriver()->draw2DImage(source, dest_area_right, source_area_right,
0 /* no clipping */, 0, true /* alpha */);
GUIEngine::getDriver()->draw2DImage(source, dest_area_top, source_area_top,
0 /* no clipping */, 0, true /* alpha */);
GUIEngine::getDriver()->draw2DImage(source, dest_area_bottom, source_area_bottom,
0 /* no clipping */, 0, true /* alpha */);
GUIEngine::getDriver()->draw2DImage(source, dest_area_top_left, source_area_top_left,
0 /* no clipping */, 0, true /* alpha */);
GUIEngine::getDriver()->draw2DImage(source, dest_area_top_right, source_area_top_right,
0 /* no clipping */, 0, true /* alpha */);
GUIEngine::getDriver()->draw2DImage(source, dest_area_bottom_left, source_area_bottom_left,
0 /* no clipping */, 0, true /* alpha */);
GUIEngine::getDriver()->draw2DImage(source, dest_area_bottom_right, source_area_bottom_right,
0 /* no clipping */, 0, true /* alpha */);
}
}
void Skin::drawButton(const core::rect< s32 > &rect, const bool pressed, const bool focused) void Skin::drawButton(const core::rect< s32 > &rect, const bool pressed, const bool focused)
{ {
if(focused) // FIXME - move these numbers to a config file
GUIEngine::getDriver()->draw2DRectangle( SColor(255, 100, 0, 0), rect ); const int left_border = 80;
else const int right_border = 80;
GUIEngine::getDriver()->draw2DRectangle( SColor(255, 0, 100, 0), rect ); const int border_above = 0;
const int border_below = 36;
drawBoxFromStretchableTexture(rect, (focused ? m_tex_fbutton : m_tex_button),
left_border, right_border,
border_above, border_below);
/*
if(focused)
GUIEngine::getDriver()->draw2DRectangle( SColor(255, 100, 0, 0), rect );
else
GUIEngine::getDriver()->draw2DRectangle( SColor(255, 0, 100, 0), rect );
*/
} }
void Skin::drawRibbon(const core::rect< s32 > &rect, const Widget* widget, const bool pressed, bool focused) void Skin::drawRibbon(const core::rect< s32 > &rect, const Widget* widget, const bool pressed, bool focused)
{ {
@ -57,7 +230,7 @@ void Skin::drawRibbon(const core::rect< s32 > &rect, const Widget* widget, const
} }
if(!draw_border) return; if(!draw_border) return;
if(focused) if(focused)
GUIEngine::getDriver()->draw2DRectangle( SColor(255, 150, 0, 0), rect ); GUIEngine::getDriver()->draw2DRectangle( SColor(255, 150, 0, 0), rect );
else else
@ -124,7 +297,7 @@ void Skin::process3DPane(IGUIElement *element, const core::rect< s32 > &rect, co
const WidgetType type = widget->m_type; const WidgetType type = widget->m_type;
// buttons are used for other uses than plain clickable buttons because irrLicht // buttons are used for other uses than plain clickable buttons because irrLicht
// does not have widgets for everything we need. so at render time, we just check // does not have widgets for everything we need. so at render time, we just check
// which type this button represents and render accordingly // which type this button represents and render accordingly
@ -154,7 +327,7 @@ void Skin::process3DPane(IGUIElement *element, const core::rect< s32 > &rect, co
{ {
drawSpinnerBody(rect, widget, pressed, focused); drawSpinnerBody(rect, widget, pressed, focused);
} }
} }
void Skin::draw3DButtonPanePressed (IGUIElement *element, const core::rect< s32 > &rect, const core::rect< s32 > *clip) void Skin::draw3DButtonPanePressed (IGUIElement *element, const core::rect< s32 > &rect, const core::rect< s32 > *clip)

View File

@ -17,6 +17,20 @@ namespace GUIEngine
class Skin : public IGUISkin class Skin : public IGUISkin
{ {
IGUISkin* m_fallback_skin; IGUISkin* m_fallback_skin;
ITexture* m_tex_button;
ITexture* m_tex_fbutton;
ITexture* m_tex_spinner;
ITexture* m_tex_fspinner;
ITexture* m_tex_tab;
ITexture* m_tex_ftab;
ITexture* m_tex_iconhighlight;
ITexture* m_tex_ficonhighlight;
void drawBoxFromStretchableTexture(const core::rect< s32 > &dest, ITexture* source,
const int left_border, const int right_border,
const int top_border, const int bottom_border);
public: public:
Skin(IGUISkin* fallback_skin); Skin(IGUISkin* fallback_skin);
~Skin(); ~Skin();

View File

@ -312,6 +312,10 @@ namespace StateManager
else if(selection == "page2") replaceTopMostMenu("help2.stkgui"); else if(selection == "page2") replaceTopMostMenu("help2.stkgui");
else if(selection == "page3") replaceTopMostMenu("help3.stkgui"); else if(selection == "page3") replaceTopMostMenu("help3.stkgui");
} }
else if(name == "back")
{
StateManager::escapePressed();
}
} }

View File

@ -105,6 +105,7 @@ void Widget::readCoords(Widget* parent)
if(convertToCoord(x, &abs_x, &percent_x )) if(convertToCoord(x, &abs_x, &percent_x ))
{ {
if(abs_x > -1) this->x = parent_x + abs_x; if(abs_x > -1) this->x = parent_x + abs_x;
else if(abs_x < -1) this->x = parent_x + (parent_w + abs_x);
else if(percent_x > -1) this->x = parent_x + parent_w*percent_x/100; else if(percent_x > -1) this->x = parent_x + parent_w*percent_x/100;
} }
} }
@ -114,7 +115,13 @@ void Widget::readCoords(Widget* parent)
int abs_y = -1, percent_y = -1; int abs_y = -1, percent_y = -1;
if(convertToCoord(y, &abs_y, &percent_y )) if(convertToCoord(y, &abs_y, &percent_y ))
{ {
if(abs_y < -1)
{
std::cout << "abs_y = " << abs_y << ", y=" << (parent_y + (parent_h - abs_y)) << std::endl;
}
if(abs_y > -1) this->y = parent_y + abs_y; if(abs_y > -1) this->y = parent_y + abs_y;
else if(abs_y < -1) this->y = parent_y + (parent_h + abs_y);
else if(percent_y > -1) this->y = parent_y + parent_h*percent_y/100; else if(percent_y > -1) this->y = parent_y + parent_h*percent_y/100;
} }
} }