2014-06-25 16:29:58 +00:00
|
|
|
#include "components/RatingComponent.h"
|
2017-11-01 22:21:10 +00:00
|
|
|
|
|
|
|
#include "resources/TextureResource.h"
|
2014-06-25 16:29:58 +00:00
|
|
|
#include "Renderer.h"
|
2017-11-01 22:21:10 +00:00
|
|
|
#include "ThemeData.h"
|
2014-06-25 16:29:58 +00:00
|
|
|
|
2017-06-08 23:18:27 +00:00
|
|
|
RatingComponent::RatingComponent(Window* window) : GuiComponent(window), mColorShift(0xFFFFFFFF)
|
2014-06-25 16:29:58 +00:00
|
|
|
{
|
|
|
|
mFilledTexture = TextureResource::get(":/star_filled.svg", true);
|
|
|
|
mUnfilledTexture = TextureResource::get(":/star_unfilled.svg", true);
|
|
|
|
mValue = 0.5f;
|
2017-10-28 20:24:35 +00:00
|
|
|
mSize = Vector2f(64 * NUM_RATING_STARS, 64);
|
2014-06-25 16:29:58 +00:00
|
|
|
updateVertices();
|
2017-06-14 01:01:08 +00:00
|
|
|
updateColors();
|
2014-06-25 16:29:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void RatingComponent::setValue(const std::string& value)
|
|
|
|
{
|
|
|
|
if(value.empty())
|
|
|
|
{
|
|
|
|
mValue = 0.0f;
|
|
|
|
}else{
|
|
|
|
mValue = stof(value);
|
|
|
|
if(mValue > 1.0f)
|
|
|
|
mValue = 1.0f;
|
|
|
|
else if(mValue < 0.0f)
|
|
|
|
mValue = 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
updateVertices();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string RatingComponent::getValue() const
|
|
|
|
{
|
2015-01-25 17:26:16 +00:00
|
|
|
// do not use std::to_string here as it will use the current locale
|
|
|
|
// and that sometimes encodes decimals as commas
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << mValue;
|
|
|
|
return ss.str();
|
2014-06-25 16:29:58 +00:00
|
|
|
}
|
|
|
|
|
2017-06-08 23:18:27 +00:00
|
|
|
void RatingComponent::setOpacity(unsigned char opacity)
|
|
|
|
{
|
|
|
|
mOpacity = opacity;
|
|
|
|
mColorShift = (mColorShift >> 8 << 8) | mOpacity;
|
|
|
|
updateColors();
|
|
|
|
}
|
|
|
|
|
|
|
|
void RatingComponent::setColorShift(unsigned int color)
|
|
|
|
{
|
|
|
|
mColorShift = color;
|
|
|
|
// Grab the opacity from the color shift because we may need to apply it if
|
|
|
|
// fading textures in
|
|
|
|
mOpacity = color & 0xff;
|
|
|
|
updateColors();
|
|
|
|
}
|
|
|
|
|
2014-06-25 16:29:58 +00:00
|
|
|
void RatingComponent::onSizeChanged()
|
|
|
|
{
|
|
|
|
if(mSize.y() == 0)
|
|
|
|
mSize[1] = mSize.x() / NUM_RATING_STARS;
|
|
|
|
else if(mSize.x() == 0)
|
|
|
|
mSize[0] = mSize.y() * NUM_RATING_STARS;
|
|
|
|
|
|
|
|
if(mSize.y() > 0)
|
|
|
|
{
|
2017-11-13 22:16:38 +00:00
|
|
|
size_t heightPx = (size_t)Math::round(mSize.y());
|
2017-01-22 23:28:06 +00:00
|
|
|
if (mFilledTexture)
|
|
|
|
mFilledTexture->rasterizeAt(heightPx, heightPx);
|
|
|
|
if(mUnfilledTexture)
|
|
|
|
mUnfilledTexture->rasterizeAt(heightPx, heightPx);
|
2014-06-25 16:29:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
updateVertices();
|
|
|
|
}
|
|
|
|
|
|
|
|
void RatingComponent::updateVertices()
|
|
|
|
{
|
|
|
|
const float numStars = NUM_RATING_STARS;
|
|
|
|
|
2017-11-13 22:16:38 +00:00
|
|
|
const float h = Math::round(getSize().y()); // is the same as a single star's width
|
|
|
|
const float w = Math::round(h * mValue * numStars);
|
|
|
|
const float fw = Math::round(h * numStars);
|
2014-06-25 16:29:58 +00:00
|
|
|
|
2017-10-28 20:24:35 +00:00
|
|
|
mVertices[0].pos = Vector2f(0.0f, 0.0f);
|
|
|
|
mVertices[0].tex = Vector2f(0.0f, 1.0f);
|
|
|
|
mVertices[1].pos = Vector2f(w, h);
|
|
|
|
mVertices[1].tex = Vector2f(mValue * numStars, 0.0f);
|
|
|
|
mVertices[2].pos = Vector2f(0.0f, h);
|
|
|
|
mVertices[2].tex = Vector2f(0.0f, 0.0f);
|
2014-06-25 16:29:58 +00:00
|
|
|
|
|
|
|
mVertices[3] = mVertices[0];
|
2017-10-28 20:24:35 +00:00
|
|
|
mVertices[4].pos = Vector2f(w, 0.0f);
|
|
|
|
mVertices[4].tex = Vector2f(mValue * numStars, 1.0f);
|
2014-06-25 16:29:58 +00:00
|
|
|
mVertices[5] = mVertices[1];
|
|
|
|
|
|
|
|
mVertices[6] = mVertices[4];
|
2017-10-28 20:24:35 +00:00
|
|
|
mVertices[7].pos = Vector2f(fw, h);
|
|
|
|
mVertices[7].tex = Vector2f(numStars, 0.0f);
|
2014-06-25 16:29:58 +00:00
|
|
|
mVertices[8] = mVertices[1];
|
|
|
|
|
|
|
|
mVertices[9] = mVertices[6];
|
2017-10-28 20:24:35 +00:00
|
|
|
mVertices[10].pos = Vector2f(fw, 0.0f);
|
|
|
|
mVertices[10].tex = Vector2f(numStars, 1.0f);
|
2014-06-25 16:29:58 +00:00
|
|
|
mVertices[11] = mVertices[7];
|
|
|
|
}
|
|
|
|
|
2017-06-08 23:18:27 +00:00
|
|
|
void RatingComponent::updateColors()
|
|
|
|
{
|
|
|
|
Renderer::buildGLColorArray(mColors, mColorShift, 12);
|
|
|
|
}
|
|
|
|
|
2017-10-28 20:24:35 +00:00
|
|
|
void RatingComponent::render(const Transform4x4f& parentTrans)
|
2014-06-25 16:29:58 +00:00
|
|
|
{
|
2017-10-28 20:24:35 +00:00
|
|
|
Transform4x4f trans = parentTrans * getTransform();
|
|
|
|
trans.round();
|
2014-06-25 16:29:58 +00:00
|
|
|
Renderer::setMatrix(trans);
|
|
|
|
|
|
|
|
glEnable(GL_TEXTURE_2D);
|
|
|
|
glEnable(GL_BLEND);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
|
|
|
|
glEnableClientState(GL_VERTEX_ARRAY);
|
|
|
|
glEnableClientState(GL_TEXTURE_COORD_ARRAY);
|
2017-06-08 23:18:27 +00:00
|
|
|
glEnableClientState(GL_COLOR_ARRAY);
|
2014-06-25 16:29:58 +00:00
|
|
|
|
|
|
|
glVertexPointer(2, GL_FLOAT, sizeof(Vertex), &mVertices[0].pos);
|
|
|
|
glTexCoordPointer(2, GL_FLOAT, sizeof(Vertex), &mVertices[0].tex);
|
2017-06-08 23:18:27 +00:00
|
|
|
glColorPointer(4, GL_UNSIGNED_BYTE, 0, mColors);
|
2014-06-25 16:29:58 +00:00
|
|
|
|
|
|
|
mFilledTexture->bind();
|
|
|
|
glDrawArrays(GL_TRIANGLES, 0, 6);
|
|
|
|
|
|
|
|
mUnfilledTexture->bind();
|
|
|
|
glDrawArrays(GL_TRIANGLES, 6, 6);
|
|
|
|
|
|
|
|
glDisableClientState(GL_VERTEX_ARRAY);
|
|
|
|
glDisableClientState(GL_TEXTURE_COORD_ARRAY);
|
2017-06-08 23:18:27 +00:00
|
|
|
glDisableClientState(GL_COLOR_ARRAY);
|
2014-06-25 16:29:58 +00:00
|
|
|
|
|
|
|
glDisable(GL_TEXTURE_2D);
|
|
|
|
glDisable(GL_BLEND);
|
|
|
|
|
|
|
|
renderChildren(trans);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RatingComponent::input(InputConfig* config, Input input)
|
|
|
|
{
|
|
|
|
if(config->isMappedTo("a", input) && input.value != 0)
|
|
|
|
{
|
|
|
|
mValue += 1.f / NUM_RATING_STARS;
|
|
|
|
if(mValue > 1.0f)
|
|
|
|
mValue = 0.0f;
|
|
|
|
|
|
|
|
updateVertices();
|
|
|
|
}
|
|
|
|
|
|
|
|
return GuiComponent::input(config, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RatingComponent::applyTheme(const std::shared_ptr<ThemeData>& theme, const std::string& view, const std::string& element, unsigned int properties)
|
|
|
|
{
|
|
|
|
GuiComponent::applyTheme(theme, view, element, properties);
|
|
|
|
|
|
|
|
using namespace ThemeFlags;
|
|
|
|
|
|
|
|
const ThemeData::ThemeElement* elem = theme->getElement(view, element, "rating");
|
|
|
|
if(!elem)
|
|
|
|
return;
|
|
|
|
|
|
|
|
bool imgChanged = false;
|
|
|
|
if(properties & PATH && elem->has("filledPath"))
|
|
|
|
{
|
|
|
|
mFilledTexture = TextureResource::get(elem->get<std::string>("filledPath"), true);
|
|
|
|
imgChanged = true;
|
|
|
|
}
|
|
|
|
if(properties & PATH && elem->has("unfilledPath"))
|
|
|
|
{
|
|
|
|
mUnfilledTexture = TextureResource::get(elem->get<std::string>("unfilledPath"), true);
|
|
|
|
imgChanged = true;
|
|
|
|
}
|
|
|
|
|
2017-06-08 23:18:27 +00:00
|
|
|
|
|
|
|
if(properties & COLOR && elem->has("color"))
|
|
|
|
setColorShift(elem->get<unsigned int>("color"));
|
|
|
|
|
2014-06-25 16:29:58 +00:00
|
|
|
if(imgChanged)
|
|
|
|
onSizeChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<HelpPrompt> RatingComponent::getHelpPrompts()
|
|
|
|
{
|
|
|
|
std::vector<HelpPrompt> prompts;
|
|
|
|
prompts.push_back(HelpPrompt("a", "add star"));
|
|
|
|
return prompts;
|
|
|
|
}
|