SFML community forums

Help => Graphics => Topic started by: Debonair on July 30, 2015, 12:43:37 pm

Title: [Solved] Fade effect
Post by: Debonair on July 30, 2015, 12:43:37 pm
Been trying to create a fading effect, made a separate class for it, everything works fine, except not really. The basic idea was to draw a RectangleShape across the whole screen with a black fillcolor, and an alpha of 0, then the class gradually increases it over a period of time. When I run the thing, that period of time is noticeable, so the class does what it is supposed to do, except I don't see the rectangleshape at all, even though it is being drawn, it is transparent during the whole thing. Tried to do it just within the loop as well, without the class, the issue is the same.

Without class version:
        Clock clock;
        RectangleShape fade;
        fade.setPosition(Vector2f(0,0));
        fade.setFillColor(Color(0,0,0,0));
        fade.setSize(Vector2f(800,600));
        int i=0;
    while (window.isOpen())
    {
        if(clock.getElapsedTime()>seconds(0.04)){
            i++;
            fade.setFillColor(Color(0,0,0,i));
            clock.restart();
        }
     ///other stuff
        window.draw(fade);
     }
 

Am I just not supposed to be doing this with a RectangleShape?
Title: Re: Fade effect
Post by: Hapax on July 30, 2015, 12:51:32 pm
What is "other stuff"? If it's window.clear(), that's good. However, nothing suggests that you're using window.display() after that draw.
You should also be aware that this rectangle is fading from black transparent to black opaque so if the window is already black, you still won't see the rectangle.

If you're still having problems, a complete and minimal (http://en.sfml-dev.org/forums/index.php?topic=5559#msg36368) example should be provided.

EDIT: I just added your code to a standard SFML skeleton and it does what I expected it to: fades from green (I cleared the window to green) to black (the colour of the rectangle). It's slow, and takes around 10 seconds (as expected).
You might want to also check i has reached 255 as continuing after that makes no sense (and can cause unseen errors).
Title: Re: Fade effect
Post by: Debonair on July 30, 2015, 12:58:23 pm
What is "other stuff"? If it's window.clear(), that's good. However, nothing suggests that you're using window.display() after that draw.

My bad. Yeah, clear, display is there, along with other stuff being drawn, so it should be noticeable.
Title: Re: Fade effect
Post by: Hapax on July 30, 2015, 01:01:21 pm
Yeah, clear, display is there, along with other stuff being drawn, so it should be noticeable.
Again I ask, are you clearing the window as black? (window.clear() defaults to black)
If you're also drawing other objects, make sure the rectangle is not behind them all (draw before them).

Also, see my edit in my previous post.
Title: Re: Fade effect
Post by: Debonair on July 30, 2015, 01:03:29 pm
Yes, the fade is on the top of everything else (drawn last).
EDIT: Tried the classless version again, it does work actually, didn't notice because I didn't wait enough I guess, so it did work. Something with my class did not though, I'll figure that out. Thanks for the help.
Title: Re: [Solved] Fade effect
Post by: Synyster_Coder on July 31, 2015, 01:55:52 pm
Hey I know this is solved but I thought I might share a little more knowledge. If you use linear interpolation you should get a more manageable effect:

        int startAlpha = 0;
        int endAlpha = 255;
        int targetTime = 1000;
        sf::Clock timer;

        int currentTime = timer.getElapsedTime().asMilliseconds();
        int currentAlpha = endAlpha;
        if (currentTime >= targetTime)
        {
                //you are done
        }
        else
        {
                currentAlpha = startAlpha + (endAlpha - startAlpha)*currentTime / targetTime;
        }
        fade.setFillColor(Color(0, 0, 0, currentAlpha));
 
Title: Re: [Solved] Fade effect
Post by: Hapax on July 31, 2015, 02:35:32 pm
There's no need to explicitly use certain formats of time. You can use sf::Time more intuitively like so:
// this is outside the loop
int startAlpha = 0;
int endAlpha = 255;
sf::Time targetTime = sf::Seconds(1);
sf::Clock timer;

{
    // this is inside the loop
    sf::Time currentTime = timer.getElapsedTime();
    int currentAlpha = endAlpha;
    if (currentTime >= targetTime)
    {
        // oops: currentAlpha = endAlpha; // make certain that the alpha is at its final destination
        //you are done
    }
    else
    {
        currentAlpha = startAlpha + (endAlpha - startAlpha) * currentTime / targetTime);
    }
    // apply alpha to whatever colour is previously set
    sf::Color fadeColor = fade.getFillColor();
    fadeColor.a = currentAlpha;
    fade.setFillColor(fadeColor);
}
Be aware that alpha could be less than the end alpha when the time passes the target time.

EDIT: I realised that my point about alpha passing the final alpha was incorrect.
Title: Re: [Solved] Fade effect
Post by: Nexus on August 02, 2015, 10:03:44 am
Or you just use thor::FadeAnimation (http://www.bromeon.ch/libraries/thor/v2.0/doc/classthor_1_1_fade_animation.html) ;)
Title: Re: [Solved] Fade effect
Post by: Debonair on August 02, 2015, 07:41:54 pm
Thanks for the input guys.
Havent't tried Thor out yet, will check it out sometime.
Title: Re: [Solved] Fade effect
Post by: Ashenwraith on August 18, 2015, 04:20:33 am
Hi, I solved a similar issue and wrote a simple code example 5 years ago, maybe it might help:

http://en.sfml-dev.org/forums/index.php?topic=2693.msg17640#msg17640