Blame gdk/quartz/gdkimage-quartz.c

Packit 98cdb6
/* gdkimage-quartz.c
Packit 98cdb6
 *
Packit 98cdb6
 * Copyright (C) 2005 Imendio AB
Packit 98cdb6
 *
Packit 98cdb6
 * This library is free software; you can redistribute it and/or
Packit 98cdb6
 * modify it under the terms of the GNU Lesser General Public
Packit 98cdb6
 * License as published by the Free Software Foundation; either
Packit 98cdb6
 * version 2 of the License, or (at your option) any later version.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is distributed in the hope that it will be useful,
Packit 98cdb6
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 98cdb6
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 98cdb6
 * Lesser General Public License for more details.
Packit 98cdb6
 *
Packit 98cdb6
 * You should have received a copy of the GNU Lesser General Public
Packit 98cdb6
 * License along with this library; if not, write to the
Packit 98cdb6
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit 98cdb6
 * Boston, MA 02111-1307, USA.
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
#include "config.h"
Packit 98cdb6
Packit 98cdb6
#include "gdk.h"
Packit 98cdb6
#include "gdkimage.h"
Packit 98cdb6
#include "gdkprivate-quartz.h"
Packit 98cdb6
Packit 98cdb6
static GObjectClass *parent_class;
Packit 98cdb6
Packit 98cdb6
GdkImage *
Packit 98cdb6
_gdk_quartz_image_copy_to_image (GdkDrawable *drawable,
Packit 98cdb6
				 GdkImage    *image,
Packit 98cdb6
				 gint         src_x,
Packit 98cdb6
				 gint         src_y,
Packit 98cdb6
				 gint         dest_x,
Packit 98cdb6
				 gint         dest_y,
Packit 98cdb6
				 gint         width,
Packit 98cdb6
				 gint         height)
Packit 98cdb6
{
Packit 98cdb6
  GdkScreen *screen;
Packit 98cdb6
  
Packit 98cdb6
  g_return_val_if_fail (GDK_IS_DRAWABLE_IMPL_QUARTZ (drawable), NULL);
Packit 98cdb6
  g_return_val_if_fail (image != NULL || (dest_x == 0 && dest_y == 0), NULL);
Packit 98cdb6
Packit 98cdb6
  screen = gdk_drawable_get_screen (drawable);
Packit 98cdb6
  if (!image)
Packit 98cdb6
    image = _gdk_image_new_for_depth (screen, GDK_IMAGE_FASTEST, NULL, 
Packit 98cdb6
				      width, height,
Packit 98cdb6
				      gdk_drawable_get_depth (drawable));
Packit 98cdb6
  
Packit 98cdb6
  if (GDK_IS_PIXMAP_IMPL_QUARTZ (drawable))
Packit 98cdb6
    {
Packit 98cdb6
      GdkPixmapImplQuartz *pix_impl;
Packit 98cdb6
      gint bytes_per_row;
Packit 98cdb6
      guchar *data;
Packit 98cdb6
      int x, y;
Packit 98cdb6
Packit 98cdb6
      pix_impl = GDK_PIXMAP_IMPL_QUARTZ (drawable);
Packit 98cdb6
      data = (guchar *)(pix_impl->data);
Packit 98cdb6
Packit 98cdb6
      if (src_x + width > pix_impl->width || src_y + height > pix_impl->height)
Packit 98cdb6
      	{
Packit 98cdb6
          g_warning ("Out of bounds copy-area for pixmap -> image conversion\n");
Packit 98cdb6
          return image;
Packit 98cdb6
        }
Packit 98cdb6
Packit 98cdb6
      switch (gdk_drawable_get_depth (drawable))
Packit 98cdb6
        {
Packit 98cdb6
        case 24:
Packit 98cdb6
          bytes_per_row = pix_impl->width * 4;
Packit 98cdb6
          for (y = 0; y < height; y++)
Packit 98cdb6
            {
Packit 98cdb6
              guchar *src = data + ((y + src_y) * bytes_per_row) + (src_x * 4);
Packit 98cdb6
Packit 98cdb6
              for (x = 0; x < width; x++)
Packit 98cdb6
                {
Packit 98cdb6
                  gint32 pixel;
Packit 98cdb6
	  
Packit 98cdb6
                  /* RGB24, 4 bytes per pixel, skip first. */
