Blame gdk/win32/gdkselection-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 <string.h>
Packit Service fb6fa5
#include <stdlib.h>
Packit Service fb6fa5
Packit Service fb6fa5
#include "gdkproperty.h"
Packit Service fb6fa5
#include "gdkselection.h"
Packit Service fb6fa5
#include "gdkdisplay.h"
Packit Service fb6fa5
#include "gdkprivate-win32.h"
Packit Service fb6fa5
Packit Service fb6fa5
/* We emulate the GDK_SELECTION window properties of windows (as used
Packit Service fb6fa5
 * in the X11 backend) by using a hash table from window handles to
Packit Service fb6fa5
 * GdkSelProp structs.
Packit Service fb6fa5
 */
Packit Service fb6fa5
Packit Service fb6fa5
typedef struct {
Packit Service fb6fa5
  guchar *data;
Packit Service fb6fa5
  gsize length;
Packit Service fb6fa5
  gint format;
Packit Service fb6fa5
  GdkAtom type;
Packit Service fb6fa5
} GdkSelProp;
Packit Service fb6fa5
Packit Service fb6fa5
static GHashTable *sel_prop_table = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
static GdkSelProp *dropfiles_prop = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
/* We store the owner of each selection in this table. Obviously, this only
Packit Service fb6fa5
 * is valid intra-app, and in fact it is necessary for the intra-app DND to work.
Packit Service fb6fa5
 */
Packit Service fb6fa5
static GHashTable *sel_owner_table = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
/* GdkAtoms for well-known image formats */
Packit Service fb6fa5
static GdkAtom *known_pixbuf_formats;
Packit Service fb6fa5
static int n_known_pixbuf_formats;
Packit Service fb6fa5
Packit Service fb6fa5
/* GdkAtoms for well-known text formats */
Packit Service fb6fa5
static GdkAtom text_plain;
Packit Service fb6fa5
static GdkAtom text_plain_charset_utf_8;
Packit Service fb6fa5
static GdkAtom text_plain_charset_CP1252;
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
_gdk_win32_selection_init (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GSList *pixbuf_formats;
Packit Service fb6fa5
  GSList *rover;
Packit Service fb6fa5
Packit Service fb6fa5
  sel_prop_table = g_hash_table_new (NULL, NULL);
Packit Service fb6fa5
  sel_owner_table = g_hash_table_new (NULL, NULL);
Packit Service fb6fa5
  _format_atom_table = g_hash_table_new (NULL, NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  pixbuf_formats = gdk_pixbuf_get_formats ();
Packit Service fb6fa5
Packit Service fb6fa5
  n_known_pixbuf_formats = 0;
Packit Service fb6fa5
  for (rover = pixbuf_formats; rover != NULL; rover = rover->next)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gchar **mime_types =
Packit Service fb6fa5
	gdk_pixbuf_format_get_mime_types ((GdkPixbufFormat *) rover->data);
Packit Service fb6fa5
Packit Service fb6fa5
      gchar **mime_type;
Packit Service fb6fa5
Packit Service fb6fa5
      for (mime_type = mime_types; *mime_type != NULL; mime_type++)
Packit Service fb6fa5
	n_known_pixbuf_formats++;
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  known_pixbuf_formats = g_new (GdkAtom, n_known_pixbuf_formats);
Packit Service fb6fa5
Packit Service fb6fa5
  n_known_pixbuf_formats = 0;
Packit Service fb6fa5
  for (rover = pixbuf_formats; rover != NULL; rover = rover->next)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gchar **mime_types =
Packit Service fb6fa5
	gdk_pixbuf_format_get_mime_types ((GdkPixbufFormat *) rover->data);
Packit Service fb6fa5
Packit Service fb6fa5
      gchar **mime_type;
Packit Service fb6fa5
Packit Service fb6fa5
      for (mime_type = mime_types; *mime_type != NULL; mime_type++)
Packit Service fb6fa5
	known_pixbuf_formats[n_known_pixbuf_formats++] = gdk_atom_intern (*mime_type, FALSE);
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  g_slist_free (pixbuf_formats);
Packit Service fb6fa5
  
Packit Service fb6fa5
  text_plain = gdk_atom_intern ("text/plain", FALSE);
Packit Service fb6fa5
  text_plain_charset_utf_8= gdk_atom_intern ("text/plain;charset=utf-8", FALSE);
Packit Service fb6fa5
  text_plain_charset_CP1252 = gdk_atom_intern ("text/plain;charset=CP1252", FALSE);
Packit Service fb6fa5
Packit Service fb6fa5
  g_hash_table_replace (_format_atom_table,
Packit Service fb6fa5
			GINT_TO_POINTER (_cf_png),
Packit Service fb6fa5
			_image_png);
Packit Service fb6fa5
Packit Service fb6fa5
  g_hash_table_replace (_format_atom_table,
Packit Service fb6fa5
			GINT_TO_POINTER (CF_DIB),
Packit Service fb6fa5
			_image_bmp);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
/* The specifications for COMPOUND_TEXT and STRING specify that C0 and
Packit Service fb6fa5
 * C1 are not allowed except for \n and \t, however the X conversions
Packit Service fb6fa5
 * routines for COMPOUND_TEXT only enforce this in one direction,
Packit Service fb6fa5
 * causing cut-and-paste of \r and \r\n separated text to fail.
Packit Service fb6fa5
 * This routine strips out all non-allowed C0 and C1 characters
Packit Service fb6fa5
 * from the input string and also canonicalizes \r, and \r\n to \n
Packit Service fb6fa5
 */
Packit Service fb6fa5
static gchar * 
Packit Service fb6fa5
sanitize_utf8 (const gchar *src,
Packit Service fb6fa5
	       gint         length)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GString *result = g_string_sized_new (length + 1);
Packit Service fb6fa5
  const gchar *p = src;
Packit Service fb6fa5
  const gchar *endp = src + length;
Packit Service fb6fa5
Packit Service fb6fa5
  while (p < endp)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      if (*p == '\r')
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  p++;
Packit Service fb6fa5
	  if (*p == '\n')
Packit Service fb6fa5
	    p++;
Packit Service fb6fa5
Packit Service fb6fa5
	  g_string_append_c (result, '\n');
Packit Service fb6fa5
	}
Packit Service fb6fa5
      else
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  gunichar ch = g_utf8_get_char (p);
Packit Service fb6fa5
	  char buf[7];
Packit Service fb6fa5
	  gint buflen;
Packit Service fb6fa5
	  
Packit Service fb6fa5
	  if (!((ch < 0x20 && ch != '\t' && ch != '\n') || (ch >= 0x7f && ch < 0xa0)))
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      buflen = g_unichar_to_utf8 (ch, buf);
Packit Service fb6fa5
	      g_string_append_len (result, buf, buflen);
Packit Service fb6fa5
	    }
Packit Service fb6fa5
Packit Service fb6fa5
	  p = g_utf8_next_char (p);
Packit Service fb6fa5
	}
Packit Service fb6fa5
    }
Packit Service fb6fa5
  g_string_append_c (result, '\0');
Packit Service fb6fa5
Packit Service fb6fa5
  return g_string_free (result, FALSE);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static gchar *
Packit Service fb6fa5
_gdk_utf8_to_string_target_internal (const gchar *str,
Packit Service fb6fa5
				     gint         length)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GError *error = NULL;
Packit Service fb6fa5
  
Packit Service fb6fa5
  gchar *tmp_str = sanitize_utf8 (str, length);
Packit Service fb6fa5
  gchar *result =  g_convert_with_fallback (tmp_str, -1,
Packit Service fb6fa5
					    "ISO-8859-1", "UTF-8",
Packit Service fb6fa5
					    NULL, NULL, NULL, &error);
Packit Service fb6fa5
  if (!result)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      g_warning ("Error converting from UTF-8 to STRING: %s",
Packit Service fb6fa5
		 error->message);
Packit Service fb6fa5
      g_error_free (error);
Packit Service fb6fa5
    }
Packit Service fb6fa5
  
Packit Service fb6fa5
  g_free (tmp_str);
Packit Service fb6fa5
  return result;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
selection_property_store (GdkWindow *owner,
Packit Service fb6fa5
			  GdkAtom    type,
Packit Service fb6fa5
			  gint       format,
Packit Service fb6fa5
			  guchar    *data,
Packit Service fb6fa5
			  gint       length)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GdkSelProp *prop;
