mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-04 09:40:19 +00:00
1d838f586c
There are sure regressions but basic stuff seems to be working again after all the API breakage done with comments like "Win32 and Quartz need to be ported still."
442 lines
14 KiB
C
442 lines
14 KiB
C
/* GDK - The GIMP Drawing Kit
|
|
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
|
* Copyright (C) 1998-2002 Tor Lillqvist
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
/*
|
|
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
|
|
* file for a list of people on the GTK+ Team. See the ChangeLog
|
|
* files for a list of changes. These files are distributed with
|
|
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <glib/gprintf.h>
|
|
|
|
#include "gdkscreen.h"
|
|
#include "gdkproperty.h"
|
|
#include "gdkselection.h"
|
|
#include "gdkprivate-win32.h"
|
|
#include "gdkwin32.h"
|
|
|
|
GdkAtom
|
|
_gdk_win32_display_manager_atom_intern (GdkDisplayManager *manager,
|
|
const gchar *atom_name,
|
|
gint only_if_exists)
|
|
{
|
|
ATOM win32_atom;
|
|
GdkAtom retval;
|
|
static GHashTable *atom_hash = NULL;
|
|
|
|
if (!atom_hash)
|
|
atom_hash = g_hash_table_new (g_str_hash, g_str_equal);
|
|
|
|
retval = g_hash_table_lookup (atom_hash, atom_name);
|
|
if (!retval)
|
|
{
|
|
if (strcmp (atom_name, "PRIMARY") == 0)
|
|
retval = GDK_SELECTION_PRIMARY;
|
|
else if (strcmp (atom_name, "SECONDARY") == 0)
|
|
retval = GDK_SELECTION_SECONDARY;
|
|
else if (strcmp (atom_name, "CLIPBOARD") == 0)
|
|
retval = GDK_SELECTION_CLIPBOARD;
|
|
else if (strcmp (atom_name, "ATOM") == 0)
|
|
retval = GDK_SELECTION_TYPE_ATOM;
|
|
else if (strcmp (atom_name, "BITMAP") == 0)
|
|
retval = GDK_SELECTION_TYPE_BITMAP;
|
|
else if (strcmp (atom_name, "COLORMAP") == 0)
|
|
retval = GDK_SELECTION_TYPE_COLORMAP;
|
|
else if (strcmp (atom_name, "DRAWABLE") == 0)
|
|
retval = GDK_SELECTION_TYPE_DRAWABLE;
|
|
else if (strcmp (atom_name, "INTEGER") == 0)
|
|
retval = GDK_SELECTION_TYPE_INTEGER;
|
|
else if (strcmp (atom_name, "PIXMAP") == 0)
|
|
retval = GDK_SELECTION_TYPE_PIXMAP;
|
|
else if (strcmp (atom_name, "WINDOW") == 0)
|
|
retval = GDK_SELECTION_TYPE_WINDOW;
|
|
else if (strcmp (atom_name, "STRING") == 0)
|
|
retval = GDK_SELECTION_TYPE_STRING;
|
|
else
|
|
{
|
|
win32_atom = GlobalAddAtom (atom_name);
|
|
retval = GUINT_TO_POINTER ((guint) win32_atom);
|
|
}
|
|
g_hash_table_insert (atom_hash,
|
|
g_strdup (atom_name),
|
|
retval);
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
|
|
gchar *
|
|
_gdk_win32_display_manager_get_atom_name (GdkDisplayManager *manager,
|
|
GdkAtom atom)
|
|
{
|
|
ATOM win32_atom;
|
|
gchar name[256];
|
|
|
|
if (GDK_NONE == atom) return g_strdup ("<none>");
|
|
else if (GDK_SELECTION_PRIMARY == atom) return g_strdup ("PRIMARY");
|
|
else if (GDK_SELECTION_SECONDARY == atom) return g_strdup ("SECONDARY");
|
|
else if (GDK_SELECTION_CLIPBOARD == atom) return g_strdup ("CLIPBOARD");
|
|
else if (GDK_SELECTION_TYPE_ATOM == atom) return g_strdup ("ATOM");
|
|
else if (GDK_SELECTION_TYPE_BITMAP == atom) return g_strdup ("BITMAP");
|
|
else if (GDK_SELECTION_TYPE_COLORMAP == atom) return g_strdup ("COLORMAP");
|
|
else if (GDK_SELECTION_TYPE_DRAWABLE == atom) return g_strdup ("DRAWABLE");
|
|
else if (GDK_SELECTION_TYPE_INTEGER == atom) return g_strdup ("INTEGER");
|
|
else if (GDK_SELECTION_TYPE_PIXMAP == atom) return g_strdup ("PIXMAP");
|
|
else if (GDK_SELECTION_TYPE_WINDOW == atom) return g_strdup ("WINDOW");
|
|
else if (GDK_SELECTION_TYPE_STRING == atom) return g_strdup ("STRING");
|
|
|
|
win32_atom = GPOINTER_TO_UINT (atom);
|
|
|
|
if (win32_atom < 0xC000)
|
|
return g_strdup_printf ("#%p", atom);
|
|
else if (GlobalGetAtomName (win32_atom, name, sizeof (name)) == 0)
|
|
return NULL;
|
|
return g_strdup (name);
|
|
}
|
|
|
|
gint
|
|
_gdk_win32_window_get_property (GdkWindow *window,
|
|
GdkAtom property,
|
|
GdkAtom type,
|
|
gulong offset,
|
|
gulong length,
|
|
gint pdelete,
|
|
GdkAtom *actual_property_type,
|
|
gint *actual_format_type,
|
|
gint *actual_length,
|
|
guchar **data)
|
|
{
|
|
g_return_val_if_fail (window != NULL, FALSE);
|
|
g_return_val_if_fail (GDK_IS_WINDOW (window), FALSE);
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return FALSE;
|
|
|
|
g_warning ("gdk_property_get: Not implemented");
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
_gdk_win32_window_change_property (GdkWindow *window,
|
|
GdkAtom property,
|
|
GdkAtom type,
|
|
gint format,
|
|
GdkPropMode mode,
|
|
const guchar *data,
|
|
gint nelements)
|
|
{
|
|
HGLOBAL hdata;
|
|
gint i, size;
|
|
guchar *ucptr, *buf = NULL;
|
|
wchar_t *wcptr, *p;
|
|
glong wclen;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
GDK_NOTE (DND, {
|
|
gchar *prop_name = gdk_atom_name (property);
|
|
gchar *type_name = gdk_atom_name (type);
|
|
|
|
g_print ("gdk_property_change: %p %s %s %s %d*%d bits: %s\n",
|
|
GDK_WINDOW_HWND (window),
|
|
prop_name,
|
|
type_name,
|
|
(mode == GDK_PROP_MODE_REPLACE ? "REPLACE" :
|
|
(mode == GDK_PROP_MODE_PREPEND ? "PREPEND" :
|
|
(mode == GDK_PROP_MODE_APPEND ? "APPEND" :
|
|
"???"))),
|
|
format, nelements,
|
|
_gdk_win32_data_to_string (data, MIN (10, format*nelements/8)));
|
|
g_free (prop_name);
|
|
g_free (type_name);
|
|
});
|
|
|
|
/* We should never come here for these types */
|
|
g_return_if_fail (type != GDK_TARGET_STRING);
|
|
g_return_if_fail (type != _text);
|
|
g_return_if_fail (type != _compound_text);
|
|
g_return_if_fail (type != _save_targets);
|
|
|
|
if (property == _gdk_selection &&
|
|
format == 8 &&
|
|
mode == GDK_PROP_MODE_REPLACE)
|
|
{
|
|
if (type == _utf8_string)
|
|
{
|
|
if (!OpenClipboard (GDK_WINDOW_HWND (window)))
|
|
{
|
|
WIN32_API_FAILED ("OpenClipboard");
|
|
return;
|
|
}
|
|
|
|
wcptr = g_utf8_to_utf16 ((char *) data, nelements, NULL, &wclen, NULL);
|
|
|
|
wclen++; /* Terminating 0 */
|
|
size = wclen * 2;
|
|
for (i = 0; i < wclen; i++)
|
|
if (wcptr[i] == '\n')
|
|
size += 2;
|
|
|
|
if (!(hdata = GlobalAlloc (GMEM_MOVEABLE, size)))
|
|
{
|
|
WIN32_API_FAILED ("GlobalAlloc");
|
|
if (!CloseClipboard ())
|
|
WIN32_API_FAILED ("CloseClipboard");
|
|
g_free (buf);
|
|
return;
|
|
}
|
|
|
|
ucptr = GlobalLock (hdata);
|
|
|
|
p = (wchar_t *) ucptr;
|
|
for (i = 0; i < wclen; i++)
|
|
{
|
|
if (wcptr[i] == '\n')
|
|
*p++ = '\r';
|
|
*p++ = wcptr[i];
|
|
}
|
|
g_free (wcptr);
|
|
|
|
GlobalUnlock (hdata);
|
|
GDK_NOTE (DND, g_print ("... SetClipboardData(CF_UNICODETEXT,%p)\n",
|
|
hdata));
|
|
if (!SetClipboardData (CF_UNICODETEXT, hdata))
|
|
WIN32_API_FAILED ("SetClipboardData");
|
|
|
|
if (!CloseClipboard ())
|
|
WIN32_API_FAILED ("CloseClipboard");
|
|
}
|
|
else
|
|
{
|
|
/* We use delayed rendering for everything else than
|
|
* text. We can't assign hdata to the clipboard here as type
|
|
* may be "image/png", "image/jpg", etc. In this case
|
|
* there's a further conversion afterwards.
|
|
*/
|
|
GDK_NOTE (DND, g_print ("... delayed rendering\n"));
|
|
_delayed_rendering_data = NULL;
|
|
if (!(hdata = GlobalAlloc (GMEM_MOVEABLE, nelements > 0 ? nelements : 1)))
|
|
{
|
|
WIN32_API_FAILED ("GlobalAlloc");
|
|
return;
|
|
}
|
|
ucptr = GlobalLock (hdata);
|
|
memcpy (ucptr, data, nelements);
|
|
GlobalUnlock (hdata);
|
|
_delayed_rendering_data = hdata;
|
|
}
|
|
}
|
|
else if (property == _gdk_ole2_dnd)
|
|
{
|
|
/* Will happen only if gdkdnd-win32.c has OLE2 dnd support compiled in */
|
|
_gdk_win32_ole2_dnd_property_change (type, format, data, nelements);
|
|
}
|
|
else
|
|
g_warning ("gdk_property_change: General case not implemented");
|
|
}
|
|
|
|
void
|
|
_gdk_win32_window_delete_property (GdkWindow *window,
|
|
GdkAtom property)
|
|
{
|
|
gchar *prop_name;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
GDK_NOTE (DND, {
|
|
prop_name = gdk_atom_name (property);
|
|
|
|
g_print ("gdk_property_delete: %p %s\n",
|
|
GDK_WINDOW_HWND (window),
|
|
prop_name);
|
|
g_free (prop_name);
|
|
});
|
|
|
|
if (property == _gdk_selection)
|
|
_gdk_selection_property_delete (window);
|
|
else if (property == _wm_transient_for)
|
|
gdk_window_set_transient_for (window, _gdk_root);
|
|
else
|
|
{
|
|
prop_name = gdk_atom_name (property);
|
|
g_warning ("gdk_property_delete: General case (%s) not implemented",
|
|
prop_name);
|
|
g_free (prop_name);
|
|
}
|
|
}
|
|
|
|
/*
|
|
For reference, from gdk/x11/gdksettings.c:
|
|
|
|
"Net/DoubleClickTime\0" "gtk-double-click-time\0"
|
|
"Net/DoubleClickDistance\0" "gtk-double-click-distance\0"
|
|
"Net/DndDragThreshold\0" "gtk-dnd-drag-threshold\0"
|
|
"Net/CursorBlink\0" "gtk-cursor-blink\0"
|
|
"Net/CursorBlinkTime\0" "gtk-cursor-blink-time\0"
|
|
"Net/ThemeName\0" "gtk-theme-name\0"
|
|
"Net/IconThemeName\0" "gtk-icon-theme-name\0"
|
|
"Gtk/CanChangeAccels\0" "gtk-can-change-accels\0"
|
|
"Gtk/ColorPalette\0" "gtk-color-palette\0"
|
|
"Gtk/FontName\0" "gtk-font-name\0"
|
|
"Gtk/IconSizes\0" "gtk-icon-sizes\0"
|
|
"Gtk/KeyThemeName\0" "gtk-key-theme-name\0"
|
|
"Gtk/ToolbarStyle\0" "gtk-toolbar-style\0"
|
|
"Gtk/ToolbarIconSize\0" "gtk-toolbar-icon-size\0"
|
|
"Gtk/IMPreeditStyle\0" "gtk-im-preedit-style\0"
|
|
"Gtk/IMStatusStyle\0" "gtk-im-status-style\0"
|
|
"Gtk/Modules\0" "gtk-modules\0"
|
|
"Gtk/FileChooserBackend\0" "gtk-file-chooser-backend\0"
|
|
"Gtk/ButtonImages\0" "gtk-button-images\0"
|
|
"Gtk/MenuImages\0" "gtk-menu-images\0"
|
|
"Gtk/MenuBarAccel\0" "gtk-menu-bar-accel\0"
|
|
"Gtk/CursorBlinkTimeout\0" "gtk-cursor-blink-timeout\0"
|
|
"Gtk/CursorThemeName\0" "gtk-cursor-theme-name\0"
|
|
"Gtk/CursorThemeSize\0" "gtk-cursor-theme-size\0"
|
|
"Gtk/ShowInputMethodMenu\0" "gtk-show-input-method-menu\0"
|
|
"Gtk/ShowUnicodeMenu\0" "gtk-show-unicode-menu\0"
|
|
"Gtk/TimeoutInitial\0" "gtk-timeout-initial\0"
|
|
"Gtk/TimeoutRepeat\0" "gtk-timeout-repeat\0"
|
|
"Gtk/ColorScheme\0" "gtk-color-scheme\0"
|
|
"Gtk/EnableAnimations\0" "gtk-enable-animations\0"
|
|
"Xft/Antialias\0" "gtk-xft-antialias\0"
|
|
"Xft/Hinting\0" "gtk-xft-hinting\0"
|
|
"Xft/HintStyle\0" "gtk-xft-hintstyle\0"
|
|
"Xft/RGBA\0" "gtk-xft-rgba\0"
|
|
"Xft/DPI\0" "gtk-xft-dpi\0"
|
|
"Net/FallbackIconTheme\0" "gtk-fallback-icon-theme\0"
|
|
"Gtk/TouchscreenMode\0" "gtk-touchscreen-mode\0"
|
|
"Gtk/EnableAccels\0" "gtk-enable-accels\0"
|
|
"Gtk/EnableMnemonics\0" "gtk-enable-mnemonics\0"
|
|
"Gtk/ScrolledWindowPlacement\0" "gtk-scrolled-window-placement\0"
|
|
"Gtk/IMModule\0" "gtk-im-module\0"
|
|
"Fontconfig/Timestamp\0" "gtk-fontconfig-timestamp\0"
|
|
"Net/SoundThemeName\0" "gtk-sound-theme-name\0"
|
|
"Net/EnableInputFeedbackSounds\0" "gtk-enable-input-feedback-sounds\0"
|
|
"Net/EnableEventSounds\0" "gtk-enable-event-sounds\0";
|
|
|
|
More, from various places in gtk sources:
|
|
|
|
gtk-entry-select-on-focus
|
|
gtk-split-cursor
|
|
|
|
*/
|
|
gboolean
|
|
_gdk_win32_screen_get_setting (GdkScreen *screen,
|
|
const gchar *name,
|
|
GValue *value)
|
|
{
|
|
g_return_val_if_fail (GDK_IS_SCREEN (screen), FALSE);
|
|
|
|
/*
|
|
* XXX : if these values get changed through the Windoze UI the
|
|
* respective gdk_events are not generated yet.
|
|
*/
|
|
if (strcmp ("gtk-theme-name", name) == 0)
|
|
{
|
|
g_value_set_string (value, "ms-windows");
|
|
}
|
|
else if (strcmp ("gtk-double-click-time", name) == 0)
|
|
{
|
|
gint i = GetDoubleClickTime ();
|
|
GDK_NOTE(MISC, g_print("gdk_screen_get_setting(\"%s\") : %d\n", name, i));
|
|
g_value_set_int (value, i);
|
|
return TRUE;
|
|
}
|
|
else if (strcmp ("gtk-double-click-distance", name) == 0)
|
|
{
|
|
gint i = MAX(GetSystemMetrics (SM_CXDOUBLECLK), GetSystemMetrics (SM_CYDOUBLECLK));
|
|
GDK_NOTE(MISC, g_print("gdk_screen_get_setting(\"%s\") : %d\n", name, i));
|
|
g_value_set_int (value, i);
|
|
return TRUE;
|
|
}
|
|
else if (strcmp ("gtk-dnd-drag-threshold", name) == 0)
|
|
{
|
|
gint i = MAX(GetSystemMetrics (SM_CXDRAG), GetSystemMetrics (SM_CYDRAG));
|
|
GDK_NOTE(MISC, g_print("gdk_screen_get_setting(\"%s\") : %d\n", name, i));
|
|
g_value_set_int (value, i);
|
|
return TRUE;
|
|
}
|
|
else if (strcmp ("gtk-split-cursor", name) == 0)
|
|
{
|
|
GDK_NOTE(MISC, g_print("gdk_screen_get_setting(\"%s\") : FALSE\n", name));
|
|
g_value_set_boolean (value, FALSE);
|
|
return TRUE;
|
|
}
|
|
else if (strcmp ("gtk-alternative-button-order", name) == 0)
|
|
{
|
|
GDK_NOTE(MISC, g_print("gdk_screen_get_setting(\"%s\") : TRUE\n", name));
|
|
g_value_set_boolean (value, TRUE);
|
|
return TRUE;
|
|
}
|
|
else if (strcmp ("gtk-alternative-sort-arrows", name) == 0)
|
|
{
|
|
GDK_NOTE(MISC, g_print("gdk_screen_get_setting(\"%s\") : TRUE\n", name));
|
|
g_value_set_boolean (value, TRUE);
|
|
return TRUE;
|
|
}
|
|
#if 0
|
|
/*
|
|
* With 'MS Sans Serif' as windows menu font (default on win98se) you'll get a
|
|
* bunch of :
|
|
* WARNING **: Couldn't load font "MS Sans Serif 8" falling back to "Sans 8"
|
|
* at least with testfilechooser (regardless of the bitmap check below)
|
|
* so just disabling this code seems to be the best we can do --hb
|
|
*/
|
|
else if (strcmp ("gtk-font-name", name) == 0)
|
|
{
|
|
NONCLIENTMETRICS ncm;
|
|
ncm.cbSize = sizeof(NONCLIENTMETRICS);
|
|
if (SystemParametersInfo (SPI_GETNONCLIENTMETRICS, ncm.cbSize, &ncm, FALSE))
|
|
{
|
|
/* Pango finally uses GetDeviceCaps to scale, we use simple
|
|
* approximation here.
|
|
*/
|
|
int nHeight = (0 > ncm.lfMenuFont.lfHeight ? -3*ncm.lfMenuFont.lfHeight/4 : 10);
|
|
if (OUT_STRING_PRECIS == ncm.lfMenuFont.lfOutPrecision)
|
|
GDK_NOTE(MISC, g_print("gdk_screen_get_setting(%s) : ignoring bitmap font '%s'\n",
|
|
name, ncm.lfMenuFont.lfFaceName));
|
|
else if (ncm.lfMenuFont.lfFaceName && strlen(ncm.lfMenuFont.lfFaceName) > 0 &&
|
|
/* Avoid issues like those described in bug #135098 */
|
|
g_utf8_validate (ncm.lfMenuFont.lfFaceName, -1, NULL))
|
|
{
|
|
char* s = g_strdup_printf ("%s %d", ncm.lfMenuFont.lfFaceName, nHeight);
|
|
GDK_NOTE(MISC, g_print("gdk_screen_get_setting(%s) : %s\n", name, s));
|
|
g_value_set_string (value, s);
|
|
|
|
g_free(s);
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return FALSE;
|
|
}
|