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
BIN
data/gui/glass_iconhighlight.png
Normal file
After Width: | Height: | Size: 75 KiB |
BIN
data/gui/glass_iconhighlight_focus.png
Normal file
After Width: | Height: | Size: 57 KiB |
BIN
data/gui/glassbutton.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
data/gui/glassbutton_focused.png
Normal file
After Width: | Height: | Size: 27 KiB |
BIN
data/gui/glassspinner.png
Normal file
After Width: | Height: | Size: 25 KiB |
BIN
data/gui/glassspinner_focus.png
Normal file
After Width: | Height: | Size: 28 KiB |
BIN
data/gui/glasstab.png
Normal file
After Width: | Height: | Size: 11 KiB |
BIN
data/gui/glasstab_focus.png
Normal file
After Width: | Height: | Size: 12 KiB |
@ -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"/>
|
||||||
|
|
||||||
|
@ -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"/>
|
||||||
|
@ -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"/>
|
||||||
|
187
src/gui/skin.cpp
@ -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)
|
||||||
|
@ -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();
|
||||||
|
@ -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();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|