Packit Service fb6fa5
Packit Service fb6fa5
  g_return_if_fail (type != GDK_TARGET_STRING);
Packit Service fb6fa5
Packit Service fb6fa5
  prop = g_hash_table_lookup (sel_prop_table, GDK_WINDOW_HWND (owner));
Packit Service fb6fa5
Packit Service fb6fa5
  if (prop != NULL)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      g_free (prop->data);
Packit Service fb6fa5
      g_free (prop);
Packit Service fb6fa5
      g_hash_table_remove (sel_prop_table, GDK_WINDOW_HWND (owner));
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  prop = g_new (GdkSelProp, 1);
Packit Service fb6fa5
Packit Service fb6fa5
  prop->data = data;
Packit Service fb6fa5
  prop->length = length;
Packit Service fb6fa5
  prop->format = format;
Packit Service fb6fa5
  prop->type = type;
Packit Service fb6fa5
Packit Service fb6fa5
  g_hash_table_insert (sel_prop_table, GDK_WINDOW_HWND (owner), prop);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
_gdk_dropfiles_store (gchar *data)
Packit Service fb6fa5
{
Packit Service fb6fa5
  if (data != NULL)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      g_assert (dropfiles_prop == NULL);
Packit Service fb6fa5
Packit Service fb6fa5
      dropfiles_prop = g_new (GdkSelProp, 1);
Packit Service fb6fa5
      dropfiles_prop->data = (guchar *) data;
Packit Service fb6fa5
      dropfiles_prop->length = strlen (data) + 1;
Packit Service fb6fa5
      dropfiles_prop->format = 8;
Packit Service fb6fa5
      dropfiles_prop->type = _text_uri_list;
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else
Packit Service fb6fa5
    {
Packit Service fb6fa5
      if (dropfiles_prop != NULL)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  g_free (dropfiles_prop->data);
Packit Service fb6fa5
	  g_free (dropfiles_prop);
Packit Service fb6fa5
	}
Packit Service fb6fa5
      dropfiles_prop = NULL;
Packit Service fb6fa5
    }
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static gchar *
Packit Service fb6fa5
get_mapped_gdk_atom_name (GdkAtom gdk_target)
Packit Service fb6fa5
{
Packit Service fb6fa5
  if (gdk_target == _image_png)
Packit Service fb6fa5
    return g_strdup ("PNG");
Packit Service fb6fa5
Packit Service fb6fa5
  if (gdk_target == _image_jpeg)
Packit Service fb6fa5
    return g_strdup ("JFIF");
Packit Service fb6fa5
  
Packit Service fb6fa5
  if (gdk_target == _image_gif)
Packit Service fb6fa5
    return g_strdup ("GIF");
Packit Service fb6fa5
  
Packit Service fb6fa5
  return gdk_atom_name (gdk_target);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
gboolean
Packit Service fb6fa5
gdk_selection_owner_set_for_display (GdkDisplay *display,
Packit Service fb6fa5
                                     GdkWindow  *owner,
Packit Service fb6fa5
                                     GdkAtom     selection,
Packit Service fb6fa5
                                     guint32     time,
Packit Service fb6fa5
                                     gboolean    send_event)
Packit Service fb6fa5
{
Packit Service fb6fa5
  HWND hwnd;
Packit Service fb6fa5
  GdkEvent tmp_event;
Packit Service fb6fa5
Packit Service fb6fa5
  g_return_val_if_fail (display == _gdk_display, FALSE);
Packit Service fb6fa5
  g_return_val_if_fail (selection != GDK_NONE, FALSE);
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (DND, {
Packit Service fb6fa5
      gchar *sel_name = gdk_atom_name (selection);
Packit Service fb6fa5
Packit Service fb6fa5
      g_print ("gdk_selection_owner_set_for_display: %p %s\n",
Packit Service fb6fa5
	       (owner ? GDK_WINDOW_HWND (owner) : NULL),
Packit Service fb6fa5
	       sel_name);
Packit Service fb6fa5
      g_free (sel_name);
Packit Service fb6fa5
    });
Packit Service fb6fa5
Packit Service fb6fa5
  if (selection != GDK_SELECTION_CLIPBOARD)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      if (owner != NULL)
Packit Service fb6fa5
	g_hash_table_insert (sel_owner_table, selection, GDK_WINDOW_HWND (owner));
Packit Service fb6fa5
      else
Packit Service fb6fa5
	g_hash_table_remove (sel_owner_table, selection);
Packit Service fb6fa5
      return TRUE;
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  /* Rest of this function handles the CLIPBOARD selection */
Packit Service fb6fa5
  if (owner != NULL)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      if (GDK_WINDOW_DESTROYED (owner))
Packit Service fb6fa5
	return FALSE;
Packit Service fb6fa5
Packit Service fb6fa5
      hwnd = GDK_WINDOW_HWND (owner);
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else
Packit Service fb6fa5
    hwnd = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
  if (!API_CALL (OpenClipboard, (hwnd)))
Packit Service fb6fa5
    return FALSE;
Packit Service fb6fa5
Packit Service fb6fa5
  _ignore_destroy_clipboard = TRUE;
Packit Service fb6fa5
  GDK_NOTE (DND, g_print ("... EmptyClipboard()\n"));
Packit Service fb6fa5
  if (!API_CALL (EmptyClipboard, ()))
Packit Service fb6fa5
    {
Packit Service fb6fa5
      _ignore_destroy_clipboard = FALSE;
Packit Service fb6fa5
      API_CALL (CloseClipboard, ());
Packit Service fb6fa5
      return FALSE;
Packit Service fb6fa5
    }
Packit Service fb6fa5
  _ignore_destroy_clipboard = FALSE;
Packit Service fb6fa5
Packit Service fb6fa5
  if (!API_CALL (CloseClipboard, ()))
Packit Service fb6fa5
    return FALSE;
Packit Service fb6fa5
Packit Service fb6fa5
  if (owner != NULL)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      /* Send ourselves a selection request message so that
Packit Service fb6fa5
       * gdk_property_change will be called to store the clipboard
Packit Service fb6fa5
       * data.
Packit Service fb6fa5
       */
Packit Service fb6fa5
      GDK_NOTE (DND, g_print ("... sending GDK_SELECTION_REQUEST to ourselves\n"));
Packit Service fb6fa5
      tmp_event.selection.type = GDK_SELECTION_REQUEST;
Packit Service fb6fa5
      tmp_event.selection.window = owner;
Packit Service fb6fa5
      tmp_event.selection.send_event = FALSE;
Packit Service fb6fa5
      tmp_event.selection.selection = selection;
Packit Service fb6fa5
      tmp_event.selection.target = _utf8_string;
Packit Service fb6fa5
      tmp_event.selection.property = _gdk_selection;
Packit Service fb6fa5
      tmp_event.selection.requestor = hwnd;
Packit Service fb6fa5
      tmp_event.selection.time = time;
Packit Service fb6fa5
Packit Service fb6fa5
      gdk_event_put (&tmp_event);
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  return TRUE;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
GdkWindow*
Packit Service fb6fa5
gdk_selection_owner_get_for_display (GdkDisplay *display,
Packit Service fb6fa5
                                     GdkAtom     selection)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GdkWindow *window;
Packit Service fb6fa5
Packit Service fb6fa5
  g_return_val_if_fail (display == _gdk_display, NULL);
Packit Service fb6fa5
  g_return_val_if_fail (selection != GDK_NONE, NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  if (selection == GDK_SELECTION_CLIPBOARD)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      HWND owner = GetClipboardOwner ();
Packit Service fb6fa5
Packit Service fb6fa5
      if (owner == NULL)
Packit Service fb6fa5
	return NULL;
Packit Service fb6fa5
Packit Service fb6fa5
      return gdk_win32_handle_table_lookup ((GdkNativeWindow) owner);
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  window = gdk_window_lookup ((GdkNativeWindow) g_hash_table_lookup (sel_owner_table, selection));
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (DND, {
Packit Service fb6fa5
      gchar *sel_name = gdk_atom_name (selection);
Packit Service fb6fa5
      
Packit Service fb6fa5
      g_print ("gdk_selection_owner_get: %s = %p\n",
Packit Service fb6fa5
	       sel_name,
Packit Service fb6fa5
	       (window ? GDK_WINDOW_HWND (window) : NULL));
Packit Service fb6fa5
      g_free (sel_name);
Packit Service fb6fa5
    });
Packit Service fb6fa5
Packit Service fb6fa5
  return window;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static void
Packit Service fb6fa5
generate_selection_notify (GdkWindow *requestor,
Packit Service fb6fa5
			   GdkAtom    selection,
Packit Service fb6fa5
			   GdkAtom    target,
Packit Service fb6fa5
			   GdkAtom    property,
Packit Service fb6fa5
			   guint32    time)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GdkEvent tmp_event;
Packit Service fb6fa5
Packit Service fb6fa5
  tmp_event.selection.type = GDK_SELECTION_NOTIFY;
Packit Service fb6fa5
  tmp_event.selection.window = requestor;
Packit Service fb6fa5
  tmp_event.selection.send_event = FALSE;
Packit Service fb6fa5
  tmp_event.selection.selection = selection;
Packit Service fb6fa5
  tmp_event.selection.target = target;
Packit Service fb6fa5
  tmp_event.selection.property = property;
Packit Service fb6fa5
  tmp_event.selection.requestor = 0;
Packit Service fb6fa5
  tmp_event.selection.time = time;
Packit Service fb6fa5
Packit Service fb6fa5
  gdk_event_put (&tmp_event);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
gdk_selection_convert (GdkWindow *requestor,
Packit Service fb6fa5
		       GdkAtom    selection,
Packit Service fb6fa5
		       GdkAtom    target,
Packit Service fb6fa5
		       guint32    time)
Packit Service fb6fa5
{
Packit Service fb6fa5
  HGLOBAL hdata;
Packit Service fb6fa5
  GdkAtom property = _gdk_selection;
Packit Service fb6fa5
Packit Service fb6fa5
  g_return_if_fail (selection != GDK_NONE);
Packit Service fb6fa5
  g_return_if_fail (requestor != NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  if (GDK_WINDOW_DESTROYED (requestor))
Packit Service fb6fa5
    return;
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (DND, {
Packit Service fb6fa5
      gchar *sel_name = gdk_atom_name (selection);
Packit Service fb6fa5
      gchar *tgt_name = gdk_atom_name (target);
Packit Service fb6fa5
      
Packit Service fb6fa5
      g_print ("gdk_selection_convert: %p %s %s\n",
Packit Service fb6fa5
	       GDK_WINDOW_HWND (requestor),
Packit Service fb6fa5
	       sel_name, tgt_name);
Packit Service fb6fa5
      g_free (sel_name);
Packit Service fb6fa5
      g_free (tgt_name);
Packit Service fb6fa5
    });
Packit Service fb6fa5
Packit Service fb6fa5
  if (selection == GDK_SELECTION_CLIPBOARD && target == _targets)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gint ntargets, fmt;
Packit Service fb6fa5
      GdkAtom *targets;
Packit Service fb6fa5
      gboolean has_text = FALSE;
Packit Service fb6fa5
      gboolean has_png = FALSE;
Packit Service fb6fa5
      gboolean has_bmp = FALSE;
Packit Service fb6fa5
Packit Service fb6fa5
      if (!API_CALL (OpenClipboard, (GDK_WINDOW_HWND (requestor))))
Packit Service fb6fa5
	return;
Packit Service fb6fa5
Packit Service fb6fa5
      targets = g_new (GdkAtom, CountClipboardFormats ());
Packit Service fb6fa5
      ntargets = 0;
Packit Service fb6fa5
Packit Service fb6fa5
      for (fmt = 0; 0 != (fmt = EnumClipboardFormats (fmt)); )
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  if (fmt == _cf_png)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      targets[ntargets++] = _image_png;
Packit Service fb6fa5
	      has_png = TRUE;
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	}
Packit Service fb6fa5
Packit Service fb6fa5
      for (fmt = 0; 0 != (fmt = EnumClipboardFormats (fmt)); )
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  gchar sFormat[80];
Packit Service fb6fa5
Packit Service fb6fa5
	  if (fmt == CF_UNICODETEXT || fmt == CF_TEXT)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      /* Advertise text to GDK always as UTF8_STRING */
Packit Service fb6fa5
	      if (!has_text)
Packit Service fb6fa5
		targets[ntargets++] = _utf8_string;
Packit Service fb6fa5
	      has_text = TRUE;
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  else if (fmt == _cf_png)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      /* Already handled above */
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  else if (fmt == CF_DIB ||
Packit Service fb6fa5
		   fmt == CF_DIBV5)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      /* Don't bother telling that a bitmap is present if there is
Packit Service fb6fa5
	       * also PNG, which is much more reliable in transferring
Packit Service fb6fa5
	       * transparency.
Packit Service fb6fa5
	       */
Packit Service fb6fa5
	      if (!has_bmp && !has_png)
Packit Service fb6fa5
		targets[ntargets++] = _image_bmp;
Packit Service fb6fa5
	      has_bmp = TRUE;
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  else if (fmt == _cf_jfif)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      /* Ditto for JPEG */
Packit Service fb6fa5
	      if (!has_png)
Packit Service fb6fa5
		targets[ntargets++] = _image_jpeg;
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  else if (fmt == _cf_gif)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      /* Ditto for GIF.
Packit Service fb6fa5
	       */
Packit Service fb6fa5
	      if (!has_png)
Packit Service fb6fa5
		targets[ntargets++] = _image_gif;
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  else if (GetClipboardFormatName (fmt, sFormat, 80) > 0)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      if (strcmp (sFormat, "image/bmp") == 0 ||
Packit Service fb6fa5
		  strcmp (sFormat, "image/x-bmp") == 0 ||
Packit Service fb6fa5
		  strcmp (sFormat, "image/x-MS-bmp") == 0 ||
Packit Service fb6fa5
		  strcmp (sFormat, "image/x-icon") == 0 ||
Packit Service fb6fa5
		  strcmp (sFormat, "image/x-ico") == 0 ||
Packit Service fb6fa5
		  strcmp (sFormat, "image/x-win-bitmap") == 0)
Packit Service fb6fa5
		{
Packit Service fb6fa5
		  /* Ignore these (from older GTK+ versions
Packit Service fb6fa5
		   * presumably), as the same image in the CF_DIB
Packit Service fb6fa5
		   * format will also be on the clipboard anyway.
Packit Service fb6fa5
		   */
Packit Service fb6fa5
		}
Packit Service fb6fa5
	      else
Packit Service fb6fa5
		targets[ntargets++] = gdk_atom_intern (sFormat, FALSE);
Packit Service fb6fa5
            }
Packit Service fb6fa5
        }
Packit Service fb6fa5
Packit Service fb6fa5
      GDK_NOTE (DND, {
Packit Service fb6fa5
	  int i;
Packit Service fb6fa5
	  
Packit Service fb6fa5
	  g_print ("... ");
Packit Service fb6fa5
	  for (i = 0; i < ntargets; i++)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      gchar *atom_name = gdk_atom_name (targets[i]);
Packit Service fb6fa5
Packit Service fb6fa5
	      g_print ("%s", atom_name);
Packit Service fb6fa5
	      g_free (atom_name);
Packit Service fb6fa5
	      if (i < ntargets - 1)
Packit Service fb6fa5
		g_print (", ");
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  g_print ("\n");
Packit Service fb6fa5
	});
Packit Service fb6fa5
Packit Service fb6fa5
      if (ntargets > 0)
Packit Service fb6fa5
	selection_property_store (requestor, GDK_SELECTION_TYPE_ATOM,
Packit Service fb6fa5
				  32, (guchar *) targets,
Packit Service fb6fa5
				  ntargets * sizeof (GdkAtom));
Packit Service fb6fa5
      else
Packit Service fb6fa5
	property = GDK_NONE;
Packit Service fb6fa5
Packit Service fb6fa5
      API_CALL (CloseClipboard, ());
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (selection == GDK_SELECTION_CLIPBOARD && target == _utf8_string)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      /* Converting the CLIPBOARD selection means he wants the
Packit Service fb6fa5
       * contents of the clipboard. Get the clipboard data, and store
Packit Service fb6fa5
       * it for later.
Packit Service fb6fa5
       */
Packit Service fb6fa5
      if (!API_CALL (OpenClipboard, (GDK_WINDOW_HWND (requestor))))
Packit Service fb6fa5
	return;
Packit Service fb6fa5
Packit Service fb6fa5
      if ((hdata = GetClipboardData (CF_UNICODETEXT)) != NULL)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  wchar_t *ptr, *p, *q;
Packit Service fb6fa5
	  guchar *data;
Packit Service fb6fa5
	  glong length, wclen;
Packit Service fb6fa5
Packit Service fb6fa5
	  if ((ptr = GlobalLock (hdata)) != NULL)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      length = GlobalSize (hdata);
Packit Service fb6fa5
Packit Service fb6fa5
	      GDK_NOTE (DND, g_print ("... CF_UNICODETEXT: %ld bytes\n",
Packit Service fb6fa5
				      length));
Packit Service fb6fa5
Packit Service fb6fa5
	      /* Strip out \r */
Packit Service fb6fa5
	      p = ptr;
Packit Service fb6fa5
	      q = ptr;
Packit Service fb6fa5
	      wclen = 0;
Packit Service fb6fa5
	      while (p < ptr + length / 2)
Packit Service fb6fa5
		{
Packit Service fb6fa5
		  if (*p != '\r')
Packit Service fb6fa5
		    {
Packit Service fb6fa5
		      *q++ = *p;
Packit Service fb6fa5
		      wclen++;
Packit Service fb6fa5
		    }
Packit Service fb6fa5
		  p++;
Packit Service fb6fa5
		}
Packit Service fb6fa5
Packit Service fb6fa5
	      data = g_utf16_to_utf8 (ptr, wclen, NULL, NULL, NULL);
Packit Service fb6fa5
Packit Service fb6fa5
	      if (data)
Packit Service fb6fa5
		selection_property_store (requestor, _utf8_string, 8,
Packit Service fb6fa5
					  data, strlen (data) + 1);
Packit Service fb6fa5
	      GlobalUnlock (hdata);
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	}
Packit Service fb6fa5
      else
Packit Service fb6fa5
	property = GDK_NONE;
Packit Service fb6fa5
Packit Service fb6fa5
      API_CALL (CloseClipboard, ());
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (selection == GDK_SELECTION_CLIPBOARD && target == _image_bmp)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      if (!API_CALL (OpenClipboard, (GDK_WINDOW_HWND (requestor))))
Packit Service fb6fa5
	return;
Packit Service fb6fa5
Packit Service fb6fa5
      if ((hdata = GetClipboardData (CF_DIB)) != NULL)
Packit Service fb6fa5
        {
Packit Service fb6fa5
          BITMAPINFOHEADER *bi;
Packit Service fb6fa5
Packit Service fb6fa5
          if ((bi = GlobalLock (hdata)) != NULL)
Packit Service fb6fa5
            {
Packit Service fb6fa5
	      /* Need to add a BMP file header so gdk-pixbuf can load
Packit Service fb6fa5
	       * it.
Packit Service fb6fa5
	       *
Packit Service fb6fa5
	       * If the data is from Mozilla Firefox or IE7, and
Packit Service fb6fa5
	       * starts with an "old fashioned" BITMAPINFOHEADER,
Packit Service fb6fa5
	       * i.e. with biSize==40, and biCompression == BI_RGB and
Packit Service fb6fa5
	       * biBitCount==32, we assume that the "extra" byte in
Packit Service fb6fa5
	       * each pixel in fact is alpha.
Packit Service fb6fa5
	       *
Packit Service fb6fa5
	       * The gdk-pixbuf bmp loader doesn't trust 32-bit BI_RGB
Packit Service fb6fa5
	       * bitmaps to in fact have alpha, so we have to convince
Packit Service fb6fa5
	       * it by changing the bitmap header to a version 5
Packit Service fb6fa5
	       * BI_BITFIELDS one with explicit alpha mask indicated.
Packit Service fb6fa5
	       *
Packit Service fb6fa5
	       * The RGB bytes that are in bitmaps on the clipboard
Packit Service fb6fa5
	       * originating from Firefox or IE7 seem to be
Packit Service fb6fa5
	       * premultiplied with alpha. The gdk-pixbuf bmp loader
Packit Service fb6fa5
	       * of course doesn't expect that, so we have to undo the
Packit Service fb6fa5
	       * premultiplication before feeding the bitmap to the
Packit Service fb6fa5
	       * bmp loader.
Packit Service fb6fa5
	       *
Packit Service fb6fa5
	       * Note that for some reason the bmp loader used to want
Packit Service fb6fa5
	       * the alpha bytes in its input to actually be
Packit Service fb6fa5
	       * 255-alpha, but here we assume that this has been
Packit Service fb6fa5
	       * fixed before this is committed.
Packit Service fb6fa5
	       */
Packit Service fb6fa5
              BITMAPFILEHEADER *bf;
Packit Service fb6fa5
	      gpointer data;
Packit Service fb6fa5
	      gint data_length = GlobalSize (hdata);
Packit Service fb6fa5
	      gint new_length;
Packit Service fb6fa5
	      gboolean make_dibv5 = FALSE;
Packit Service fb6fa5
Packit Service fb6fa5
	      GDK_NOTE (DND, g_print ("... CF_DIB: %d bytes\n", data_length));
Packit Service fb6fa5
Packit Service fb6fa5
	      if (bi->biSize == sizeof (BITMAPINFOHEADER) &&
Packit Service fb6fa5
		  bi->biPlanes == 1 &&
Packit Service fb6fa5
		  bi->biBitCount == 32 &&
Packit Service fb6fa5
		  bi->biCompression == BI_RGB &&
Packit Service fb6fa5
#if 0
Packit Service fb6fa5
		  /* Maybe check explicitly for Mozilla or IE7?
Packit Service fb6fa5
		   *
Packit Service fb6fa5
		   * If the clipboard format
Packit Service fb6fa5
		   * application/x-moz-nativeimage is present, that is
Packit Service fb6fa5
		   * a reliable indicator that the data is offered by
Packit Service fb6fa5
		   * Mozilla one would think. For IE7,
Packit Service fb6fa5
		   * UniformResourceLocatorW is presumably not that
Packit Service fb6fa5
		   * uniqie, so probably need to do some
Packit Service fb6fa5
		   * GetClipboardOwner(), GetWindowThreadProcessId(),
Packit Service fb6fa5
		   * OpenProcess(), GetModuleFileNameEx() dance to
Packit Service fb6fa5
		   * check?
Packit Service fb6fa5
		   */
Packit Service fb6fa5
		  (IsClipboardFormatAvailable
Packit Service fb6fa5
		   (RegisterClipboardFormat ("application/x-moz-nativeimage")) ||
Packit Service fb6fa5
		   IsClipboardFormatAvailable
Packit Service fb6fa5
		   (RegisterClipboardFormat ("UniformResourceLocatorW"))) &&
Packit Service fb6fa5
#endif
Packit Service fb6fa5
		  TRUE)
Packit Service fb6fa5
		{
Packit Service fb6fa5
		  /* We turn the BITMAPINFOHEADER into a
Packit Service fb6fa5
		   * BITMAPV5HEADER before feeding it to gdk-pixbuf.
Packit Service fb6fa5
		   */
Packit Service fb6fa5
		  new_length = (data_length +
Packit Service fb6fa5
				sizeof (BITMAPFILEHEADER) +
Packit Service fb6fa5
				(sizeof (BITMAPV5HEADER) - sizeof (BITMAPINFOHEADER)));
Packit Service fb6fa5
		  make_dibv5 = TRUE;
Packit Service fb6fa5
		}
Packit Service fb6fa5
	      else
Packit Service fb6fa5
		{
Packit Service fb6fa5
		  new_length = data_length + sizeof (BITMAPFILEHEADER);
Packit Service fb6fa5
		}
Packit Service fb6fa5
	      
Packit Service fb6fa5
              data = g_try_malloc (new_length);
Packit Service fb6fa5
Packit Service fb6fa5
              if (data)
Packit Service fb6fa5
                {
Packit Service fb6fa5
                  bf = (BITMAPFILEHEADER *)data;
Packit Service fb6fa5
                  bf->bfType = 0x4d42; /* "BM" */
Packit Service fb6fa5
                  bf->bfSize = new_length;
Packit Service fb6fa5
                  bf->bfReserved1 = 0;
Packit Service fb6fa5
                  bf->bfReserved2 = 0;
Packit Service fb6fa5
Packit Service fb6fa5
		  if (make_dibv5)
Packit Service fb6fa5
		    {
Packit Service fb6fa5
		      BITMAPV5HEADER *bV5 = (BITMAPV5HEADER *) ((char *) data + sizeof (BITMAPFILEHEADER));
Packit Service fb6fa5
		      guchar *p;
Packit Service fb6fa5
		      int i;
Packit Service fb6fa5
Packit Service fb6fa5
		      bV5->bV5Size = sizeof (BITMAPV5HEADER);
Packit Service fb6fa5
		      bV5->bV5Width = bi->biWidth;
Packit Service fb6fa5
		      bV5->bV5Height = bi->biHeight;
Packit Service fb6fa5
		      bV5->bV5Planes = 1;
Packit Service fb6fa5
		      bV5->bV5BitCount = 32;
Packit Service fb6fa5
		      bV5->bV5Compression = BI_BITFIELDS;
Packit Service fb6fa5
		      bV5->bV5SizeImage = 4 * bV5->bV5Width * ABS (bV5->bV5Height);
Packit Service fb6fa5
		      bV5->bV5XPelsPerMeter = bi->biXPelsPerMeter;
Packit Service fb6fa5
		      bV5->bV5YPelsPerMeter = bi->biYPelsPerMeter;
Packit Service fb6fa5
		      bV5->bV5ClrUsed = 0;
Packit Service fb6fa5
		      bV5->bV5ClrImportant = 0;
Packit Service fb6fa5
		      /* Now the added mask fields */
Packit Service fb6fa5
		      bV5->bV5RedMask   = 0x00ff0000;
Packit Service fb6fa5
		      bV5->bV5GreenMask = 0x0000ff00;
Packit Service fb6fa5
		      bV5->bV5BlueMask  = 0x000000ff;
Packit Service fb6fa5
		      bV5->bV5AlphaMask = 0xff000000;
Packit Service fb6fa5
		      ((char *) &bV5->bV5CSType)[3] = 's';
Packit Service fb6fa5
		      ((char *) &bV5->bV5CSType)[2] = 'R';
Packit Service fb6fa5
		      ((char *) &bV5->bV5CSType)[1] = 'G';
Packit Service fb6fa5
		      ((char *) &bV5->bV5CSType)[0] = 'B';
Packit Service fb6fa5
		      /* Ignore colorspace and profile fields */
Packit Service fb6fa5
		      bV5->bV5Intent = LCS_GM_GRAPHICS;
Packit Service fb6fa5
		      bV5->bV5Reserved = 0;
Packit Service fb6fa5
Packit Service fb6fa5
		      bf->bfOffBits = (sizeof (BITMAPFILEHEADER) +
Packit Service fb6fa5
				       bV5->bV5Size);
Packit Service fb6fa5
Packit Service fb6fa5
		      p = ((guchar *) data) + sizeof (BITMAPFILEHEADER) + sizeof (BITMAPV5HEADER);
Packit Service fb6fa5
		      memcpy (p, ((char *) bi) + bi->biSize,
Packit Service fb6fa5
			      data_length - sizeof (BITMAPINFOHEADER));
Packit Service fb6fa5
Packit Service fb6fa5
		      for (i = 0; i < bV5->bV5SizeImage/4; i++)
Packit Service fb6fa5
			{
Packit Service fb6fa5
			  if (p[3] != 0)
Packit Service fb6fa5
			    {
Packit Service fb6fa5
			      gdouble inverse_alpha = 255./p[3];
Packit Service fb6fa5
			      
Packit Service fb6fa5
			      p[0] = p[0] * inverse_alpha + 0.5;
Packit Service fb6fa5
			      p[1] = p[1] * inverse_alpha + 0.5;
Packit Service fb6fa5
			      p[2] = p[2] * inverse_alpha + 0.5;
Packit Service fb6fa5
			    }
Packit Service fb6fa5
Packit Service fb6fa5
			  p += 4;
Packit Service fb6fa5
			}
Packit Service fb6fa5
		    }
Packit Service fb6fa5
		  else
Packit Service fb6fa5
		    {
Packit Service fb6fa5
		      bf->bfOffBits = (sizeof (BITMAPFILEHEADER) +
Packit Service fb6fa5
				       bi->biSize +
Packit Service fb6fa5
				       bi->biClrUsed * sizeof (RGBQUAD));
Packit Service fb6fa5
Packit Service fb6fa5
		      if (bi->biCompression == BI_BITFIELDS && bi->biBitCount >= 16)
Packit Service fb6fa5
		        {
Packit Service fb6fa5
                          /* Screenshots taken with PrintScreen or
Packit Service fb6fa5
                           * Alt + PrintScreen are found on the clipboard in
Packit Service fb6fa5
                           * this format. In this case the BITMAPINFOHEADER is
Packit Service fb6fa5
                           * followed by three DWORD specifying the masks of the
Packit Service fb6fa5
                           * red green and blue components, so adjust the offset
Packit Service fb6fa5
                           * accordingly. */
Packit Service fb6fa5
		          bf->bfOffBits += (3 * sizeof (DWORD));
Packit Service fb6fa5
		        }
Packit Service fb6fa5
Packit Service fb6fa5
		      memcpy ((char *) data + sizeof (BITMAPFILEHEADER),
Packit Service fb6fa5
			      bi,
Packit Service fb6fa5
			      data_length);
Packit Service fb6fa5
		    }
Packit Service fb6fa5
Packit Service fb6fa5
	          selection_property_store (requestor, _image_bmp, 8,
Packit Service fb6fa5
					    data, new_length);
Packit Service fb6fa5
                }
Packit Service fb6fa5
	      GlobalUnlock (hdata);
Packit Service fb6fa5
            }
Packit Service fb6fa5
      }
Packit Service fb6fa5
Packit Service fb6fa5
      API_CALL (CloseClipboard, ());
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (selection == GDK_SELECTION_CLIPBOARD)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gchar *mapped_target_name;
Packit Service fb6fa5
      UINT fmt = 0;
Packit Service fb6fa5
Packit Service fb6fa5
      if (!API_CALL (OpenClipboard, (GDK_WINDOW_HWND (requestor))))
Packit Service fb6fa5
	return;
Packit Service fb6fa5
Packit Service fb6fa5
      mapped_target_name = get_mapped_gdk_atom_name (target);
Packit Service fb6fa5
Packit Service fb6fa5
      /* Check if it's available. We could simply call
Packit Service fb6fa5
       * GetClipboardData (RegisterClipboardFormat (targetname)), but
Packit Service fb6fa5
       * the global custom format ID space is limited,
Packit Service fb6fa5
       * (0xC000~0xFFFF), and we better not waste an format ID if we
Packit Service fb6fa5
       * are just a requestor.
Packit Service fb6fa5
       */
Packit Service fb6fa5
      for ( ; 0 != (fmt = EnumClipboardFormats (fmt)); )
Packit Service fb6fa5
        {
Packit Service fb6fa5
          char sFormat[80];
Packit Service fb6fa5
Packit Service fb6fa5
          if (GetClipboardFormatName (fmt, sFormat, 80) > 0 && 
Packit Service fb6fa5
              strcmp (sFormat, mapped_target_name) == 0)
Packit Service fb6fa5
            {
Packit Service fb6fa5
              if ((hdata = GetClipboardData (fmt)) != NULL)
Packit Service fb6fa5
	        {
Packit Service fb6fa5
	          /* Simply get it without conversion */
Packit Service fb6fa5
                  guchar *ptr;
Packit Service fb6fa5
                  gint length;
Packit Service fb6fa5
Packit Service fb6fa5
                  if ((ptr = GlobalLock (hdata)) != NULL)
Packit Service fb6fa5
                    {
Packit Service fb6fa5
                      length = GlobalSize (hdata);
Packit Service fb6fa5
	      
Packit Service fb6fa5
                      GDK_NOTE (DND, g_print ("... %s: %d bytes\n", mapped_target_name, length));
Packit Service fb6fa5
	      
Packit Service fb6fa5
                      selection_property_store (requestor, target, 8,
Packit Service fb6fa5
						g_memdup (ptr, length), length);
Packit Service fb6fa5
	              GlobalUnlock (hdata);
Packit Service fb6fa5
                      break;
Packit Service fb6fa5
                    }
Packit Service fb6fa5
                }
Packit Service fb6fa5
            }
Packit Service fb6fa5
        }
Packit Service fb6fa5
      g_free (mapped_target_name);
Packit Service fb6fa5
      API_CALL (CloseClipboard, ());
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (selection == _gdk_win32_dropfiles)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      /* This means he wants the names of the dropped files.
Packit Service fb6fa5
       * gdk_dropfiles_filter already has stored the text/uri-list
Packit Service fb6fa5
       * data temporarily in dropfiles_prop.
Packit Service fb6fa5
       */
Packit Service fb6fa5
      if (dropfiles_prop != NULL)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  selection_property_store
Packit Service fb6fa5
	    (requestor, dropfiles_prop->type, dropfiles_prop->format,
Packit Service fb6fa5
	     dropfiles_prop->data, dropfiles_prop->length);
Packit Service fb6fa5
	  g_free (dropfiles_prop);
Packit Service fb6fa5
	  dropfiles_prop = NULL;
Packit Service fb6fa5
	}
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else
Packit Service fb6fa5
    property = GDK_NONE;
