Blame gdk/win32/gdkvisual-win32.c

Packit Service fb6fa5
/* GDK - The GIMP Drawing Kit
Packit Service fb6fa5
 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
Packit Service fb6fa5
 * Copyright (C) 1998-2002 Tor Lillqvist
Packit Service fb6fa5
 *
Packit Service fb6fa5
 * This library is free software; you can redistribute it and/or
Packit Service fb6fa5
 * modify it under the terms of the GNU Lesser General Public
Packit Service fb6fa5
 * License as published by the Free Software Foundation; either
Packit Service fb6fa5
 * version 2 of the License, or (at your option) any later version.
Packit Service fb6fa5
 *
Packit Service fb6fa5
 * This library is distributed in the hope that it will be useful,
Packit Service fb6fa5
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service fb6fa5
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service fb6fa5
 * Lesser General Public License for more details.
Packit Service fb6fa5
 *
Packit Service fb6fa5
 * You should have received a copy of the GNU Lesser General Public
Packit Service fb6fa5
 * License along with this library; if not, write to the
Packit Service fb6fa5
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit Service fb6fa5
 * Boston, MA 02111-1307, USA.
Packit Service fb6fa5
 */
Packit Service fb6fa5
Packit Service fb6fa5
/*
Packit Service fb6fa5
 * Modified by the GTK+ Team and others 1997-2000.  See the AUTHORS
Packit Service fb6fa5
 * file for a list of people on the GTK+ Team.  See the ChangeLog
Packit Service fb6fa5
 * files for a list of changes.  These files are distributed with
Packit Service fb6fa5
 * GTK+ at ftp://ftp.gtk.org/pub/gtk/. 
Packit Service fb6fa5
 */
Packit Service fb6fa5
Packit Service fb6fa5
#include "config.h"
Packit Service fb6fa5
#include <stdlib.h>
Packit Service fb6fa5
Packit Service fb6fa5
#include "gdkvisual.h"
Packit Service fb6fa5
#include "gdkscreen.h" /* gdk_screen_get_default() */
Packit Service fb6fa5
#include "gdkprivate-win32.h"
Packit Service fb6fa5
Packit Service fb6fa5
static void  gdk_visual_decompose_mask (gulong     mask,
Packit Service fb6fa5
					gint      *shift,
Packit Service fb6fa5
					gint      *prec);
