SFML community forums

Help => Graphics => Topic started by: sam on November 12, 2010, 06:17:04 pm

Title: Most efficient way to draw a picture pixel by pixel?
Post 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:
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Laurent on November 12, 2010, 06:47:34 pm
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.
Title: Most efficient way to draw a picture pixel by pixel?
Post by: sam on November 12, 2010, 07:07:22 pm
Quote from: "Laurent"
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?  :?
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Laurent on November 12, 2010, 11:34:27 pm
It must be an array of sf::Uint8, with size Width * Height * 4. Components are stored in RGBA order.
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Groogy on November 13, 2010, 12:02:33 am
Quote from: "Laurent"
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.
Title: Most efficient way to draw a picture pixel by pixel?
Post by: sam on November 13, 2010, 12:57:37 am
I don't get it (stupid nonsense example of what I tried follows)  :?

Code: [Select]
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)
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Clairvoire on November 13, 2010, 09:04:37 am
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.  

Code: [Select]
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
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Xorlium on November 13, 2010, 10:31:16 am
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:

Code: [Select]

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
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Laurent on November 13, 2010, 11:13:53 am
Quote
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.
Title: Most efficient way to draw a picture pixel by pixel?
Post by: sam on November 13, 2010, 03:16:35 pm
Quote from: "Clairvoire"
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.  

Code: [Select]
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:
Code: [Select]
#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... :?



Quote from: "Xorlium"
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
Quote from: "sam"
(stupid nonsense example of what I tried follows)
But I even failed this simple test.  :?
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Laurent on November 13, 2010, 03:39:51 pm
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:
Title: Most efficient way to draw a picture pixel by pixel?
Post by: sam on November 13, 2010, 05:28:54 pm
Quote from: "Laurent"
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?
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Laurent on November 13, 2010, 06:08:19 pm
Yes it is.
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Spodi on November 13, 2010, 07:00:42 pm
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.
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Spidyy on November 13, 2010, 07:28:42 pm
Errors can lead to some nice and artistic effects
Title: Most efficient way to draw a picture pixel by pixel?
Post by: panithadrum on November 13, 2010, 09:28:54 pm
Quote from: "Spidyy"
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
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Xorlium on November 14, 2010, 01:02:55 am
Quote from: "sam"

Quote from: "Xorlium"
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
Quote from: "sam"
(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
Title: Most efficient way to draw a picture pixel by pixel?
Post by: AlexM on November 24, 2010, 07:05:45 am
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).


Code: [Select]

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;

}

}}
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Laurent on November 24, 2010, 08:21:57 am
You should use the debugger to see what's happening, and where.

Or at least you should print the value of arrayLength.

Quote
Code: [Select]
  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:
Code: [Select]
  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
Code: [Select]
std::fill(colorArray, colorArray + arrayLength + 4, 255);
And use std::vector, not raw arrays ;)
Title: Most efficient way to draw a picture pixel by pixel?
Post by: AlexM on November 24, 2010, 05:41:07 pm
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?
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Laurent on November 24, 2010, 05:56:36 pm
Quote
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.
Title: Most efficient way to draw a picture pixel by pixel?
Post by: AlexM on November 24, 2010, 06:05:06 pm
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 :)
Title: Most efficient way to draw a picture pixel by pixel?
Post by: Nexus on November 25, 2010, 12:15:44 am
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 ;)
Title: Most efficient way to draw a picture pixel by pixel?
Post by: AlexM on November 25, 2010, 02:08:20 am
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 :)