Packit Service fb6fa5
Packit Service fb6fa5
  /* Generate a selection notify message so that we actually fetch the
Packit Service fb6fa5
   * data (if property == _gdk_selection) or indicating failure (if
Packit Service fb6fa5
   * property == GDK_NONE).
Packit Service fb6fa5
   */
Packit Service fb6fa5
  generate_selection_notify (requestor, selection, target, property, time);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
gint
Packit Service fb6fa5
gdk_selection_property_get (GdkWindow  *requestor,
Packit Service fb6fa5
			    guchar    **data,
Packit Service fb6fa5
			    GdkAtom    *ret_type,
Packit Service fb6fa5
			    gint       *ret_format)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GdkSelProp *prop;
Packit Service fb6fa5
Packit Service fb6fa5
  g_return_val_if_fail (requestor != NULL, 0);
Packit Service fb6fa5
  g_return_val_if_fail (GDK_IS_WINDOW (requestor), 0);
Packit Service fb6fa5
Packit Service fb6fa5
  if (GDK_WINDOW_DESTROYED (requestor))
Packit Service fb6fa5
    return 0;
Packit Service fb6fa5
  
Packit Service fb6fa5
  GDK_NOTE (DND, g_print ("gdk_selection_property_get: %p",
Packit Service fb6fa5
			   GDK_WINDOW_HWND (requestor)));
Packit Service fb6fa5
Packit Service fb6fa5
  prop = g_hash_table_lookup (sel_prop_table, GDK_WINDOW_HWND (requestor));
Packit Service fb6fa5
Packit Service fb6fa5
  if (prop == NULL)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      GDK_NOTE (DND, g_print (" (nothing)\n"));
Packit Service fb6fa5
      *data = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
      return 0;
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  *data = g_malloc (prop->length + 1);
Packit Service fb6fa5
  (*data)[prop->length] = '\0';
Packit Service fb6fa5
  if (prop->length > 0)
Packit Service fb6fa5
    memmove (*data, prop->data, prop->length);
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (DND, {
Packit Service fb6fa5
      gchar *type_name = gdk_atom_name (prop->type);
Packit Service fb6fa5
Packit Service fb6fa5
      g_print (" %s format:%d length:%d\n", type_name, prop->format, prop->length);
Packit Service fb6fa5
      g_free (type_name);
Packit Service fb6fa5
    });
Packit Service fb6fa5
Packit Service fb6fa5
  if (ret_type)
Packit Service fb6fa5
    *ret_type = prop->type;
Packit Service fb6fa5
Packit Service fb6fa5
  if (ret_format)
Packit Service fb6fa5
    *ret_format = prop->format;
Packit Service fb6fa5
Packit Service fb6fa5
  return prop->length;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