Packit Service fb6fa5
Packit Service fb6fa5
static GdkVisual *system_visual = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
static gint available_depths[1];
Packit Service fb6fa5
Packit Service fb6fa5
static GdkVisualType available_types[1];
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
gdk_visual_finalize (GObject *object)
Packit Service fb6fa5
{
Packit Service fb6fa5
  g_error ("A GdkVisual object was finalized. This should not happen");
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
gdk_visual_class_init (GObjectClass *class)
Packit Service fb6fa5
{
Packit Service fb6fa5
  class->finalize = gdk_visual_finalize;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GType
Packit Service fb6fa5
gdk_visual_get_type (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  static GType object_type = 0;
Packit Service fb6fa5
Packit Service fb6fa5
  if (!object_type)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      const GTypeInfo object_info =
Packit Service fb6fa5
      {
Packit Service fb6fa5
        sizeof (GdkVisualClass),
Packit Service fb6fa5
        (GBaseInitFunc) NULL,
Packit Service fb6fa5
        (GBaseFinalizeFunc) NULL,
Packit Service fb6fa5
        (GClassInitFunc) gdk_visual_class_init,
Packit Service fb6fa5
        NULL,           /* class_finalize */
Packit Service fb6fa5
        NULL,           /* class_data */
Packit Service fb6fa5
        sizeof (GdkVisual),
Packit Service fb6fa5
        0,              /* n_preallocs */
Packit Service fb6fa5
        (GInstanceInitFunc) NULL,
Packit Service fb6fa5
      };
Packit Service fb6fa5
      
Packit Service fb6fa5
      object_type = g_type_register_static (G_TYPE_OBJECT,
Packit Service fb6fa5
                                            "GdkVisual",
Packit Service fb6fa5
                                            &object_info, 0);
Packit Service fb6fa5
    }
Packit Service fb6fa5
  
Packit Service fb6fa5
  return object_type;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
_gdk_visual_init (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  struct
Packit Service fb6fa5
  {
Packit Service fb6fa5
    BITMAPINFOHEADER bi;
Packit Service fb6fa5
    union
Packit Service fb6fa5
    {
Packit Service fb6fa5
      RGBQUAD colors[256];
Packit Service fb6fa5
      DWORD fields[256];
Packit Service fb6fa5
    } u;
Packit Service fb6fa5
  } bmi;
Packit Service fb6fa5
  HBITMAP hbm;
Packit Service fb6fa5
Packit Service fb6fa5
  const gint rastercaps = GetDeviceCaps (_gdk_display_hdc, RASTERCAPS);
Packit Service fb6fa5
  const int numcolors = GetDeviceCaps (_gdk_display_hdc, NUMCOLORS);
Packit Service fb6fa5
  gint bitspixel = GetDeviceCaps (_gdk_display_hdc, BITSPIXEL);
Packit Service fb6fa5
  gint map_entries = 0;
Packit Service fb6fa5
Packit Service fb6fa5
  system_visual = g_object_new (GDK_TYPE_VISUAL, NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (COLORMAP, g_print ("BITSPIXEL=%d NUMCOLORS=%d\n",
Packit Service fb6fa5
			       bitspixel, numcolors));
Packit Service fb6fa5
Packit Service fb6fa5
  if (rastercaps & RC_PALETTE)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      const int sizepalette = GetDeviceCaps (_gdk_display_hdc, SIZEPALETTE);
Packit Service fb6fa5
      gchar *max_colors = getenv ("GDK_WIN32_MAX_COLORS");
Packit Service fb6fa5
      system_visual->type = GDK_VISUAL_PSEUDO_COLOR;
Packit Service fb6fa5
Packit Service fb6fa5
      GDK_NOTE (COLORMAP, g_print ("SIZEPALETTE=%d\n", sizepalette));
Packit Service fb6fa5
      g_assert (sizepalette == 256);
Packit Service fb6fa5
Packit Service fb6fa5
      if (max_colors != NULL)
Packit Service fb6fa5
	_gdk_max_colors = atoi (max_colors);
Packit Service fb6fa5
      
Packit Service fb6fa5
      map_entries = _gdk_max_colors;
Packit Service fb6fa5
Packit Service fb6fa5
      if (map_entries >= 16 && map_entries < sizepalette)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  /* The calls to gdk_rgb_set_min_colors() here have knowledge
Packit Service fb6fa5
	   * of what color cubes gdk_rgb_do_colormaps() will try, and
Packit Service fb6fa5
	   * of the static system palette colors... XXX
Packit Service fb6fa5
	   */
Packit Service fb6fa5
	  if (map_entries < 32)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      map_entries = 16;
Packit Service fb6fa5
	      system_visual->type = GDK_VISUAL_STATIC_COLOR;
Packit Service fb6fa5
	      bitspixel = 4;
Packit Service fb6fa5
	      gdk_rgb_set_min_colors (2*2*2);
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  else if (map_entries < 64)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      map_entries = 32;
Packit Service fb6fa5
	      bitspixel = 5;
Packit Service fb6fa5
	      gdk_rgb_set_min_colors (3*3*3);
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  else if (map_entries < 128)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      map_entries = 64;
Packit Service fb6fa5
	      bitspixel = 6;
Packit Service fb6fa5
	      gdk_rgb_set_min_colors (3*3*3);
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  else if (map_entries < 256)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      map_entries = 128;
Packit Service fb6fa5
	      bitspixel = 7;
Packit Service fb6fa5
	      gdk_rgb_set_min_colors (5*5*4);
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  else
Packit Service fb6fa5
	    g_assert_not_reached ();
Packit Service fb6fa5
	}
Packit Service fb6fa5
      else
Packit Service fb6fa5
	map_entries = sizepalette;
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (bitspixel == 1 && numcolors == 16)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      bitspixel = 4;
Packit Service fb6fa5
      system_visual->type = GDK_VISUAL_STATIC_COLOR;
Packit Service fb6fa5
      map_entries = 16;
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (bitspixel == 1)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      system_visual->type = GDK_VISUAL_STATIC_GRAY;
Packit Service fb6fa5
      map_entries = 2;
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (bitspixel == 4)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      system_visual->type = GDK_VISUAL_STATIC_COLOR;
Packit Service fb6fa5
      map_entries = 16;
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (bitspixel == 8)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      system_visual->type = GDK_VISUAL_STATIC_COLOR;
Packit Service fb6fa5
      map_entries = 256;
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (bitspixel == 16)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      system_visual->type = GDK_VISUAL_TRUE_COLOR;
Packit Service fb6fa5
#if 1
Packit Service fb6fa5
      /* This code by Mike Enright,
Packit Service fb6fa5
       * see http://www.users.cts.com/sd/m/menright/display.html
Packit Service fb6fa5
       */
Packit Service fb6fa5
      memset (&bmi, 0, sizeof (bmi));
Packit Service fb6fa5
      bmi.bi.biSize = sizeof (bmi.bi);
Packit Service fb6fa5
Packit Service fb6fa5
      hbm = CreateCompatibleBitmap (_gdk_display_hdc, 1, 1);
Packit Service fb6fa5
      GetDIBits (_gdk_display_hdc, hbm, 0, 1, NULL,
Packit Service fb6fa5
		 (BITMAPINFO *) &bmi, DIB_RGB_COLORS);
Packit Service fb6fa5
      GetDIBits (_gdk_display_hdc, hbm, 0, 1, NULL,
Packit Service fb6fa5
		 (BITMAPINFO *) &bmi, DIB_RGB_COLORS);
Packit Service fb6fa5
      DeleteObject (hbm);
Packit Service fb6fa5
Packit Service fb6fa5
      if (bmi.bi.biCompression != BI_BITFIELDS)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  /* Either BI_RGB or BI_RLE_something
Packit Service fb6fa5
	   * .... or perhaps (!!) something else.
Packit Service fb6fa5
	   * Theoretically biCompression might be
Packit Service fb6fa5
	   * mmioFourCC('c','v','i','d') but I doubt it.
Packit Service fb6fa5
	   */
Packit Service fb6fa5
	  if (bmi.bi.biCompression == BI_RGB)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      /* It's 555 */
Packit Service fb6fa5
	      bitspixel = 15;
Packit Service fb6fa5
	      system_visual->red_mask   = 0x00007C00;
Packit Service fb6fa5
	      system_visual->green_mask = 0x000003E0;
Packit Service fb6fa5
	      system_visual->blue_mask  = 0x0000001F;
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  else
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      g_assert_not_reached ();
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	}
Packit Service fb6fa5
      else
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  DWORD allmasks =
Packit Service fb6fa5
	    bmi.u.fields[0] | bmi.u.fields[1] | bmi.u.fields[2];
Packit Service fb6fa5
	  int k = 0;
Packit Service fb6fa5
	  while (allmasks)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      if (allmasks&1)
Packit Service fb6fa5
		k++;
Packit Service fb6fa5
	      allmasks/=2;
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  bitspixel = k;
Packit Service fb6fa5
	  system_visual->red_mask = bmi.u.fields[0];
Packit Service fb6fa5
	  system_visual->green_mask = bmi.u.fields[1];
Packit Service fb6fa5
	  system_visual->blue_mask  = bmi.u.fields[2];
Packit Service fb6fa5
	}
Packit Service fb6fa5
#else
Packit Service fb6fa5
      /* Old, incorrect (but still working) code. */
Packit Service fb6fa5
#if 0
Packit Service fb6fa5
      system_visual->red_mask   = 0x0000F800;
Packit Service fb6fa5
      system_visual->green_mask = 0x000007E0;
Packit Service fb6fa5
      system_visual->blue_mask  = 0x0000001F;
Packit Service fb6fa5
#else
Packit Service fb6fa5
      system_visual->red_mask   = 0x00007C00;
Packit Service fb6fa5
      system_visual->green_mask = 0x000003E0;
Packit Service fb6fa5
      system_visual->blue_mask  = 0x0000001F;
Packit Service fb6fa5
#endif
Packit Service fb6fa5
#endif
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (bitspixel == 24 || bitspixel == 32)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      bitspixel = 24;
Packit Service fb6fa5
      system_visual->type = GDK_VISUAL_TRUE_COLOR;
Packit Service fb6fa5
      system_visual->red_mask   = 0x00FF0000;
Packit Service fb6fa5
      system_visual->green_mask = 0x0000FF00;
Packit Service fb6fa5
      system_visual->blue_mask  = 0x000000FF;
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else
Packit Service fb6fa5
    g_error ("_gdk_visual_init: unsupported BITSPIXEL: %d\n", bitspixel);
Packit Service fb6fa5
Packit Service fb6fa5
  system_visual->depth = bitspixel;
Packit Service fb6fa5
  system_visual->byte_order = GDK_LSB_FIRST;
Packit Service fb6fa5
  system_visual->bits_per_rgb = 42; /* Not used? */
Packit Service fb6fa5
Packit Service fb6fa5
  if ((system_visual->type == GDK_VISUAL_TRUE_COLOR) ||
Packit Service fb6fa5
      (system_visual->type == GDK_VISUAL_DIRECT_COLOR))
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gdk_visual_decompose_mask (system_visual->red_mask,
Packit Service fb6fa5
				 &system_visual->red_shift,
Packit Service fb6fa5
				 &system_visual->red_prec);
Packit Service fb6fa5
Packit Service fb6fa5
      gdk_visual_decompose_mask (system_visual->green_mask,
Packit Service fb6fa5
				 &system_visual->green_shift,
Packit Service fb6fa5
				 &system_visual->green_prec);
Packit Service fb6fa5
Packit Service fb6fa5
      gdk_visual_decompose_mask (system_visual->blue_mask,
Packit Service fb6fa5
				 &system_visual->blue_shift,
Packit Service fb6fa5
				 &system_visual->blue_prec);
Packit Service fb6fa5
      map_entries = 1 << (MAX (system_visual->red_prec,
Packit Service fb6fa5
			       MAX (system_visual->green_prec,
Packit Service fb6fa5
				    system_visual->blue_prec)));
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else
Packit Service fb6fa5
    {
Packit Service fb6fa5
      system_visual->red_mask = 0;
Packit Service fb6fa5
      system_visual->red_shift = 0;
Packit Service fb6fa5
      system_visual->red_prec = 0;
Packit Service fb6fa5
Packit Service fb6fa5
      system_visual->green_mask = 0;
Packit Service fb6fa5
      system_visual->green_shift = 0;
Packit Service fb6fa5
      system_visual->green_prec = 0;
Packit Service fb6fa5
Packit Service fb6fa5
      system_visual->blue_mask = 0;
Packit Service fb6fa5
      system_visual->blue_shift = 0;
Packit Service fb6fa5
      system_visual->blue_prec = 0;
Packit Service fb6fa5
    }
Packit Service fb6fa5
  system_visual->colormap_size = map_entries;
Packit Service fb6fa5
Packit Service fb6fa5
  available_depths[0] = system_visual->depth;
Packit Service fb6fa5
  available_types[0] = system_visual->type;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
gint
Packit Service fb6fa5
gdk_visual_get_best_depth (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  return available_depths[0];
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GdkVisualType
Packit Service fb6fa5
gdk_visual_get_best_type (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  return available_types[0];
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GdkVisual*
Packit Service fb6fa5
gdk_screen_get_system_visual (GdkScreen *screen)
Packit Service fb6fa5
{
Packit Service fb6fa5
  return system_visual;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GdkVisual*
Packit Service fb6fa5
gdk_visual_get_best (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  return ((GdkVisual*) system_visual);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GdkVisual*
Packit Service fb6fa5
gdk_visual_get_best_with_depth (gint depth)
Packit Service fb6fa5
{
Packit Service fb6fa5
  if (depth == system_visual->depth)
Packit Service fb6fa5
    return (GdkVisual*) system_visual;
Packit Service fb6fa5
  else
Packit Service fb6fa5
    return NULL;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GdkVisual*
Packit Service fb6fa5
gdk_visual_get_best_with_type (GdkVisualType visual_type)
Packit Service fb6fa5
{
Packit Service fb6fa5
  if (visual_type == system_visual->type)
Packit Service fb6fa5
    return system_visual;
Packit Service fb6fa5
  else
Packit Service fb6fa5
    return NULL;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GdkVisual*
Packit Service fb6fa5
gdk_visual_get_best_with_both (gint          depth,
Packit Service fb6fa5
			       GdkVisualType visual_type)
Packit Service fb6fa5
{
Packit Service fb6fa5
  if ((depth == system_visual->depth) && (visual_type == system_visual->type))
Packit Service fb6fa5
    return system_visual;
Packit Service fb6fa5
  else
Packit Service fb6fa5
    return NULL;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
gdk_query_depths  (gint **depths,
Packit Service fb6fa5
		   gint  *count)
Packit Service fb6fa5
{
Packit Service fb6fa5
  *count = 1;
Packit Service fb6fa5
  *depths = available_depths;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
gdk_query_visual_types (GdkVisualType **visual_types,
Packit Service fb6fa5
			gint           *count)
Packit Service fb6fa5
{
Packit Service fb6fa5
  *count = 1;
Packit Service fb6fa5
  *visual_types = available_types;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GList*
Packit Service fb6fa5
gdk_screen_list_visuals (GdkScreen *screen)
Packit Service fb6fa5
{
Packit Service fb6fa5
  return g_list_append (NULL, (gpointer) system_visual);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GdkScreen *
Packit Service fb6fa5
gdk_visual_get_screen (GdkVisual *visual)
Packit Service fb6fa5
{
Packit Service fb6fa5
  g_return_val_if_fail (GDK_IS_VISUAL (visual), NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  return gdk_screen_get_default ();
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
gdk_visual_decompose_mask (gulong  mask,
Packit Service fb6fa5
			   gint   *shift,
Packit Service fb6fa5
			   gint   *prec)
Packit Service fb6fa5
{
Packit Service fb6fa5
  *shift = 0;
Packit Service fb6fa5
  *prec = 0;
Packit Service fb6fa5
Packit Service fb6fa5
  while (!(mask & 0x1))
Packit Service fb6fa5
    {
Packit Service fb6fa5
      (*shift)++;
Packit Service fb6fa5
      mask >>= 1;
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  while (mask & 0x1)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      (*prec)++;
Packit Service fb6fa5
      mask >>= 1;
Packit Service fb6fa5
    }
Packit Service fb6fa5
}