SFML community forums
Help => Graphics => Topic started by: sam on November 12, 2010, 06:17:04 pm
-
Hello,
in case I have a whole lot(!) of coordinates in an two-dimensional array, what would be the most efficient way to draw them on my window?
With GDI+ I used the LockBits-method like so (http://supercomputingblog.com/graphics/using-lockbits-in-gdi/) (just in reverse). Is there anything similar in SFML?
Thanks in advance! :oops:
-
Use your own array of pixels, and pass it to sf::Image when it's completely filled and ready to be displayed. That's the most efficient way of doing it, since all pixels will be transfered to the GPU at once.
-
Use your own array of pixels, and pass it to sf::Image when it's completely filled and ready to be displayed. That's the most efficient way of doing it, since all pixels will be transfered to the GPU at once.
Thank you, but how does the array have to look like? :?
-
It must be an array of sf::Uint8, with size Width * Height * 4. Components are stored in RGBA order.
-
It must be an array of sf::Uint8, with size Width * Height * 4. Components are stored in RGBA order.
Is this written somewhere in the documentation? If not that would be quite handy.
-
I don't get it (stupid nonsense example of what I tried follows) :?
sf::RenderWindow window(sf::VideoMode(800, 600, 32), "Test");
sf::Image image(800, 600, sf::Color(0, 0, 0));
sf::Sprite sprite;
sf::Uint8 *pixels = new sf::Uint8[800 * 600 * 4];
// ...
while(window.IsOpened())
{
// ...
for(int x = 0; x < 800; x++)
{
for(int y = 0; y < 600; y++)
{
pixels[y * x] = 255; // R?
pixels[y * x + 1] = 255; // G?
pixels[y * x + 2] = 255; // B?
pixels[y * x + 3] = 255; // A?
}
}
// ...
image.LoadFromPixels(800, 600, pixels);
sprite.SetImage(image);
window.Draw(sprite);
window.Display();
}
// ...
delete [] pixels;
// ...
Result:
(http://img5.imagebanana.com/img/z1qyadvu/doh.png)
-
In the code, you didn't account for x and y needing to move 4 bytes, since every pixel is 4 bytes, but in your array, it's going through it byte by byte.
for(int x = 0; x < 800; x++)
{
for(int y = 0; y < 600; y++)
{
pixels[(y * x)*4] = 255; // R?
pixels[(y * x)*4 + 1] = 255; // G?
pixels[(y * x)*4+ 2] = 255; // B?
pixels[(y * x)*4 + 3] = 255; // A?
}
}
Made the minor adjustments. This may be more favourable
-
Check your math there :)
For example, if x = 2, y =17 and x=17, y =2, you'd be getting always the same colour, but you clearly don't want that.
What you actually want is something like this:
for(int x = 0; x < 800; x++)
{
for(int y = 0; y < 600; y++)
{
pixels[4*(x * 600+y)] = 255; // R?
pixels[4*(x * 600+y)+1] = 255; // G?
pixels[4*(x * 600+y)+2] = 255; // B?
pixels[4*(x * 600+y)+3] = 255; // A?
}
}
or you might try with y*800+x, I don't remember how sfml expects it right now (it's late!)
Xorlium
-
Is this written somewhere in the documentation?
It is written in the documentation of LoadFromPixels, I think. If not, it is at least in SFML 2.
-
In the code, you didn't account for x and y needing to move 4 bytes, since every pixel is 4 bytes, but in your array, it's going through it byte by byte.
for(int x = 0; x < 800; x++)
{
for(int y = 0; y < 600; y++)
{
pixels[(y * x)*4] = 255; // R?
pixels[(y * x)*4 + 1] = 255; // G?
pixels[(y * x)*4+ 2] = 255; // B?
pixels[(y * x)*4 + 3] = 255; // A?
}
}
Made the minor adjustments. This may be more favourable
Doh! I guess I was tired yesterday. :oops:
The example's source code now looks like this:#include <SFML\Graphics.hpp>
#include <SFML\Window.hpp>
int main(int argc, char *argv[])
{
sf::RenderWindow window(sf::VideoMode(800, 600, 32), "Test");
sf::Image image(800, 600, sf::Color(0, 0, 0));
sf::Sprite sprite;
sf::Uint8 *pixels = new sf::Uint8[800 * 600 * 4];
while(window.IsOpened())
{
for(int x = 0; x < 800; x++)
{
for(int y = 0; y < 600; y++)
{
pixels[(y * x) * 4] = 255; // R?
pixels[(y * x) * 4 + 1] = 255; // G?
pixels[(y * x) * 4 + 2] = 255; // B?
pixels[(y * x) * 4 + 3] = 255; // A?
}
}
image.LoadFromPixels(800, 600, pixels);
sprite.SetImage(image);
window.Draw(sprite);
window.Display();
}
delete [] pixels;
return 0;
}
And this is the result:
(http://img5.imagebanana.com/img/ydt6v8ew/huh.png)
I kind of expected it to be all white now... :?
For example, if x = 2, y =17 and x=17, y =2, you'd be getting always the same colour, but you clearly don't want that.
No, it's right. I wanted to see if I can get everything to be painted white before I make the next big step (adding the array). That's why I wrote (stupid nonsense example of what I tried follows)
But I even failed this simple test. :?
-
It's not (x * y * 4), but (x + y * 800) * 4.
Try your formulas with simple numbers like 0, 1, 2 ... you clearly see that the result is wrong ;)
PS: that was a nice effect :lol:
-
It's not (x * y * 4), but (x + y * 800) * 4.
Try your formulas with simple numbers like 0, 1, 2 ... you clearly see that the result is wrong ;)
PS: that was a nice effect :lol:
Where does the 800 come from? Is it the width? So it's (x + y * width) * 4?
-
Yes it is.
-
Think of it this way: row one (top-left corner to top-right corner) is index 0 to Width. To get to row two (right below row one), you have to add the number of pixels in row one, which is = Width. So (y * Width) is your row offset, and x is your column offset.
-
Errors can lead to some nice and artistic effects
-
Errors can lead to some nice and artistic effects
Yes. My first impression was: Wow, it looks like a complex effect. Then I realized it was an error. :D
-
For example, if x = 2, y =17 and x=17, y =2, you'd be getting always the same colour, but you clearly don't want that.
No, it's right. I wanted to see if I can get everything to be painted white before I make the next big step (adding the array). That's why I wrote (stupid nonsense example of what I tried follows)
But I even failed this simple test. :?
No, I know you wanted everything white, what I'm saying is that x*y doesn't uniquely identify one pixel, so it can't be right. Do what I said and then Laurent repeated: 4(800*y+x) and then that plus 1, 2, and 3. Notice that uniquely identifies each pixel, in the sense that if you have a number n = 800*y+x, there are no two different x and y that give that n.
Xorlium
-
hey I've been trying to get this to work and I've been getting "std::bad_alloc at memory location" errors at runtime when populating the array.
I tried pasting sam's code that outputted the funky array since it at least compiles but I still get the same error on my end.
Here is the my original code (setting up the array in a constructor).
ScreenArray::ScreenArray( int width, int height )
{
int arrayLength = width * height;
colorArray = new sf::Uint8[arrayLength * 4];
for (int pos = 1; pos < arrayLength; pos++)
{
colorArray[(pos*4)-1] = 255;
colorArray[(pos*4)-2] = 255;
colorArray[(pos*4)-3] = 255;
colorArray[(pos*4)-4] = 255;
}
}}
-
You should use the debugger to see what's happening, and where.
Or at least you should print the value of arrayLength.
for (int pos = 1; pos < arrayLength; pos++)
{
colorArray[(pos*4)-1] = 255;
colorArray[(pos*4)-2] = 255;
colorArray[(pos*4)-3] = 255;
colorArray[(pos*4)-4] = 255;
}
This should be:
for (int pos = 0; pos < arrayLength; pos++)
{
colorArray[(pos*4)+0] = 255;
colorArray[(pos*4)+1] = 255;
colorArray[(pos*4)+2] = 255;
colorArray[(pos*4)+3] = 255;
}
Or simply
std::fill(colorArray, colorArray + arrayLength + 4, 255);
And use std::vector, not raw arrays ;)
-
I was using the debugger, I just wasn't using the debugger enough :D
Turns out the error was due to a method called right after populating the array, seems the array logic was fine.
One question regarding std::vector. You would recommend using it even when I know I won't be resizing the array?
-
You would recommend using it even when I know I won't be resizing the array?
Yes, because:
- you won't have to store the size yourself
- you won't have to manage the memory manually
std::vector is more than a resizable array, it's a generic abstraction of dynamic arrays.
-
ok thanks, I do use them when I know I'll be resizing often but I wasn't sure if I should use them in situations like this.
Thanks for the info, I'll make sure I do that in the future :)
-
And if you need static arrays (size known at compile time), you should still abandon raw arrays and use std::tr1::array or boost::array instead. It's worth all the comfortable features like copy semantics, debug assertions or the iterator interface -- with zero overhead in release mode ;)
-
Really? awesome.
I've been meaning to check out boost. I'll take a look into it this weekend. From a quick look I see a lot of useful classes there :)