Posting in response to this feature request issue:
https://github.com/SFML/SFML/issues/1128
To kick off the thread again, here's my first post from there ->
For simplicity and brevity in working with colors, it may be useful to support scalar math operators.
Color darkGreen = Color::Green - 127;
Color darkGray = Color::Black + 63;
Color darkerColor = someComputedColor * 63;
The equivalents are already possible now, but are more verbose:
Color darkGreen = Color::Green - Color(127, 127, 127, 0);
Color darkGray = Color::Black + Color(63, 63, 63, 0);
Color darkerColor = someComputedColor * Color(63, 63, 63, 255);
Problems I see:
- Accidental implicit conversions. I think this is already taken care of by providing operator uint32_t as a named member function instead of an overloaded operator. It is also taken care of by making the Color(uint32_t) constructor explicit.
- The operator * might be confusing, since it scales down, not up. The existing operator * already does this, and anyone who is "multiplying" colors already has to understand this difference as it stands today. I'd change my mind if operator / was introduced, but I think that's a very confusing construct to begin with for someone who isn't super familiar with color math, so I think it should not go into the library anyway.
- There might be anxiety on the user's side on whether the alpha channel is touched or not. This could be cleared up by documentation of the scalar operators. I think preserving the alpha channel in these operators is the more common case, and is the least surprising.
If there was interest, I'm perfectly happy and able to put in a pull request implementing these features. I'd prefer to ask for comments and/or a rejection before I spend time on it.
I don't think subtracting a single integer from a colour that stores four makes any sense. It's worse that it doesn't even affect all four, just a chosen three. It's not intuitive. Normally if you want a different colour, you would state the colour you want.
However, since alpha is a single value, this is the most obvious use for the this calculation. I'm not saying it should do this but sf::Color::Green - 20 looks more like the alpha should be reduced by that amount.
I don't understand why the so-called verbose version is bad.
Even if you do think it's ugly, you could always use a function for this.
// simple functions solution
sf::Color offsetGray(const sf::Uint8 value)
{
return sf::Color(value, value, value, 0);
}
// simple and clear usage
const sf::Color darkGreen = sf::Color::Green - offsetGray(127);
const sf::Color palerBlue = sf::Color::Blue + offsetGray(64);;
In fact, you could simply just create a colour from a single value:
const sf::Color darkGreen = sf::Color::Green - sf::Color(0x7f7f7f00);
const sf::Color palerBlue = sf::Color::Blue + sf::Color(0x40404000);
You also mention in your issue that you think there could be a float-based colour class and an integer-based colour class. That's obviously too confusing and there isn't really much need for it in SFML since you can build your own and convert to SFML colour when you actually use it with SFML. In fact, that's exactly what I did (https://github.com/Hapaxia/Plinth/blob/master/Plinth/Color.hpp) ;)
As for the scaling operator, you can simply add this yourself if you must have it; you don't need to modify SFML:
sf::Color operator*(const sf::Color color, const float scalar)
{
return sf::Color(static_cast<sf::Uint8>(color.r * scalar), static_cast<sf::Uint8>(color.g * scalar), static_cast<sf::Uint8>(color.b * scalar), color.a);
}
or
sf::Color operator*(const sf::Color color, const sf::Uint8 value)
{
const float scalar = value / 255.f;
return sf::Color(static_cast<sf::Uint8>(color.r * scalar), static_cast<sf::Uint8>(color.g * scalar), static_cast<sf::Uint8>(color.b * scalar), color.a);
}
You can, of course, also add the addition and subtract operators if you don't like my other simple method :p :
sf::Color operator+(const sf::Color color, const sf::Uint8 value)
{
return sf::Color(color.r + value, color.g + value, color.b + value, color.a);
}
sf::Color operator-(const sf::Color color, const sf::Uint8 value)
{
return sf::Color(color.r - value, color.g - value, color.b - value, color.a);
}
Again, in your own code without SFML being modified.
You may also want to consider the possibility of having some form of linear interpolation between colours (https://github.com/Hapaxia/Plinth/blob/8344e3581ee57bc41ef4a82abf6111247437dcad/Plinth/Sfml/Tween.hpp#L43-L46). Pretty useful for "mixing" colours.
I don't think affecting alpha and colors at the same time is useful
I think that is the problem.
The colour as a whole and the alpha are often used separately. This is not how sf::Color is; it stores four components and acts on them equally. An sf::Color is, therefore, the colour and the alpha as one (RGBA) and any operation on it should affect it all. Otherwise, it would be like subtracting 1.8 from 2.5 and it only affected the integer part (resulting in 1.5). You can manipulate the components separately but the colour components and the alpha components are treated equally.
The only intuitive "solution" :-X would be for a colour class that contained the actual colour as one part as the alpha as another:
sf::Color color;
color.color.r = red;
color.color.g = green;
color.color.b = blue;
color.alpha = alpha;
sf::Color orange(sf::Color::Color(255, 128, 0), 255); // resulting in something like this?
Then you could perform operation on the color-color that wouldn't affect the alpha.
Yes, color-color is very confusing :D
I'm fine with the way it is... :P