Packit 98cdb6
                  pixel = src[0] << 16 | src[1] << 8 | src[2];
Packit 98cdb6
                  src += 4;
Packit 98cdb6
Packit 98cdb6
                  gdk_image_put_pixel (image, dest_x + x, dest_y + y, pixel);
Packit 98cdb6
                }
Packit 98cdb6
            }
Packit 98cdb6
          break;
Packit 98cdb6
Packit 98cdb6
        case 32:
Packit 98cdb6
          bytes_per_row = pix_impl->width * 4;
Packit 98cdb6
          for (y = 0; y < height; y++)
Packit 98cdb6
            {
Packit 98cdb6
              guchar *src = data + ((y + src_y) * bytes_per_row) + (src_x * 4);
Packit 98cdb6
Packit 98cdb6
              for (x = 0; x < width; x++)
Packit 98cdb6
                {
Packit 98cdb6
                  gint32 pixel;
Packit 98cdb6
	  
Packit 98cdb6
                  /* ARGB32, 4 bytes per pixel. */
Packit 98cdb6
                  pixel = src[0] << 24 | src[1] << 16 | src[2] << 8 | src[3];
Packit 98cdb6
                  src += 4;
Packit 98cdb6
Packit 98cdb6
                  gdk_image_put_pixel (image, dest_x + x, dest_y + y, pixel);
Packit 98cdb6
                }
Packit 98cdb6
            }
Packit 98cdb6
          break;
Packit 98cdb6
Packit 98cdb6
        case 1: /* TODO: optimize */
Packit 98cdb6
          bytes_per_row = pix_impl->width;
Packit 98cdb6
          for (y = 0; y < height; y++)
Packit 98cdb6
            {
Packit 98cdb6
              guchar *src = data + ((y + src_y) * bytes_per_row) + src_x;
Packit 98cdb6
Packit 98cdb6
              for (x = 0; x < width; x++)
Packit 98cdb6
                {
Packit 98cdb6
                  gint32 pixel;
Packit 98cdb6
	  
Packit 98cdb6
                  /* 8 bits */
Packit 98cdb6
                  pixel = src[0];
Packit 98cdb6
                  src++;
Packit 98cdb6
Packit 98cdb6
                  gdk_image_put_pixel (image, dest_x + x, dest_y + y, pixel);
Packit 98cdb6
                }
Packit 98cdb6
            }
Packit 98cdb6
          break;
Packit 98cdb6
Packit 98cdb6
        default:
Packit 98cdb6
          g_warning ("Unsupported bit depth %d\n", gdk_drawable_get_depth (drawable));
Packit 98cdb6
          return image;
Packit 98cdb6
        }
Packit 98cdb6
    }
Packit 98cdb6
  else if (GDK_IS_WINDOW_IMPL_QUARTZ (drawable))
