diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-19 03:13:10 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-19 03:13:10 +0000 |
commit | 3c57dd931145d43f2b0aef96c4d178135956bf91 (patch) | |
tree | 3de698981e9f0cc2c4f9569b19a5f3595e741f6b /modules/gimpinputdevicestore-dx.c | |
parent | Initial commit. (diff) | |
download | gimp-3c57dd931145d43f2b0aef96c4d178135956bf91.tar.xz gimp-3c57dd931145d43f2b0aef96c4d178135956bf91.zip |
Adding upstream version 2.10.36.upstream/2.10.36
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'modules/gimpinputdevicestore-dx.c')
-rw-r--r-- | modules/gimpinputdevicestore-dx.c | 482 |
1 files changed, 482 insertions, 0 deletions
diff --git a/modules/gimpinputdevicestore-dx.c b/modules/gimpinputdevicestore-dx.c new file mode 100644 index 0000000..66ca6f2 --- /dev/null +++ b/modules/gimpinputdevicestore-dx.c @@ -0,0 +1,482 @@ +/* GIMP - The GNU Image Manipulation Program + * Copyright (C) 1995 Spencer Kimball and Peter Mattis + * + * gimpinputdevicestore-dx.c + * Input device store based on DirectX. + * Copyright (C) 2007 Sven Neumann <sven@gimp.org> + * Copyright (C) 2007 Tor Lillqvist <tml@novell.com> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 3 of the License, or + * (at your option) any later version. + * + * This program 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 General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <https://www.gnu.org/licenses/>. + */ + +#include "config.h" + +#include <string.h> + +#include <gtk/gtk.h> + +#ifdef HAVE_DX_DINPUT +#define _WIN32_WINNT 0x0501 +#include <windows.h> +#define DIRECTINPUT_VERSION 0x0800 +#include <dinput.h> +#include <rpc.h> + +#ifndef GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS +#define GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS 0x00000004 +#endif + +#include <gdk/gdkwin32.h> + +#include "libgimpmodule/gimpmodule.h" + +#include "gimpinputdevicestore.h" + + +enum +{ + COLUMN_GUID, + COLUMN_LABEL, + COLUMN_IDEVICE, + NUM_COLUMNS +}; + +enum +{ + DEVICE_ADDED, + DEVICE_REMOVED, + LAST_SIGNAL +}; + +typedef struct _GimpInputDeviceStoreClass GimpInputDeviceStoreClass; + +struct _GimpInputDeviceStore +{ + GtkListStore parent_instance; + + GdkWindow *window; + + LPDIRECTINPUT8W directinput8; + + GError *error; +}; + + +struct _GimpInputDeviceStoreClass +{ + GtkListStoreClass parent_class; + + void (* device_added) (GimpInputDeviceStore *store, + const gchar *udi); + void (* device_removed) (GimpInputDeviceStore *store, + const gchar *udi); +}; + + +static void gimp_input_device_store_finalize (GObject *object); + +static gboolean gimp_input_device_store_add (GimpInputDeviceStore *store, + const GUID *guid); +static gboolean gimp_input_device_store_remove (GimpInputDeviceStore *store, + const gchar *udi); + + +G_DEFINE_DYNAMIC_TYPE (GimpInputDeviceStore, gimp_input_device_store, + GTK_TYPE_LIST_STORE) + +static guint store_signals[LAST_SIGNAL] = { 0 }; + + +void +gimp_input_device_store_register_types (GTypeModule *module) +{ + gimp_input_device_store_register_type (module); +} + +static void +gimp_input_device_store_class_init (GimpInputDeviceStoreClass *klass) +{ + GObjectClass *object_class = G_OBJECT_CLASS (klass); + + store_signals[DEVICE_ADDED] = + g_signal_new ("device-added", + G_TYPE_FROM_CLASS (klass), + G_SIGNAL_RUN_FIRST, + G_STRUCT_OFFSET (GimpInputDeviceStoreClass, device_added), + NULL, NULL, + g_cclosure_marshal_VOID__STRING, + G_TYPE_NONE, 1, G_TYPE_STRING); + + store_signals[DEVICE_REMOVED] = + g_signal_new ("device-removed", + G_TYPE_FROM_CLASS (klass), + G_SIGNAL_RUN_FIRST, + G_STRUCT_OFFSET (GimpInputDeviceStoreClass, device_removed), + NULL, NULL, + g_cclosure_marshal_VOID__STRING, + G_TYPE_NONE, 1, G_TYPE_STRING); + + object_class->finalize = gimp_input_device_store_finalize; + + klass->device_added = NULL; + klass->device_removed = NULL; +} + +static void +gimp_input_device_store_class_finalize (GimpInputDeviceStoreClass *klass) +{ +} + +static GdkFilterReturn +aux_window_filter (GdkXEvent *xevent, + GdkEvent *event, + gpointer data) +{ +#if 0 + GimpInputDeviceStore *store = (GimpInputDeviceStore *) data; + const MSG *msg = (MSG *) xevent; + + /* Look for deviced being added or removed */ + switch (msg->message) + { + } +#endif + + return GDK_FILTER_REMOVE; +} + +static GdkWindow * +create_aux_window (GimpInputDeviceStore *store) +{ + GdkWindowAttr wa; + GdkWindow *retval; + + /* Create a dummy window to be associated with DirectInput devices */ + wa.wclass = GDK_INPUT_OUTPUT; + wa.event_mask = GDK_ALL_EVENTS_MASK; + wa.width = 2; + wa.height = 2; + wa.x = -100; + wa.y = -100; + wa.window_type = GDK_WINDOW_TOPLEVEL; + if ((retval = gdk_window_new (NULL, &wa, GDK_WA_X|GDK_WA_Y)) == NULL) + return NULL; + g_object_ref (retval); + + gdk_window_add_filter (retval, aux_window_filter, store); + + return retval; +} + +static BOOL CALLBACK +enum_devices (const DIDEVICEINSTANCEW *di, + void *user_data) +{ + GimpInputDeviceStore *store = (GimpInputDeviceStore *) user_data; + + gimp_input_device_store_add (store, &di->guidInstance); + + return DIENUM_CONTINUE; +} + +static void +gimp_input_device_store_init (GimpInputDeviceStore *store) +{ + GType types[] = { G_TYPE_STRING, G_TYPE_STRING, G_TYPE_POINTER }; + HRESULT hresult; + HMODULE thismodule; + HMODULE dinput8; + + typedef HRESULT (WINAPI *t_DirectInput8Create) (HINSTANCE, DWORD, REFIID, LPVOID *, LPUNKNOWN); + t_DirectInput8Create p_DirectInput8Create; + + g_assert (G_N_ELEMENTS (types) == NUM_COLUMNS); + + gtk_list_store_set_column_types (GTK_LIST_STORE (store), + G_N_ELEMENTS (types), types); + + if (!GetModuleHandleEx (GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS, + (LPCTSTR) &gimp_input_device_store_init, + &thismodule)) + return; + + if ((store->window = create_aux_window (store)) == NULL) + { + g_set_error_literal (&store->error, GIMP_MODULE_ERROR, GIMP_MODULE_FAILED, + "Could not create aux window"); + return; + } + + if ((dinput8 = LoadLibrary ("dinput8.dll")) == NULL) + { + g_set_error_literal (&store->error, GIMP_MODULE_ERROR, GIMP_MODULE_FAILED, + "Could not load dinput8.dll"); + return; + } + + if ((p_DirectInput8Create = (t_DirectInput8Create) GetProcAddress (dinput8, "DirectInput8Create")) == NULL) + { + g_set_error_literal (&store->error, GIMP_MODULE_ERROR, GIMP_MODULE_FAILED, + "Could not find DirectInput8Create in dinput8.dll"); + return; + } + + if (FAILED ((hresult = (*p_DirectInput8Create) (thismodule, + DIRECTINPUT_VERSION, + &IID_IDirectInput8W, + (LPVOID *) &store->directinput8, + NULL)))) + { + g_set_error (&store->error, GIMP_MODULE_ERROR, GIMP_MODULE_FAILED, + "DirectInput8Create failed: %s", + g_win32_error_message (hresult)); + return; + } + + if (FAILED ((hresult = IDirectInput8_EnumDevices (store->directinput8, + DI8DEVCLASS_GAMECTRL, + enum_devices, + store, + DIEDFL_ATTACHEDONLY)))) + { + g_set_error (&store->error, GIMP_MODULE_ERROR, GIMP_MODULE_FAILED, + "IDirectInput8::EnumDevices failed: %s", + g_win32_error_message (hresult)); + return; + } +} + +static void +gimp_input_device_store_finalize (GObject *object) +{ + GimpInputDeviceStore *store = GIMP_INPUT_DEVICE_STORE (object); + + if (store->directinput8) + { + IDirectInput8_Release (store->directinput8); + store->directinput8 = NULL; + } + + if (store->error) + { + g_error_free (store->error); + store->error = NULL; + } + + G_OBJECT_CLASS (gimp_input_device_store_parent_class)->finalize (object); +} + +static gboolean +gimp_input_device_store_lookup (GimpInputDeviceStore *store, + const gchar *guid, + GtkTreeIter *iter) +{ + GtkTreeModel *model = GTK_TREE_MODEL (store); + GValue value = G_VALUE_INIT; + gboolean iter_valid; + + for (iter_valid = gtk_tree_model_get_iter_first (model, iter); + iter_valid; + iter_valid = gtk_tree_model_iter_next (model, iter)) + { + const gchar *str; + + gtk_tree_model_get_value (model, iter, COLUMN_GUID, &value); + + str = g_value_get_string (&value); + + if (strcmp (str, guid) == 0) + { + g_value_unset (&value); + break; + } + + g_value_unset (&value); + } + + return iter_valid; +} + +/* insert in alphabetic order */ +static void +gimp_input_device_store_insert (GimpInputDeviceStore *store, + const gchar *guid, + const gchar *label, + LPDIRECTINPUTDEVICE8W didevice8) +{ + GtkTreeModel *model = GTK_TREE_MODEL (store); + GtkTreeIter iter; + GValue value = G_VALUE_INIT; + gint pos = 0; + gboolean iter_valid; + + for (iter_valid = gtk_tree_model_get_iter_first (model, &iter); + iter_valid; + iter_valid = gtk_tree_model_iter_next (model, &iter), pos++) + { + const gchar *str; + + gtk_tree_model_get_value (model, &iter, COLUMN_LABEL, &value); + + str = g_value_get_string (&value); + + if (g_utf8_collate (label, str) < 0) + { + g_value_unset (&value); + break; + } + + g_value_unset (&value); + } + + gtk_list_store_insert_with_values (GTK_LIST_STORE (store), &iter, pos, + COLUMN_GUID, guid, + COLUMN_LABEL, label, + COLUMN_IDEVICE, didevice8, + -1); +} + +static gboolean +gimp_input_device_store_add (GimpInputDeviceStore *store, + const GUID *guid) +{ + HRESULT hresult; + LPDIRECTINPUTDEVICE8W didevice8; + DIDEVICEINSTANCEW di; + gboolean added = FALSE; + unsigned char *s; + gchar *guidstring; + gchar *name; + + if (UuidToString (guid, &s) != S_OK) + return FALSE; + guidstring = g_strdup (s); + RpcStringFree (&s); + + if (FAILED ((hresult = IDirectInput8_CreateDevice (store->directinput8, + guid, + &didevice8, + NULL)))) + { + g_free (guidstring); + return FALSE; + } + + if (FAILED ((hresult = IDirectInputDevice8_SetCooperativeLevel (didevice8, + (HWND) gdk_win32_drawable_get_handle (store->window), + DISCL_NONEXCLUSIVE | DISCL_BACKGROUND)))) + { + g_warning ("IDirectInputDevice8::SetCooperativeLevel failed: %s", + g_win32_error_message (hresult)); + g_free (guidstring); + return FALSE; + } + + di.dwSize = sizeof (DIDEVICEINSTANCEW); + if (FAILED ((hresult = IDirectInputDevice8_GetDeviceInfo (didevice8, + &di)))) + { + g_warning ("IDirectInputDevice8::GetDeviceInfo failed: %s", + g_win32_error_message (hresult)); + g_free (guidstring); + return FALSE; + } + + name = g_utf16_to_utf8 (di.tszInstanceName, -1, NULL, NULL, NULL); + gimp_input_device_store_insert (store, guidstring, name, didevice8); + + return added; +} + +static gboolean +gimp_input_device_store_remove (GimpInputDeviceStore *store, + const gchar *guid) +{ + GtkTreeIter iter; + + if (gimp_input_device_store_lookup (store, guid, &iter)) + { + gtk_list_store_remove (GTK_LIST_STORE (store), &iter); + return TRUE; + } + + return FALSE; +} + +#if 0 + +static void +gimp_input_device_store_device_added (LibHalContext *ctx, + const char *guid) +{ + GimpInputDeviceStore *store = libhal_ctx_get_user_data (ctx); + + if (gimp_input_device_store_add (store, udi)) + { + g_signal_emit (store, store_signals[DEVICE_ADDED], 0, udi); + } +} + +static void +gimp_input_device_store_device_removed (LibHalContext *ctx, + const char *udi) +{ + GimpInputDeviceStore *store = libhal_ctx_get_user_data (ctx); + + if (gimp_input_device_store_remove (store, udi)) + { + g_signal_emit (store, store_signals[DEVICE_REMOVED], 0, udi); + } +} + +#endif + +GimpInputDeviceStore * +gimp_input_device_store_new (void) +{ + return g_object_new (GIMP_TYPE_INPUT_DEVICE_STORE, NULL); +} + +gchar * +gimp_input_device_store_get_device_file (GimpInputDeviceStore *store, + const gchar *udi) +{ + GtkTreeIter iter; + GValue value = G_VALUE_INIT; + + g_return_val_if_fail (GIMP_IS_INPUT_DEVICE_STORE (store), NULL); + g_return_val_if_fail (udi != NULL, NULL); + + if (! store->directinput8) + return NULL; + + if (gimp_input_device_store_lookup (store, udi, &iter)) + { + gtk_tree_model_get_value (GTK_TREE_MODEL (store), + &iter, COLUMN_IDEVICE, &value); + return g_value_get_pointer (&value); + } + + return NULL; +} + +GError * +gimp_input_device_store_get_error (GimpInputDeviceStore *store) +{ + g_return_val_if_fail (GIMP_IS_INPUT_DEVICE_STORE (store), NULL); + + return store->error ? g_error_copy (store->error) : NULL; +} + +#endif /* HAVE_DX_DINPUT */ |