SFML community forums

Help => General => Topic started by: Jycerian on October 12, 2014, 07:13:14 pm

Title: Drag and Drop - Code Review
Post by: Jycerian on October 12, 2014, 07:13:14 pm
Hey guys, I have basically finished my first game named Gravity Surge (http://en.sfml-dev.org/forums/index.php?topic=15512.0 (http://en.sfml-dev.org/forums/index.php?topic=15512.0)) and now I am busy with my next game and learning more about programming.

My new game will be some sort of level editor/game combo and this is my first attempt for a drag and drop but I have no clue if my code is any good or not because I am not that advanced yet, I would really like to know what your opinions are!

Note : I intentionally made this one file just to learn.

(http://puu.sh/c9Do5/91b7e7f57e.png)

Code :
(click to show/hide)
Title: Re: Drag and Drop - Code Review
Post by: dabbertorres on October 12, 2014, 07:36:28 pm
First thing I noticed is that you're using some C-style headers and some C-style code.

For the headers:
#include <cstdio>
#include <cstdlib>
#include <ctime>
 

is the more C++ way of doing it.

Second, in modern C++, you don't use NULL, srand, rand, and time.
You use nullptr, <random>, and <chrono> (respectively).
Just google those two headers for guides to using them.

Those are the first things I noticed in my quick look over.
Title: Re: Drag and Drop - Code Review
Post by: Xornand on October 12, 2014, 07:56:25 pm
If I could suggest something, calling draw() every time you render a rectangle isn't the most efficient way to do it - that approach just doesn't scale well and could be a major bottleneck if you were drawing hundreds of rectangles every frame.

The approach I would suggest instead, would be to externally create and then load the texture of the rectangle as sf::Texture - or, you could create sf::RenderTexture and draw sf::RectangleShape (filled with the color of your choice) to it only once, upon initialization. The next step would be to create a custom Rectangle class, which would store sf::VertexArray, representing four corners. All the Rectangle instances would be then stored in some sort of container, so on each update you would be iterating over all the rectangles and copying their vertices onto one sf::VertexArray. After copying all the vertices, you could then just make one draw call, passing the sf::VertexArray and sf::RenderStates (with the texture parameter set to the texture of your rectangles).

The code would look something like that:
struct Rectangle
{
        sf::VertexArray vertexArray;
};

void render()
{
        sf::VertexArray vertexArray(sf::Quads);

    // let's assume that Rectangle instances are stored in std::vector<Rectangle>
        std::for_each(rectangles.begin(), rectangles.end(),
                [&](Rectangle& rectangle)
                {
                        for (auto i = 0; i < 4; ++i)
                        {
                                vertexArray.append(rectangle.vertexArray[i]);
                        }
                }
        );

        sf::RenderStates states;
        states.texture = texture.get(); // "texture" is std::shared_ptr<sf::Texture> stored somewhere else

    window.clear();
        window.draw(vertexArray, states); // only one draw call for all the rectangles
        window.display();
}
Title: Re: Drag and Drop - Code Review
Post by: eXpl0it3r on October 13, 2014, 01:34:05 pm
As dabbertorres said:

Furter more: