I have some RGBA data in Python 3 and I want to display the image it represents in a GTK3 window without using any extra libraries.
First thing I tried was editing a Pixbuf
's data as in an example (http://developer.gnome.org/gdk-pixbuf/stable/gdk-pixbuf-The-GdkPixbuf-Structure.html#put-pixel - sorry, I'm only allowed 2 links) in the (C) documentation, using Pixbuf.get_pixels()
. This gives me an immutable bytes
object, though, and so obviously won't work. (A gdk-pixbuf bug report (668084, would link to it if SO would let me) covers the same function, but stuff must've changed a bit since.)
Next, I tried creating a Pixbuf from the data using Pixbuf.new_from_data()
, but this is buggy too (see gdk-pixbuf bug 674691) (yes, that's me in comment 3).
Then I looked at Cairo: ImageSurface.create_for_data
should do this, but for some reason this isn't available in the Python 3 binding yet. (I have tried this with Python 2, turning the surface into a Pixbuf
afterwards and then wrapping that in a gtk.Image
, and it works. But I'm using Python3 (and even then, it's messy - Cairo is a vector graphics library, after all)).
I found a reference somewhere to using PIL to write the image out to a PNG file then reading that into a Pixbuf, but this is horrible, uses an extra library, and PIL isn't available for Python 3 anyway.
So...any other methods?
Based on Havok's answer, here's a working example:
from array import array
from gi.repository import Gtk as gtk, GdkPixbuf
pixels = array('H')
for i in range(20):
for j in range(20):
px = (i < 10, j >= 10, (i < 10) ^ (j < 10))
pixels.extend(65535 * c for c in px)
header = b'P6 20 20 65535 '
img_data = header + pixels
w = gtk.Window()
w.connect('delete-event', gtk.main_quit)
l = GdkPixbuf.PixbufLoader.new_with_type('pnm')
l.write(img_data)
w.add(gtk.Image.new_from_pixbuf(l.get_pixbuf()))
l.close()
w.show_all()
gtk.main()
Edit: actually, for correctness, I think this will need a pixels.byteswap()
on little-endian systems before adding to the header (though of course it doesn't matter with these colours).
Wow, this will be a hard one, pretty hard.
Neither PIL or Cairo's ImageSurface.create_from_data() are available for Python 3. Your only option left is using GdkPixbuf directly, but as you said this is not a perfect solution, in particular because you're expecting to use the alpha channel. My only thought is that you try to use this:
http://developer.gnome.org/gdk-pixbuf/stable/GdkPixbufLoader.html#gdk-pixbuf-loader-new-with-type
As I did at the end of here: Converting PIL GdkPixbuf
loader = GdkPixbuf.PixbufLoader.new_with_type('pnm')
loader.write(contents)
pixbuf = loader.get_pixbuf()
loader.close()
And try to see what types are supported, could not find the PyGObject version of what the C documentation says about gdk_pixbuf_format_get_name()
and gdk_pixbuf_get_formats()
. With the pnm format you will not have the alpha channel.
EDIT
Great you found the PyGObject functions, posting here the output for documentation purposes:
>>> from gi.repository import GdkPixbuf
>>> formats = GdkPixbuf.Pixbuf.get_formats()
>>> [f.get_name() for f in formats]
['GdkPixdata', 'ras', 'tiff', 'wmf', 'icns', 'ico', 'png', 'qtif',
'wbmp', 'gif', 'pnm', 'tga', 'ani', 'xbm', 'xpm', 'jpeg2000',
'pcx', 'jpeg', 'bmp', 'svg']
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With