_gdk_selection_property_delete (GdkWindow *window)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GdkSelProp *prop;
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (DND, g_print ("_gdk_selection_property_delete: %p (no-op)\n",
Packit Service fb6fa5
			   GDK_WINDOW_HWND (window)));
Packit Service fb6fa5
Packit Service fb6fa5
#if 1 /* without this we can only paste the first image from clipboard */
Packit Service fb6fa5
  prop = g_hash_table_lookup (sel_prop_table, GDK_WINDOW_HWND (window));
Packit Service fb6fa5
  if (prop != NULL)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      g_free (prop->data);
Packit Service fb6fa5
      g_free (prop);
Packit Service fb6fa5
      g_hash_table_remove (sel_prop_table, GDK_WINDOW_HWND (window));
Packit Service fb6fa5
    }
Packit Service fb6fa5
#endif
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
gdk_selection_send_notify_for_display (GdkDisplay      *display,
Packit Service fb6fa5
                                       GdkNativeWindow  requestor,
Packit Service fb6fa5
                                       GdkAtom     	selection,
Packit Service fb6fa5
                                       GdkAtom     	target,
Packit Service fb6fa5
                                       GdkAtom     	property,
Packit Service fb6fa5
                                       guint32     	time)
Packit Service fb6fa5
{
Packit Service fb6fa5
  g_return_if_fail (display == _gdk_display);
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (DND, {
Packit Service fb6fa5
      gchar *sel_name = gdk_atom_name (selection);
Packit Service fb6fa5
      gchar *tgt_name = gdk_atom_name (target);
Packit Service fb6fa5
      gchar *prop_name = gdk_atom_name (property);
Packit Service fb6fa5
      
Packit Service fb6fa5
      g_print ("gdk_selection_send_notify_for_display: %p %s %s %s (no-op)\n",
Packit Service fb6fa5
	       requestor, sel_name, tgt_name, prop_name);
Packit Service fb6fa5
      g_free (sel_name);
Packit Service fb6fa5
      g_free (tgt_name);
Packit Service fb6fa5
      g_free (prop_name);
Packit Service fb6fa5
    });
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
/* It's hard to say whether implementing this actually is of any use
Packit Service fb6fa5
 * on the Win32 platform? gtk calls only
Packit Service fb6fa5
 * gdk_text_property_to_utf8_list_for_display().
Packit Service fb6fa5
 */
Packit Service fb6fa5
gint
Packit Service fb6fa5
gdk_text_property_to_text_list_for_display (GdkDisplay   *display,
Packit Service fb6fa5
					    GdkAtom       encoding,
Packit Service fb6fa5
					    gint          format, 
Packit Service fb6fa5
					    const guchar *text,
Packit Service fb6fa5
					    gint          length,
Packit Service fb6fa5
					    gchar      ***list)
Packit Service fb6fa5
{
Packit Service fb6fa5
  gchar *result;
Packit Service fb6fa5
  const gchar *charset;
Packit Service fb6fa5
  gchar *source_charset;
Packit Service fb6fa5
Packit Service fb6fa5
  g_return_val_if_fail (display == _gdk_display, 0);
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (DND, {
Packit Service fb6fa5
      gchar *enc_name = gdk_atom_name (encoding);
Packit Service fb6fa5
      
Packit Service fb6fa5
      g_print ("gdk_text_property_to_text_list_for_display: %s %d %.20s %d\n",
Packit Service fb6fa5
	       enc_name, format, text, length);
Packit Service fb6fa5
      g_free (enc_name);
Packit Service fb6fa5
    });
Packit Service fb6fa5
    
Packit Service fb6fa5
  if (!list)
Packit Service fb6fa5
    return 0;
Packit Service fb6fa5
Packit Service fb6fa5
  if (encoding == GDK_TARGET_STRING)
Packit Service fb6fa5
    source_charset = g_strdup ("ISO-8859-1");
Packit Service fb6fa5
  else if (encoding == _utf8_string)
Packit Service fb6fa5
    source_charset = g_strdup ("UTF-8");
Packit Service fb6fa5
  else
Packit Service fb6fa5
    source_charset = gdk_atom_name (encoding);
Packit Service fb6fa5
    
Packit Service fb6fa5
  g_get_charset (&charset);
Packit Service fb6fa5
Packit Service fb6fa5
  result = g_convert (text, length, charset, source_charset,
Packit Service fb6fa5
		      NULL, NULL, NULL);
Packit Service fb6fa5
  g_free (source_charset);
Packit Service fb6fa5
Packit Service fb6fa5
  if (!result)
Packit Service fb6fa5
    return 0;
Packit Service fb6fa5
Packit Service fb6fa5
  *list = g_new (gchar *, 1);
Packit Service fb6fa5
  **list = result;
Packit Service fb6fa5
  
Packit Service fb6fa5
  return 1;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
gdk_free_text_list (gchar **list)
Packit Service fb6fa5
{
Packit Service fb6fa5
  g_return_if_fail (list != NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  g_free (*list);
Packit Service fb6fa5
  g_free (list);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static gint
Packit Service fb6fa5
make_list (const gchar  *text,
Packit Service fb6fa5
	   gint          length,
Packit Service fb6fa5
	   gboolean      latin1,
Packit Service fb6fa5
	   gchar      ***list)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GSList *strings = NULL;
Packit Service fb6fa5
  gint n_strings = 0;
Packit Service fb6fa5
  gint i;
Packit Service fb6fa5
  const gchar *p = text;
Packit Service fb6fa5
  const gchar *q;
Packit Service fb6fa5
  GSList *tmp_list;
Packit Service fb6fa5
  GError *error = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
  while (p < text + length)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gchar *str;
Packit Service fb6fa5
      
Packit Service fb6fa5
      q = p;
Packit Service fb6fa5
      while (*q && q < text + length)
Packit Service fb6fa5
	q++;
Packit Service fb6fa5
Packit Service fb6fa5
      if (latin1)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  str = g_convert (p, q - p,
Packit Service fb6fa5
			   "UTF-8", "ISO-8859-1",
Packit Service fb6fa5
			   NULL, NULL, &error);
Packit Service fb6fa5
Packit Service fb6fa5
	  if (!str)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      g_warning ("Error converting selection from STRING: %s",
Packit Service fb6fa5
			 error->message);
Packit Service fb6fa5
	      g_error_free (error);
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	}
Packit Service fb6fa5
      else
Packit Service fb6fa5
	str = g_strndup (p, q - p);
Packit Service fb6fa5
Packit Service fb6fa5
      if (str)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  strings = g_slist_prepend (strings, str);
Packit Service fb6fa5
	  n_strings++;
Packit Service fb6fa5
	}
Packit Service fb6fa5
Packit Service fb6fa5
      p = q + 1;
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  if (list)
Packit Service fb6fa5
    *list = g_new (gchar *, n_strings + 1);
Packit Service fb6fa5
Packit Service fb6fa5
  (*list)[n_strings] = NULL;
Packit Service fb6fa5
  
Packit Service fb6fa5
  i = n_strings;
Packit Service fb6fa5
  tmp_list = strings;
Packit Service fb6fa5
  while (tmp_list)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      if (list)
Packit Service fb6fa5
	(*list)[--i] = tmp_list->data;
Packit Service fb6fa5
      else
Packit Service fb6fa5
	g_free (tmp_list->data);
Packit Service fb6fa5
Packit Service fb6fa5
      tmp_list = tmp_list->next;
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  g_slist_free (strings);
Packit Service fb6fa5
Packit Service fb6fa5
  return n_strings;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
gint 
Packit Service fb6fa5
gdk_text_property_to_utf8_list_for_display (GdkDisplay    *display,
Packit Service fb6fa5
                                            GdkAtom        encoding,
Packit Service fb6fa5
                                            gint           format,
Packit Service fb6fa5
                                            const guchar  *text,
Packit Service fb6fa5
                                            gint           length,
Packit Service fb6fa5
                                            gchar       ***list)
Packit Service fb6fa5
{
Packit Service fb6fa5
  g_return_val_if_fail (text != NULL, 0);
Packit Service fb6fa5
  g_return_val_if_fail (length >= 0, 0);
Packit Service fb6fa5
  g_return_val_if_fail (display == _gdk_display, 0);
Packit Service fb6fa5
Packit Service fb6fa5
  if (encoding == GDK_TARGET_STRING)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      return make_list ((gchar *)text, length, TRUE, list);
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else if (encoding == _utf8_string)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      return make_list ((gchar *)text, length, FALSE, list);
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gchar *enc_name = gdk_atom_name (encoding);
Packit Service fb6fa5
Packit Service fb6fa5
      g_warning ("gdk_text_property_to_utf8_list_for_display: encoding %s not handled\n", enc_name);
Packit Service fb6fa5
      g_free (enc_name);
Packit Service fb6fa5
Packit Service fb6fa5
      if (list)
Packit Service fb6fa5
	*list = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
      return 0;
Packit Service fb6fa5
    }
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
gint
Packit Service fb6fa5
gdk_string_to_compound_text_for_display (GdkDisplay  *display,
Packit Service fb6fa5
					 const gchar *str,
Packit Service fb6fa5
					 GdkAtom     *encoding,
Packit Service fb6fa5
					 gint        *format,
Packit Service fb6fa5
					 guchar     **ctext,
Packit Service fb6fa5
					 gint        *length)
Packit Service fb6fa5
{
Packit Service fb6fa5
  g_return_val_if_fail (str != NULL, 0);
Packit Service fb6fa5
  g_return_val_if_fail (length >= 0, 0);
Packit Service fb6fa5
  g_return_val_if_fail (display == _gdk_display, 0);
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (DND, g_print ("gdk_string_to_compound_text_for_display: %.20s\n", str));
Packit Service fb6fa5
Packit Service fb6fa5
  /* Always fail on Win32. No COMPOUND_TEXT support. */
Packit Service fb6fa5
Packit Service fb6fa5
  if (encoding)
Packit Service fb6fa5
    *encoding = GDK_NONE;
Packit Service fb6fa5
Packit Service fb6fa5
  if (format)
Packit Service fb6fa5
    *format = 0;
Packit Service fb6fa5
Packit Service fb6fa5
  if (ctext)
Packit Service fb6fa5
    *ctext = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
  if (length)
Packit Service fb6fa5
    *length = 0;
Packit Service fb6fa5
Packit Service fb6fa5
  return -1;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
gchar *
Packit Service fb6fa5
gdk_utf8_to_string_target (const gchar *str)
Packit Service fb6fa5
{
Packit Service fb6fa5
  return _gdk_utf8_to_string_target_internal (str, strlen (str));
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
gboolean
Packit Service fb6fa5
gdk_utf8_to_compound_text_for_display (GdkDisplay  *display,
Packit Service fb6fa5
                                       const gchar *str,
Packit Service fb6fa5
                                       GdkAtom     *encoding,
Packit Service fb6fa5
                                       gint        *format,
Packit Service fb6fa5
                                       guchar     **ctext,
Packit Service fb6fa5
                                       gint        *length)
Packit Service fb6fa5
{
Packit Service fb6fa5
  g_return_val_if_fail (str != NULL, FALSE);
Packit Service fb6fa5
  g_return_val_if_fail (display == _gdk_display, FALSE);
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (DND, g_print ("gdk_utf8_to_compound_text_for_display: %.20s\n", str));
Packit Service fb6fa5
Packit Service fb6fa5
  /* Always fail on Win32. No COMPOUND_TEXT support. */
Packit Service fb6fa5
Packit Service fb6fa5
  if (encoding)
Packit Service fb6fa5
    *encoding = GDK_NONE;
Packit Service fb6fa5
Packit Service fb6fa5
  if (format)
Packit Service fb6fa5
    *format = 0;
Packit Service fb6fa5
  
Packit Service fb6fa5
  if (ctext)
Packit Service fb6fa5
    *ctext = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
  if (length)
Packit Service fb6fa5
    *length = 0;
Packit Service fb6fa5
Packit Service fb6fa5
  return FALSE;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
gdk_free_compound_text (guchar *ctext)
Packit Service fb6fa5
{
Packit Service fb6fa5
  /* As we never generate anything claimed to be COMPOUND_TEXT, this
Packit Service fb6fa5
   * should never be called. Or if it is called, ctext should be the
Packit Service fb6fa5
   * NULL returned for conversions to COMPOUND_TEXT above.
Packit Service fb6fa5
   */
Packit Service fb6fa5
  g_return_if_fail (ctext == NULL);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
/* This function is called from gtk_selection_add_target() and
Packit Service fb6fa5
 * gtk_selection_add_targets() in gtkselection.c. It is this function
Packit Service fb6fa5
 * that takes care of setting those clipboard formats for which we use
Packit Service fb6fa5
 * delayed rendering. Formats copied directly to the clipboard are
Packit Service fb6fa5
 * handled in gdk_property_change() in gdkproperty-win32.c.
Packit Service fb6fa5
 */
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
gdk_win32_selection_add_targets (GdkWindow  *owner,
Packit Service fb6fa5
				 GdkAtom     selection,
Packit Service fb6fa5
				 gint	     n_targets,
Packit Service fb6fa5
				 GdkAtom    *targets)
Packit Service fb6fa5
{
Packit Service fb6fa5
  HWND hwnd = NULL;
Packit Service fb6fa5
  gboolean has_image = FALSE;
Packit Service fb6fa5
  gint i;
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (DND, {
Packit Service fb6fa5
      gchar *sel_name = gdk_atom_name (selection);
Packit Service fb6fa5
      
Packit Service fb6fa5
      g_print ("gdk_win32_selection_add_targets: %p: %s: ",
Packit Service fb6fa5
	       owner ? GDK_WINDOW_HWND (owner) : NULL,
Packit Service fb6fa5
	       sel_name);
Packit Service fb6fa5
      g_free (sel_name);
Packit Service fb6fa5
Packit Service fb6fa5
      for (i = 0; i < n_targets; i++)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  gchar *tgt_name = gdk_atom_name (targets[i]);
Packit Service fb6fa5
Packit Service fb6fa5
	  g_print ("%s", tgt_name);
Packit Service fb6fa5
	  g_free (tgt_name);
Packit Service fb6fa5
	  if (i < n_targets - 1)
Packit Service fb6fa5
	    g_print (", ");
Packit Service fb6fa5
	}
Packit Service fb6fa5
      g_print ("\n");
Packit Service fb6fa5
    });
Packit Service fb6fa5
Packit Service fb6fa5
  if (selection != GDK_SELECTION_CLIPBOARD)
Packit Service fb6fa5
    return;
Packit Service fb6fa5
Packit Service fb6fa5
  if (owner != NULL)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      if (GDK_WINDOW_DESTROYED (owner))
Packit Service fb6fa5
	return;
Packit Service fb6fa5
      hwnd = GDK_WINDOW_HWND (owner);
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  if (!API_CALL (OpenClipboard, (hwnd)))
Packit Service fb6fa5
    return;
Packit Service fb6fa5
Packit Service fb6fa5
  /* We have a very simple strategy: If some kind of pixmap image
Packit Service fb6fa5
   * format is being added, actually advertise just PNG and DIB. PNG
Packit Service fb6fa5
   * is our preferred format because it can losslessly represent any
Packit Service fb6fa5
   * image that gdk-pixbuf formats in general can, even with alpha,
Packit Service fb6fa5
   * unambiguously. CF_DIB is also advertised because of the general
Packit Service fb6fa5
   * support for it in Windows software, but note that alpha won't be
Packit Service fb6fa5
   * handled.
Packit Service fb6fa5
   */
Packit Service fb6fa5
  for (i = 0; !has_image && i < n_targets; ++i)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      UINT cf;
Packit Service fb6fa5
      gchar *target_name;
Packit Service fb6fa5
      int j;
Packit Service fb6fa5
      
Packit Service fb6fa5
      for (j = 0; j < n_known_pixbuf_formats; j++)
Packit Service fb6fa5
	if (targets[i] == known_pixbuf_formats[j])
Packit Service fb6fa5
	  {
Packit Service fb6fa5
	    if (!has_image)
Packit Service fb6fa5
	      {
Packit Service fb6fa5
		GDK_NOTE (DND, g_print ("... SetClipboardData(PNG,NULL)\n"));
Packit Service fb6fa5
		SetClipboardData (_cf_png, NULL);
Packit Service fb6fa5
Packit Service fb6fa5
		GDK_NOTE (DND, g_print ("... SetClipboardData(CF_DIB,NULL)\n"));
Packit Service fb6fa5
		SetClipboardData (CF_DIB, NULL);
Packit Service fb6fa5
Packit Service fb6fa5
		has_image = TRUE;
Packit Service fb6fa5
	      }
Packit Service fb6fa5
	    break;
Packit Service fb6fa5
	  }
Packit Service fb6fa5
      
Packit Service fb6fa5
      /* If it is one of the pixmap formats, already handled or not
Packit Service fb6fa5
       * needed.
Packit Service fb6fa5
       */
Packit Service fb6fa5
      if (j < n_known_pixbuf_formats)
Packit Service fb6fa5
	continue;
Packit Service fb6fa5
Packit Service fb6fa5
      /* We don't bother registering and advertising clipboard formats
Packit Service fb6fa5
       * that are X11 specific or no non-GTK+ apps will have ever
Packit Service fb6fa5
       * heard of, and when there are equivalent clipboard formats
Packit Service fb6fa5
       * that are commonly used.
Packit Service fb6fa5
       */
Packit Service fb6fa5
      if (targets[i] == _save_targets ||
Packit Service fb6fa5
	  targets[i] == _utf8_string ||
Packit Service fb6fa5
	  targets[i] == GDK_TARGET_STRING ||
Packit Service fb6fa5
	  targets[i] == _compound_text ||
Packit Service fb6fa5
	  targets[i] == _text ||
Packit Service fb6fa5
	  targets[i] == text_plain_charset_utf_8 ||
Packit Service fb6fa5
	  targets[i] == text_plain_charset_CP1252 ||
Packit Service fb6fa5
	  targets[i] == text_plain)
Packit Service fb6fa5
	continue;
Packit Service fb6fa5
Packit Service fb6fa5
      target_name = gdk_atom_name (targets[i]);
Packit Service fb6fa5
Packit Service fb6fa5
      if (g_str_has_prefix (target_name, "text/plain;charset="))
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  g_free (target_name);
Packit Service fb6fa5
	  continue;
Packit Service fb6fa5
	}
Packit Service fb6fa5
Packit Service fb6fa5
      cf = RegisterClipboardFormat (target_name);
Packit Service fb6fa5
Packit Service fb6fa5
      g_hash_table_replace (_format_atom_table,
Packit Service fb6fa5
			    GINT_TO_POINTER (cf),
Packit Service fb6fa5
			    targets[i]);
Packit Service fb6fa5
      
Packit Service fb6fa5
      GDK_NOTE (DND, g_print ("... SetClipboardData(%s,NULL)\n",
Packit Service fb6fa5
			      _gdk_win32_cf_to_string (cf)));
Packit Service fb6fa5
      SetClipboardData (cf, NULL);
Packit Service fb6fa5
Packit Service fb6fa5
      g_free (target_name);
Packit Service fb6fa5
    }
Packit Service fb6fa5
  API_CALL (CloseClipboard, ());
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
/* Convert from types such as "image/jpg" or "image/png" to DIB using
Packit Service fb6fa5
 * gdk-pixbuf so that image copied from GTK+ apps can be pasted in
Packit Service fb6fa5
 * native apps like mspaint.exe
Packit Service fb6fa5
 */
Packit Service fb6fa5
HGLOBAL
Packit Service fb6fa5
_gdk_win32_selection_convert_to_dib (HGLOBAL  hdata,
Packit Service fb6fa5
				     GdkAtom  target)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GDK_NOTE (DND, {
Packit Service fb6fa5
      gchar *target_name = gdk_atom_name (target);
Packit Service fb6fa5
Packit Service fb6fa5
      g_print ("_gdk_win32_selection_convert_to_dib: %p %s\n",
Packit Service fb6fa5
	       hdata, target_name);
Packit Service fb6fa5
      g_free (target_name);
Packit Service fb6fa5
    });
Packit Service fb6fa5
Packit Service fb6fa5
  if (target == _image_bmp)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      HGLOBAL hdatanew;
Packit Service fb6fa5
      SIZE_T size;
Packit Service fb6fa5
      guchar *ptr;
Packit Service fb6fa5
Packit Service fb6fa5
      g_return_val_if_fail (GlobalSize (hdata) >= sizeof (BITMAPFILEHEADER), NULL);
Packit Service fb6fa5
Packit Service fb6fa5
      /* No conversion is needed, just strip the BITMAPFILEHEADER */
Packit Service fb6fa5
      size = GlobalSize (hdata) - sizeof (BITMAPFILEHEADER);
Packit Service fb6fa5
      ptr = GlobalLock (hdata);
Packit Service fb6fa5
Packit Service fb6fa5
      memmove (ptr, ptr + sizeof (BITMAPFILEHEADER), size);
Packit Service fb6fa5
      GlobalUnlock (hdata);
Packit Service fb6fa5
Packit Service fb6fa5
      if ((hdatanew = GlobalReAlloc (hdata, size, GMEM_MOVEABLE)) == NULL)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  WIN32_API_FAILED ("GlobalReAlloc");
Packit Service fb6fa5
	  GlobalFree (hdata); /* The old hdata is not freed if error */
Packit Service fb6fa5
	}
Packit Service fb6fa5
      return hdatanew;
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  g_warning ("Should not happen: We provide some image format but not CF_DIB and CF_DIB is requested.");
Packit Service fb6fa5
Packit Service fb6fa5
  return NULL;
Packit Service fb6fa5
}