Packit 98cdb6
    {
Packit 98cdb6
      GdkQuartzView *view;
Packit 98cdb6
      NSBitmapImageRep *rep;
Packit 98cdb6
      guchar *data;
Packit 98cdb6
      int x, y;
Packit 98cdb6
      NSSize size;
Packit 98cdb6
      NSBitmapFormat format;
Packit 98cdb6
      gboolean has_alpha;
Packit 98cdb6
      gint bpp;
Packit 98cdb6
      gint r_byte = 0;
Packit 98cdb6
      gint g_byte = 1;
Packit 98cdb6
      gint b_byte = 2;
Packit 98cdb6
      gint a_byte = 3;
Packit 98cdb6
      gboolean le_image_data = FALSE;
Packit 98cdb6
Packit 98cdb6
      if (GDK_WINDOW_IMPL_QUARTZ (drawable) == GDK_WINDOW_IMPL_QUARTZ (GDK_WINDOW_OBJECT (_gdk_root)->impl))
Packit 98cdb6
        {
Packit 98cdb6
          /* Special case for the root window. */
Packit 98cdb6
	  CGRect rect = CGRectMake (src_x, src_y, width, height);
Packit 98cdb6
          CGImageRef root_image_ref = CGWindowListCreateImage (rect,
Packit 98cdb6
                                                               kCGWindowListOptionOnScreenOnly,
Packit 98cdb6
                                                               kCGNullWindowID,
Packit 98cdb6
                                                               kCGWindowImageDefault);
Packit 98cdb6
Packit 98cdb6
          /* HACK: the NSBitmapImageRep does not copy and convert
Packit 98cdb6
           * CGImageRef's data so it matches what NSBitmapImageRep can
Packit 98cdb6
           * express in its API (which is RGBA and ARGB, premultiplied
Packit 98cdb6
           * and unpremultiplied), it only references the CGImageRef.
Packit 98cdb6
           * Therefore we need to do the host byte swapping ourselves.
Packit 98cdb6
           */
Packit 98cdb6
          if (CGImageGetBitmapInfo (root_image_ref) & kCGBitmapByteOrder32Little)
Packit 98cdb6
            {
Packit 98cdb6
              r_byte = 3;
Packit 98cdb6
              g_byte = 2;
Packit 98cdb6
              b_byte = 1;
Packit 98cdb6
              a_byte = 0;
Packit 98cdb6
Packit 98cdb6
              le_image_data = TRUE;
Packit 98cdb6
            }
Packit 98cdb6
Packit 98cdb6
          rep = [[NSBitmapImageRep alloc] initWithCGImage: root_image_ref];
Packit 98cdb6
          CGImageRelease (root_image_ref);
Packit 98cdb6
        }
Packit 98cdb6
      else
Packit 98cdb6
        {
Packit 98cdb6
	  NSRect rect = NSMakeRect (src_x, src_y, width, height);
Packit 98cdb6
          view = GDK_WINDOW_IMPL_QUARTZ (drawable)->view;
Packit 98cdb6
Packit 98cdb6
          /* We return the image even if we can't copy to it. */
Packit 98cdb6
          if (![view lockFocusIfCanDraw])
Packit 98cdb6
            return image;
Packit 98cdb6
Packit 98cdb6
          rep = [[NSBitmapImageRep alloc] initWithFocusedViewRect: rect];
Packit 98cdb6
          [view unlockFocus];
Packit 98cdb6
        }
Packit 98cdb6
Packit 98cdb6
      data = [rep bitmapData];
Packit 98cdb6
      size = [rep size];
Packit 98cdb6
      format = [rep bitmapFormat];
Packit 98cdb6
      has_alpha = [rep hasAlpha];
Packit 98cdb6
      bpp = [rep bitsPerPixel] / 8;
Packit 98cdb6
Packit 98cdb6
      /* MORE HACK: AlphaFirst seems set for le_image_data, which is
Packit 98cdb6
       * technically correct, but really apple, are you kidding, it's
Packit 98cdb6
       * in fact ABGR, not ARGB as promised in NSBitmapImageRep's API.
Packit 98cdb6
       */
Packit 98cdb6
      if (!le_image_data && (format & NSAlphaFirstBitmapFormat))
Packit 98cdb6
        {
Packit 98cdb6
          r_byte = 1;
Packit 98cdb6
          g_byte = 2;
Packit 98cdb6
          b_byte = 3;
Packit 98cdb6
          a_byte = 0;
Packit 98cdb6
        }
Packit 98cdb6
Packit 98cdb6
      for (y = 0; y < size.height; y++)
Packit 98cdb6
        {
Packit 98cdb6
          guchar *src = data + y * [rep bytesPerRow];
Packit 98cdb6
Packit 98cdb6
          for (x = 0; x < size.width; x++)
Packit 98cdb6
            {
Packit 98cdb6
              guchar r = src[r_byte];
Packit 98cdb6
              guchar g = src[g_byte];
Packit 98cdb6
              guchar b = src[b_byte];
Packit 98cdb6
              gint32 pixel;
Packit 98cdb6
Packit 98cdb6
              if (has_alpha)
Packit 98cdb6
                {
Packit 98cdb6
                  guchar alpha = src[a_byte];
Packit 98cdb6
Packit 98cdb6
                  /* unpremultiply if alpha > 0 */
Packit 98cdb6
                  if (! (format & NSAlphaNonpremultipliedBitmapFormat) && alpha)
Packit 98cdb6
                    {
Packit 98cdb6
                      r = r * 255 / alpha;
Packit 98cdb6
                      g = g * 255 / alpha;
Packit 98cdb6
                      b = b * 255 / alpha;
Packit 98cdb6
                    }
Packit 98cdb6
Packit 98cdb6
                  if (image->byte_order == GDK_MSB_FIRST)
Packit 98cdb6
                    pixel = alpha | b << 8 | g << 16 | r << 24;
Packit 98cdb6
                  else
Packit 98cdb6
                    pixel = alpha << 24 | b << 16 | g << 8 | r;
Packit 98cdb6
                }
Packit 98cdb6
              else
Packit 98cdb6
                {
Packit 98cdb6
                  if (image->byte_order == GDK_MSB_FIRST)
Packit 98cdb6
                    pixel = b | g << 8 | r << 16;
Packit 98cdb6
                  else
Packit 98cdb6
                    pixel = b << 16 | g << 8 | r;
Packit 98cdb6
                }
Packit 98cdb6
Packit 98cdb6
              src += bpp;
Packit 98cdb6
Packit 98cdb6
              gdk_image_put_pixel (image, dest_x + x, dest_y + y, pixel);
Packit 98cdb6
            }
Packit 98cdb6
        }
Packit 98cdb6
Packit 98cdb6
      [rep release];
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return image;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_image_finalize (GObject *object)
Packit 98cdb6
{
Packit 98cdb6
  GdkImage *image = GDK_IMAGE (object);
Packit 98cdb6
Packit 98cdb6
  g_free (image->mem);
Packit 98cdb6
Packit 98cdb6
  G_OBJECT_CLASS (parent_class)->finalize (object);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gdk_image_class_init (GdkImageClass *klass)
Packit 98cdb6
{
Packit 98cdb6
  GObjectClass *object_class = G_OBJECT_CLASS (klass);
Packit 98cdb6
Packit 98cdb6
  parent_class = g_type_class_peek_parent (klass);
Packit 98cdb6
  
Packit 98cdb6
  object_class->finalize = gdk_image_finalize;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
GType
Packit 98cdb6
gdk_image_get_type (void)
Packit 98cdb6
{
Packit 98cdb6
  static GType object_type = 0;
Packit 98cdb6
Packit 98cdb6
  if (!object_type)
Packit 98cdb6
    {
Packit 98cdb6
      const GTypeInfo object_info =
Packit 98cdb6
      {
Packit 98cdb6
        sizeof (GdkImageClass),
Packit 98cdb6
        (GBaseInitFunc) NULL,
Packit 98cdb6
        (GBaseFinalizeFunc) NULL,
Packit 98cdb6
        (GClassInitFunc) gdk_image_class_init,
Packit 98cdb6
        NULL,           /* class_finalize */
Packit 98cdb6
        NULL,           /* class_data */
Packit 98cdb6
        sizeof (GdkImage),
Packit 98cdb6
        0,              /* n_preallocs */
Packit 98cdb6
        (GInstanceInitFunc) NULL,
Packit 98cdb6
      };
Packit 98cdb6
      
Packit 98cdb6
      object_type = g_type_register_static (G_TYPE_OBJECT,
Packit 98cdb6
                                            "GdkImage",
Packit 98cdb6
                                            &object_info,
Packit 98cdb6
					    0);
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  return object_type;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
GdkImage *
Packit 98cdb6
gdk_image_new_bitmap (GdkVisual *visual, gpointer data, gint width, gint height)
Packit 98cdb6
{
Packit 98cdb6
  /* We don't implement this function because it's broken, deprecated and 
Packit 98cdb6
   * tricky to implement. */
Packit 98cdb6
  g_warning ("This function is unimplemented");
Packit 98cdb6
Packit 98cdb6
  return NULL;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
GdkImage*
Packit 98cdb6
_gdk_image_new_for_depth (GdkScreen    *screen,
Packit 98cdb6
			  GdkImageType  type,
Packit 98cdb6
			  GdkVisual    *visual,
Packit 98cdb6
			  gint          width,
Packit 98cdb6
			  gint          height,
Packit 98cdb6
			  gint          depth)
Packit 98cdb6
{
Packit 98cdb6
  GdkImage *image;
Packit 98cdb6
Packit 98cdb6
  if (visual)
Packit 98cdb6
    depth = visual->depth;
Packit 98cdb6
Packit 98cdb6
  g_assert (depth == 24 || depth == 32);
Packit 98cdb6
Packit 98cdb6
  image = g_object_new (gdk_image_get_type (), NULL);
Packit 98cdb6
  image->type = type;
Packit 98cdb6
  image->visual = visual;
Packit 98cdb6
  image->width = width;
Packit 98cdb6
  image->height = height;
Packit 98cdb6
  image->depth = depth;
Packit 98cdb6
Packit 98cdb6
  image->byte_order = (G_BYTE_ORDER == G_LITTLE_ENDIAN) ? GDK_LSB_FIRST : GDK_MSB_FIRST;
Packit 98cdb6
Packit 98cdb6
  /* We only support images with bpp 4 */
Packit 98cdb6
  image->bpp = 4;
Packit 98cdb6
  image->bpl = image->width * image->bpp;
Packit 98cdb6
  image->bits_per_pixel = image->bpp * 8;
Packit 98cdb6
  
Packit 98cdb6
  image->mem = g_malloc (image->bpl * image->height);
Packit 98cdb6
  memset (image->mem, 0x00, image->bpl * image->height);
Packit 98cdb6
Packit 98cdb6
  return image;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
guint32
Packit 98cdb6
gdk_image_get_pixel (GdkImage *image,
Packit 98cdb6
		     gint x,
Packit 98cdb6
		     gint y)
Packit 98cdb6
{
Packit 98cdb6
  guchar *ptr;
Packit 98cdb6
Packit 98cdb6
  g_return_val_if_fail (image != NULL, 0);
Packit 98cdb6
  g_return_val_if_fail (x >= 0 && x < image->width, 0);
Packit 98cdb6
  g_return_val_if_fail (y >= 0 && y < image->height, 0);
Packit 98cdb6
Packit 98cdb6
  ptr = image->mem + y * image->bpl + x * image->bpp;
Packit 98cdb6
Packit 98cdb6
  return *(guint32 *)ptr;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
gdk_image_put_pixel (GdkImage *image,
Packit 98cdb6
		     gint x,
Packit 98cdb6
		     gint y,
Packit 98cdb6
		     guint32 pixel)
Packit 98cdb6
{
Packit 98cdb6
  guchar *ptr;
Packit 98cdb6
Packit 98cdb6
  ptr = image->mem + y * image->bpl + x * image->bpp;
Packit 98cdb6
Packit 98cdb6
  *(guint32 *)ptr = pixel;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
gint
Packit 98cdb6
_gdk_windowing_get_bits_for_depth (GdkDisplay *display,
Packit 98cdb6
				   gint        depth)
Packit 98cdb6
{
Packit 98cdb6
  if (depth == 24 || depth == 32)
Packit 98cdb6
    return 32;
Packit 98cdb6
  else
Packit 98cdb6
    g_assert_not_reached ();
Packit 98cdb6
Packit 98cdb6
  return 0;
Packit 98cdb6
}