]> Pileus Git - ~andy/gtk/blob - gdk/win32/gdkmain-win32.c
x11: Use gdk_display_flush()
[~andy/gtk] / gdk / win32 / gdkmain-win32.c
1 /* GDK - The GIMP Drawing Kit
2  * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
3  * Copyright (C) 1998-2002 Tor Lillqvist
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library. If not, see <http://www.gnu.org/licenses/>.
17  */
18
19 /*
20  * Modified by the GTK+ Team and others 1997-2000.  See the AUTHORS
21  * file for a list of people on the GTK+ Team.  See the ChangeLog
22  * files for a list of changes.  These files are distributed with
23  * GTK+ at ftp://ftp.gtk.org/pub/gtk/. 
24  */
25
26 #include "config.h"
27
28 #include <glib/gprintf.h>
29 #include <stdlib.h>
30 #include <string.h>
31 #include <limits.h>
32 #include <io.h>
33
34 #include "gdk.h"
35 #include "gdkkeysyms.h"
36 #include "gdkinternals.h"
37 #include "gdkintl.h"
38 #include "gdkprivate-win32.h"
39 #include "gdkwin32.h"
40
41 #include <objbase.h>
42
43 #include <windows.h>
44 #include <wintab.h>
45 #include <imm.h>
46
47 static gboolean gdk_synchronize = FALSE;
48
49 static gboolean dummy;
50
51 const GOptionEntry _gdk_windowing_args[] = {
52   { "sync", 0, 0, G_OPTION_ARG_NONE, &gdk_synchronize, 
53     /* Description of --sync in --help output */              N_("Don't batch GDI requests"), NULL },
54   { "no-wintab", 0, 0, G_OPTION_ARG_NONE, &_gdk_input_ignore_wintab, 
55     /* Description of --no-wintab in --help output */         N_("Don't use the Wintab API for tablet support"), NULL },
56   { "ignore-wintab", 0, 0, G_OPTION_ARG_NONE, &_gdk_input_ignore_wintab, 
57     /* Description of --ignore-wintab in --help output */     N_("Same as --no-wintab"), NULL },
58   { "use-wintab", 0, 0, G_OPTION_ARG_NONE, &dummy,
59     /* Description of --use-wintab in --help output */     N_("Do use the Wintab API [default]"), NULL },
60   { "max-colors", 0, 0, G_OPTION_ARG_INT, &_gdk_max_colors, 
61     /* Description of --max-colors=COLORS in --help output */ N_("Size of the palette in 8 bit mode"), 
62     /* Placeholder in --max-colors=COLORS in --help output */ N_("COLORS") },
63   { NULL }
64 };
65
66 BOOL WINAPI
67 DllMain (HINSTANCE hinstDLL,
68          DWORD     dwReason,
69          LPVOID    reserved)
70 {
71   _gdk_dll_hinstance = hinstDLL;
72
73   return TRUE;
74 }
75
76 void
77 _gdk_win32_windowing_init (void)
78 {
79   gchar buf[10];
80
81   if (getenv ("GDK_IGNORE_WINTAB") != NULL)
82     _gdk_input_ignore_wintab = TRUE;
83   else if (getenv ("GDK_USE_WINTAB") != NULL)
84     _gdk_input_ignore_wintab = FALSE;
85
86   if (gdk_synchronize)
87     GdiSetBatchLimit (1);
88
89   _gdk_app_hmodule = GetModuleHandle (NULL);
90   _gdk_display_hdc = CreateDC ("DISPLAY", NULL, NULL, NULL);
91   _gdk_input_locale = GetKeyboardLayout (0);
92   _gdk_input_locale_is_ime = ImmIsIME (_gdk_input_locale);
93   GetLocaleInfo (MAKELCID (LOWORD (_gdk_input_locale), SORT_DEFAULT),
94                  LOCALE_IDEFAULTANSICODEPAGE,
95                  buf, sizeof (buf));
96   _gdk_input_codepage = atoi (buf);
97   GDK_NOTE (EVENTS, g_print ("input_locale:%p, codepage:%d\n",
98                              _gdk_input_locale, _gdk_input_codepage));
99
100   CoInitialize (NULL);
101
102   _gdk_selection = gdk_atom_intern_static_string ("GDK_SELECTION");
103   _wm_transient_for = gdk_atom_intern_static_string ("WM_TRANSIENT_FOR");
104   _targets = gdk_atom_intern_static_string ("TARGETS");
105   _delete = gdk_atom_intern_static_string ("DELETE");
106   _save_targets = gdk_atom_intern_static_string ("SAVE_TARGETS");
107   _utf8_string = gdk_atom_intern_static_string ("UTF8_STRING");
108   _text = gdk_atom_intern_static_string ("TEXT");
109   _compound_text = gdk_atom_intern_static_string ("COMPOUND_TEXT");
110   _text_uri_list = gdk_atom_intern_static_string ("text/uri-list");
111   _text_html = gdk_atom_intern_static_string ("text/html");
112   _image_png = gdk_atom_intern_static_string ("image/png");
113   _image_jpeg = gdk_atom_intern_static_string ("image/jpeg");
114   _image_bmp = gdk_atom_intern_static_string ("image/bmp");
115   _image_gif = gdk_atom_intern_static_string ("image/gif");
116
117   _local_dnd = gdk_atom_intern_static_string ("LocalDndSelection");
118   _gdk_win32_dropfiles = gdk_atom_intern_static_string ("DROPFILES_DND");
119   _gdk_ole2_dnd = gdk_atom_intern_static_string ("OLE2_DND");
120
121   /* MS Office 2007, at least, offers images in common file formats
122    * using clipboard format names like "PNG" and "JFIF". So we follow
123    * the lead and map the GDK target name "image/png" to the clipboard
124    * format name "PNG" etc.
125    */
126   _cf_png = RegisterClipboardFormat ("PNG");
127   _cf_jfif = RegisterClipboardFormat ("JFIF");
128   _cf_gif = RegisterClipboardFormat ("GIF");
129
130   _cf_url = RegisterClipboardFormat ("UniformResourceLocatorW");
131   _cf_html_format = RegisterClipboardFormat ("HTML Format");
132   _cf_text_html = RegisterClipboardFormat ("text/html");
133
134   _gdk_win32_selection_init ();
135 }
136
137 void
138 _gdk_win32_api_failed (const gchar *where,
139                       const gchar *api)
140 {
141   gchar *msg = g_win32_error_message (GetLastError ());
142   g_warning ("%s: %s failed: %s", where, api, msg);
143   g_free (msg);
144 }
145
146 void
147 _gdk_other_api_failed (const gchar *where,
148                       const gchar *api)
149 {
150   g_warning ("%s: %s failed", where, api);
151 }
152
153
154 #ifdef G_ENABLE_DEBUG
155
156 /*
157  * Like g_strdup_printf, but to a static buffer. Return value does not
158  * have to be g_free()d. The buffer is of bounded size and reused
159  * cyclically. Thus the return value is valid only until that part of
160  * the buffer happens to get reused. This doesn't matter as this
161  * function's return value is used in debugging output right after the call,
162  * and the return value isn't used after that.
163  */
164 static gchar *
165 static_printf (const gchar *format,
166                ...)
167 {
168   static gchar buf[10000];
169   gchar *msg;
170   static gchar *bufp = buf;
171   gchar *retval;
172   va_list args;
173
174   va_start (args, format);
175   msg = g_strdup_vprintf (format, args);
176   va_end (args);
177
178   g_assert (strlen (msg) < sizeof (buf));
179
180   if (bufp + strlen (msg) + 1 > buf + sizeof (buf))
181     bufp = buf;
182   retval = bufp;
183
184   strcpy (bufp, msg);
185   bufp += strlen (msg) + 1;
186   g_free (msg);
187
188   return retval;
189 }
190
191 gchar *
192 _gdk_win32_color_to_string (const GdkColor *color)
193 {
194   return static_printf ("(%.04x,%.04x,%.04x):%.06x",
195                         color->red, color->green,
196                         color->blue, color->pixel);
197 }
198
199 void
200 _gdk_win32_print_paletteentries (const PALETTEENTRY *pep,
201                                 const int           nentries)
202 {
203   char buf[20];
204   int i;
205
206   for (i = 0; i < nentries; i++)
207     g_print ("  %3d %02x:  %02x %02x %02x%s\n",
208              i, i,
209              pep[i].peRed, pep[i].peGreen, pep[i].peBlue,
210              (pep[i].peFlags == 0 ? "" :
211               (pep[i].peFlags == PC_EXPLICIT ? " PC_EXPLICIT" :
212                (pep[i].peFlags == PC_NOCOLLAPSE ? " PC_NOCOLLAPSE" :
213                 (pep[i].peFlags == PC_RESERVED ? " PC_RESERVED" :
214                  (g_sprintf (buf, " %d", pep[i].peFlags), buf))))));
215 }
216
217 void
218 _gdk_win32_print_system_palette (void)
219 {
220   PALETTEENTRY *pe;
221   int k;
222
223   k = GetSystemPaletteEntries (_gdk_display_hdc, 0, 0, NULL);
224   pe = g_new (PALETTEENTRY, k);
225   k = GetSystemPaletteEntries (_gdk_display_hdc, 0, k, pe);
226
227   if (!k)
228     g_print ("GetSystemPaletteEntries failed: %s\n",
229              g_win32_error_message (GetLastError ()));
230   else
231     {
232       g_print ("System palette: %d entries\n", k);
233       _gdk_win32_print_paletteentries (pe, k);
234     }
235   g_free (pe);
236 }
237
238 static gint
239 palette_size (HPALETTE hpal)
240 {
241   WORD npal = 0;
242
243   if (!GetObject (hpal, sizeof (npal), &npal))
244     WIN32_GDI_FAILED ("GetObject (HPALETTE)");
245
246   return npal;
247 }
248
249 void
250 _gdk_win32_print_hpalette (HPALETTE hpal)
251 {
252   PALETTEENTRY *pe;
253   gint n, npal;
254
255   npal = palette_size (hpal);
256   pe = g_new (PALETTEENTRY, npal);
257   n = GetPaletteEntries (hpal, 0, npal, pe);
258
259   if (!n)
260     g_print ("HPALETTE %p: GetPaletteEntries failed: %s\n",
261              hpal, g_win32_error_message (GetLastError ()));
262   else
263     {
264       g_print ("HPALETTE %p: %d (%d) entries\n", hpal, n, npal);
265       _gdk_win32_print_paletteentries (pe, n);
266     }
267   g_free (pe);
268 }
269
270 void
271 _gdk_win32_print_dc (HDC hdc)
272 {
273   HGDIOBJ obj;
274   LOGBRUSH logbrush;
275   EXTLOGPEN extlogpen;
276   HRGN hrgn;
277   RECT rect;
278   int flag;
279
280   g_print ("%p:\n", hdc);
281   obj = GetCurrentObject (hdc, OBJ_BRUSH);
282   GetObject (obj, sizeof (LOGBRUSH), &logbrush);
283   g_print ("brush: %s color=%06lx hatch=%p\n",
284            _gdk_win32_lbstyle_to_string (logbrush.lbStyle),
285            logbrush.lbColor, (gpointer) logbrush.lbHatch);
286   obj = GetCurrentObject (hdc, OBJ_PEN);
287   GetObject (obj, sizeof (EXTLOGPEN), &extlogpen);
288   g_print ("pen: %s %s %s %s w=%d %s\n",
289            _gdk_win32_pstype_to_string (extlogpen.elpPenStyle),
290            _gdk_win32_psstyle_to_string (extlogpen.elpPenStyle),
291            _gdk_win32_psendcap_to_string (extlogpen.elpPenStyle),
292            _gdk_win32_psjoin_to_string (extlogpen.elpPenStyle),
293            (int) extlogpen.elpWidth,
294            _gdk_win32_lbstyle_to_string (extlogpen.elpBrushStyle));
295   g_print ("rop2: %s textcolor=%06lx\n",
296            _gdk_win32_rop2_to_string (GetROP2 (hdc)),
297            GetTextColor (hdc));
298   hrgn = CreateRectRgn (0, 0, 0, 0);
299   if ((flag = GetClipRgn (hdc, hrgn)) == -1)
300     WIN32_API_FAILED ("GetClipRgn");
301   else if (flag == 0)
302     g_print ("no clip region\n");
303   else if (flag == 1)
304     {
305       GetRgnBox (hrgn, &rect);
306       g_print ("clip region: %p bbox: %s\n",
307                hrgn, _gdk_win32_rect_to_string (&rect));
308     }
309   DeleteObject (hrgn);
310 }
311
312 gchar *
313 _gdk_win32_drag_protocol_to_string (GdkDragProtocol protocol)
314 {
315   switch (protocol)
316     {
317 #define CASE(x) case GDK_DRAG_PROTO_##x: return #x
318       CASE (MOTIF);
319       CASE (XDND);
320       CASE (ROOTWIN);
321       CASE (NONE);
322       CASE (WIN32_DROPFILES);
323       CASE (OLE2);
324       CASE (LOCAL);
325 #undef CASE
326     default: return static_printf ("illegal_%d", protocol);
327     }
328   /* NOTREACHED */
329   return NULL; 
330 }
331
332 gchar *
333 _gdk_win32_window_state_to_string (GdkWindowState state)
334 {
335   gchar buf[100];
336   gchar *bufp = buf;
337   gchar *s = "";
338
339   buf[0] = '\0';
340
341 #define BIT(x)                                          \
342   if (state & GDK_WINDOW_STATE_ ## x)                   \
343     (bufp += sprintf (bufp, "%s" #x, s), s = "|")
344
345   /* For clarity, also show the complement of WITHDRAWN, i.e. "MAPPED" */
346   if (!(state & GDK_WINDOW_STATE_WITHDRAWN))
347     (bufp += sprintf (bufp, "MAPPED"), s = "|");
348
349   BIT (WITHDRAWN);
350   BIT (ICONIFIED);
351   BIT (MAXIMIZED);
352   BIT (STICKY);
353 #undef BIT
354
355   return static_printf ("%s", buf);  
356 }
357
358 gchar *
359 _gdk_win32_window_style_to_string (LONG style)
360 {
361   gchar buf[1000];
362   gchar *bufp = buf;
363   gchar *s = "";
364
365   buf[0] = '\0';
366
367 #define BIT(x)                                          \
368   if (style & WS_ ## x)                                 \
369     (bufp += sprintf (bufp, "%s" #x, s), s = "|")
370
371   /* Note that many of the WS_* macros are in face several bits.
372    * Handle just the individual bits here. Sort as in w32api's
373    * winuser.h.
374    */
375   BIT (BORDER);
376   BIT (CHILD);
377   BIT (CLIPCHILDREN);
378   BIT (CLIPSIBLINGS);
379   BIT (DISABLED);
380   BIT (DLGFRAME);
381   BIT (GROUP);
382   BIT (HSCROLL);
383   BIT (ICONIC);
384   BIT (MAXIMIZE);
385   BIT (MAXIMIZEBOX);
386   BIT (MINIMIZE);
387   BIT (MINIMIZEBOX);
388   BIT (POPUP);
389   BIT (SIZEBOX);
390   BIT (SYSMENU);
391   BIT (TABSTOP);
392   BIT (THICKFRAME);
393   BIT (VISIBLE);
394   BIT (VSCROLL);
395 #undef BIT
396
397   return static_printf ("%s", buf);  
398 }
399
400 gchar *
401 _gdk_win32_window_exstyle_to_string (LONG style)
402 {
403   gchar buf[1000];
404   gchar *bufp = buf;
405   gchar *s = "";
406
407   buf[0] = '\0';
408
409 #define BIT(x)                                          \
410   if (style & WS_EX_ ## x)                              \
411     (bufp += sprintf (bufp, "%s" #x, s), s = "|")
412
413   /* Note that many of the WS_EX_* macros are in face several bits.
414    * Handle just the individual bits here. Sort as in w32api's
415    * winuser.h.
416    */
417   BIT (ACCEPTFILES);
418   BIT (APPWINDOW);
419   BIT (CLIENTEDGE);
420 #ifndef WS_EX_COMPOSITED
421 #  define WS_EX_COMPOSITED 0x02000000L
422 #endif
423   BIT (COMPOSITED);
424   BIT (CONTEXTHELP);
425   BIT (CONTROLPARENT);
426   BIT (DLGMODALFRAME);
427   BIT (LAYERED);
428   BIT (LAYOUTRTL);
429   BIT (LEFTSCROLLBAR);
430   BIT (MDICHILD);
431   BIT (NOACTIVATE);
432   BIT (NOINHERITLAYOUT);
433   BIT (NOPARENTNOTIFY);
434   BIT (RIGHT);
435   BIT (RTLREADING);
436   BIT (STATICEDGE);
437   BIT (TOOLWINDOW);
438   BIT (TOPMOST);
439   BIT (TRANSPARENT);
440   BIT (WINDOWEDGE);
441 #undef BIT
442
443   return static_printf ("%s", buf);  
444 }
445
446 gchar *
447 _gdk_win32_window_pos_bits_to_string (UINT flags)
448 {
449   gchar buf[1000];
450   gchar *bufp = buf;
451   gchar *s = "";
452
453   buf[0] = '\0';
454
455 #define BIT(x)                                          \
456   if (flags & SWP_ ## x)                                \
457     (bufp += sprintf (bufp, "%s" #x, s), s = "|")
458
459   BIT (DRAWFRAME);
460   BIT (FRAMECHANGED);
461   BIT (HIDEWINDOW);
462   BIT (NOACTIVATE);
463   BIT (NOCOPYBITS);
464   BIT (NOMOVE);
465   BIT (NOSIZE);
466   BIT (NOREDRAW);
467   BIT (NOZORDER);
468   BIT (SHOWWINDOW);
469   BIT (NOOWNERZORDER);
470   BIT (NOSENDCHANGING);
471   BIT (DEFERERASE);
472   BIT (ASYNCWINDOWPOS);
473 #undef BIT
474
475   return static_printf ("%s", buf);  
476 }
477
478 gchar *
479 _gdk_win32_drag_action_to_string (GdkDragAction actions)
480 {
481   gchar buf[100];
482   gchar *bufp = buf;
483   gchar *s = "";
484
485   buf[0] = '\0';
486
487 #define BIT(x)                                          \
488   if (actions & GDK_ACTION_ ## x)                               \
489     (bufp += sprintf (bufp, "%s" #x, s), s = "|")
490
491   BIT (DEFAULT);
492   BIT (COPY);
493   BIT (MOVE);
494   BIT (LINK);
495   BIT (PRIVATE);
496   BIT (ASK);
497 #undef BIT
498
499   return static_printf ("%s", buf);  
500 }
501
502 gchar *
503 _gdk_win32_rop2_to_string (int rop2)
504 {
505   switch (rop2)
506     {
507 #define CASE(x) case R2_##x: return #x
508       CASE (BLACK);
509       CASE (COPYPEN);
510       CASE (MASKNOTPEN);
511       CASE (MASKPEN);
512       CASE (MASKPENNOT);
513       CASE (MERGENOTPEN);
514       CASE (MERGEPEN);
515       CASE (MERGEPENNOT);
516       CASE (NOP);
517       CASE (NOT);
518       CASE (NOTCOPYPEN);
519       CASE (NOTMASKPEN);
520       CASE (NOTMERGEPEN);
521       CASE (NOTXORPEN);
522       CASE (WHITE);
523       CASE (XORPEN);
524 #undef CASE
525     default: return static_printf ("illegal_%x", rop2);
526     }
527   /* NOTREACHED */
528   return NULL;
529 }
530
531 gchar *
532 _gdk_win32_lbstyle_to_string (UINT brush_style)
533 {
534   switch (brush_style)
535     {
536 #define CASE(x) case BS_##x: return #x
537       CASE (DIBPATTERN);
538       CASE (DIBPATTERNPT);
539       CASE (HATCHED);
540       CASE (HOLLOW);
541       CASE (PATTERN);
542       CASE (SOLID);
543 #undef CASE
544     default: return static_printf ("illegal_%d", brush_style);
545     }
546   /* NOTREACHED */
547   return NULL;
548 }
549
550 gchar *
551 _gdk_win32_pstype_to_string (DWORD pen_style)
552 {
553   switch (pen_style & PS_TYPE_MASK)
554     {
555     case PS_GEOMETRIC: return "GEOMETRIC";
556     case PS_COSMETIC: return "COSMETIC";
557     default: return static_printf ("illegal_%d", pen_style & PS_TYPE_MASK);
558     }
559   /* NOTREACHED */
560   return NULL;
561 }
562
563 gchar *
564 _gdk_win32_psstyle_to_string (DWORD pen_style)
565 {
566   switch (pen_style & PS_STYLE_MASK)
567     {
568 #define CASE(x) case PS_##x: return #x
569       CASE (ALTERNATE);
570       CASE (SOLID);
571       CASE (DASH);
572       CASE (DOT);
573       CASE (DASHDOT);
574       CASE (DASHDOTDOT);
575       CASE (NULL);
576       CASE (USERSTYLE);
577       CASE (INSIDEFRAME);
578 #undef CASE
579     default: return static_printf ("illegal_%d", pen_style & PS_STYLE_MASK);
580     }
581   /* NOTREACHED */
582   return NULL;
583 }
584
585 gchar *
586 _gdk_win32_psendcap_to_string (DWORD pen_style)
587 {
588   switch (pen_style & PS_ENDCAP_MASK)
589     {
590 #define CASE(x) case PS_ENDCAP_##x: return #x
591       CASE (ROUND);
592       CASE (SQUARE);
593       CASE (FLAT);
594 #undef CASE
595     default: return static_printf ("illegal_%d", pen_style & PS_ENDCAP_MASK);
596     }
597   /* NOTREACHED */
598   return NULL;
599 }
600
601 gchar *
602 _gdk_win32_psjoin_to_string (DWORD pen_style)
603 {
604   switch (pen_style & PS_JOIN_MASK)
605     {
606 #define CASE(x) case PS_JOIN_##x: return #x
607       CASE (ROUND);
608       CASE (BEVEL);
609       CASE (MITER);
610 #undef CASE
611     default: return static_printf ("illegal_%d", pen_style & PS_JOIN_MASK);
612     }
613   /* NOTREACHED */
614   return NULL;
615 }
616
617 gchar *
618 _gdk_win32_message_to_string (UINT msg)
619 {
620   switch (msg)
621     {
622 #define CASE(x) case x: return #x
623       CASE (WM_NULL);
624       CASE (WM_CREATE);
625       CASE (WM_DESTROY);
626       CASE (WM_MOVE);
627       CASE (WM_SIZE);
628       CASE (WM_ACTIVATE);
629       CASE (WM_SETFOCUS);
630       CASE (WM_KILLFOCUS);
631       CASE (WM_ENABLE);
632       CASE (WM_SETREDRAW);
633       CASE (WM_SETTEXT);
634       CASE (WM_GETTEXT);
635       CASE (WM_GETTEXTLENGTH);
636       CASE (WM_PAINT);
637       CASE (WM_CLOSE);
638       CASE (WM_QUERYENDSESSION);
639       CASE (WM_QUERYOPEN);
640       CASE (WM_ENDSESSION);
641       CASE (WM_QUIT);
642       CASE (WM_ERASEBKGND);
643       CASE (WM_SYSCOLORCHANGE);
644       CASE (WM_SHOWWINDOW);
645       CASE (WM_WININICHANGE);
646       CASE (WM_DEVMODECHANGE);
647       CASE (WM_ACTIVATEAPP);
648       CASE (WM_FONTCHANGE);
649       CASE (WM_TIMECHANGE);
650       CASE (WM_CANCELMODE);
651       CASE (WM_SETCURSOR);
652       CASE (WM_MOUSEACTIVATE);
653       CASE (WM_CHILDACTIVATE);
654       CASE (WM_QUEUESYNC);
655       CASE (WM_GETMINMAXINFO);
656       CASE (WM_PAINTICON);
657       CASE (WM_ICONERASEBKGND);
658       CASE (WM_NEXTDLGCTL);
659       CASE (WM_SPOOLERSTATUS);
660       CASE (WM_DRAWITEM);
661       CASE (WM_MEASUREITEM);
662       CASE (WM_DELETEITEM);
663       CASE (WM_VKEYTOITEM);
664       CASE (WM_CHARTOITEM);
665       CASE (WM_SETFONT);
666       CASE (WM_GETFONT);
667       CASE (WM_SETHOTKEY);
668       CASE (WM_GETHOTKEY);
669       CASE (WM_QUERYDRAGICON);
670       CASE (WM_COMPAREITEM);
671       CASE (WM_GETOBJECT);
672       CASE (WM_COMPACTING);
673       CASE (WM_WINDOWPOSCHANGING);
674       CASE (WM_WINDOWPOSCHANGED);
675       CASE (WM_POWER);
676       CASE (WM_COPYDATA);
677       CASE (WM_CANCELJOURNAL);
678       CASE (WM_NOTIFY);
679       CASE (WM_INPUTLANGCHANGEREQUEST);
680       CASE (WM_INPUTLANGCHANGE);
681       CASE (WM_TCARD);
682       CASE (WM_HELP);
683       CASE (WM_USERCHANGED);
684       CASE (WM_NOTIFYFORMAT);
685       CASE (WM_CONTEXTMENU);
686       CASE (WM_STYLECHANGING);
687       CASE (WM_STYLECHANGED);
688       CASE (WM_DISPLAYCHANGE);
689       CASE (WM_GETICON);
690       CASE (WM_SETICON);
691       CASE (WM_NCCREATE);
692       CASE (WM_NCDESTROY);
693       CASE (WM_NCCALCSIZE);
694       CASE (WM_NCHITTEST);
695       CASE (WM_NCPAINT);
696       CASE (WM_NCACTIVATE);
697       CASE (WM_GETDLGCODE);
698       CASE (WM_SYNCPAINT);
699       CASE (WM_NCMOUSEMOVE);
700       CASE (WM_NCLBUTTONDOWN);
701       CASE (WM_NCLBUTTONUP);
702       CASE (WM_NCLBUTTONDBLCLK);
703       CASE (WM_NCRBUTTONDOWN);
704       CASE (WM_NCRBUTTONUP);
705       CASE (WM_NCRBUTTONDBLCLK);
706       CASE (WM_NCMBUTTONDOWN);
707       CASE (WM_NCMBUTTONUP);
708       CASE (WM_NCMBUTTONDBLCLK);
709       CASE (WM_NCXBUTTONDOWN);
710       CASE (WM_NCXBUTTONUP);
711       CASE (WM_NCXBUTTONDBLCLK);
712       CASE (WM_KEYDOWN);
713       CASE (WM_KEYUP);
714       CASE (WM_CHAR);
715       CASE (WM_DEADCHAR);
716       CASE (WM_SYSKEYDOWN);
717       CASE (WM_SYSKEYUP);
718       CASE (WM_SYSCHAR);
719       CASE (WM_SYSDEADCHAR);
720       CASE (WM_KEYLAST);
721       CASE (WM_IME_STARTCOMPOSITION);
722       CASE (WM_IME_ENDCOMPOSITION);
723       CASE (WM_IME_COMPOSITION);
724       CASE (WM_INITDIALOG);
725       CASE (WM_COMMAND);
726       CASE (WM_SYSCOMMAND);
727       CASE (WM_TIMER);
728       CASE (WM_HSCROLL);
729       CASE (WM_VSCROLL);
730       CASE (WM_INITMENU);
731       CASE (WM_INITMENUPOPUP);
732       CASE (WM_MENUSELECT);
733       CASE (WM_MENUCHAR);
734       CASE (WM_ENTERIDLE);
735       CASE (WM_MENURBUTTONUP);
736       CASE (WM_MENUDRAG);
737       CASE (WM_MENUGETOBJECT);
738       CASE (WM_UNINITMENUPOPUP);
739       CASE (WM_MENUCOMMAND);
740       CASE (WM_CHANGEUISTATE);
741       CASE (WM_UPDATEUISTATE);
742       CASE (WM_QUERYUISTATE);
743       CASE (WM_CTLCOLORMSGBOX);
744       CASE (WM_CTLCOLOREDIT);
745       CASE (WM_CTLCOLORLISTBOX);
746       CASE (WM_CTLCOLORBTN);
747       CASE (WM_CTLCOLORDLG);
748       CASE (WM_CTLCOLORSCROLLBAR);
749       CASE (WM_CTLCOLORSTATIC);
750       CASE (WM_MOUSEMOVE);
751       CASE (WM_LBUTTONDOWN);
752       CASE (WM_LBUTTONUP);
753       CASE (WM_LBUTTONDBLCLK);
754       CASE (WM_RBUTTONDOWN);
755       CASE (WM_RBUTTONUP);
756       CASE (WM_RBUTTONDBLCLK);
757       CASE (WM_MBUTTONDOWN);
758       CASE (WM_MBUTTONUP);
759       CASE (WM_MBUTTONDBLCLK);
760       CASE (WM_MOUSEWHEEL);
761       CASE (WM_XBUTTONDOWN);
762       CASE (WM_XBUTTONUP);
763       CASE (WM_XBUTTONDBLCLK);
764       CASE (WM_PARENTNOTIFY);
765       CASE (WM_ENTERMENULOOP);
766       CASE (WM_EXITMENULOOP);
767       CASE (WM_NEXTMENU);
768       CASE (WM_SIZING);
769       CASE (WM_CAPTURECHANGED);
770       CASE (WM_MOVING);
771       CASE (WM_POWERBROADCAST);
772       CASE (WM_DEVICECHANGE);
773       CASE (WM_MDICREATE);
774       CASE (WM_MDIDESTROY);
775       CASE (WM_MDIACTIVATE);
776       CASE (WM_MDIRESTORE);
777       CASE (WM_MDINEXT);
778       CASE (WM_MDIMAXIMIZE);
779       CASE (WM_MDITILE);
780       CASE (WM_MDICASCADE);
781       CASE (WM_MDIICONARRANGE);
782       CASE (WM_MDIGETACTIVE);
783       CASE (WM_MDISETMENU);
784       CASE (WM_ENTERSIZEMOVE);
785       CASE (WM_EXITSIZEMOVE);
786       CASE (WM_DROPFILES);
787       CASE (WM_MDIREFRESHMENU);
788       CASE (WM_IME_SETCONTEXT);
789       CASE (WM_IME_NOTIFY);
790       CASE (WM_IME_CONTROL);
791       CASE (WM_IME_COMPOSITIONFULL);
792       CASE (WM_IME_SELECT);
793       CASE (WM_IME_CHAR);
794       CASE (WM_IME_REQUEST);
795       CASE (WM_IME_KEYDOWN);
796       CASE (WM_IME_KEYUP);
797       CASE (WM_MOUSEHOVER);
798       CASE (WM_MOUSELEAVE);
799       CASE (WM_NCMOUSEHOVER);
800       CASE (WM_NCMOUSELEAVE);
801       CASE (WM_CUT);
802       CASE (WM_COPY);
803       CASE (WM_PASTE);
804       CASE (WM_CLEAR);
805       CASE (WM_UNDO);
806       CASE (WM_RENDERFORMAT);
807       CASE (WM_RENDERALLFORMATS);
808       CASE (WM_DESTROYCLIPBOARD);
809       CASE (WM_DRAWCLIPBOARD);
810       CASE (WM_PAINTCLIPBOARD);
811       CASE (WM_VSCROLLCLIPBOARD);
812       CASE (WM_SIZECLIPBOARD);
813       CASE (WM_ASKCBFORMATNAME);
814       CASE (WM_CHANGECBCHAIN);
815       CASE (WM_HSCROLLCLIPBOARD);
816       CASE (WM_QUERYNEWPALETTE);
817       CASE (WM_PALETTEISCHANGING);
818       CASE (WM_PALETTECHANGED);
819       CASE (WM_HOTKEY);
820       CASE (WM_PRINT);
821       CASE (WM_PRINTCLIENT);
822       CASE (WM_APPCOMMAND);
823       CASE (WM_HANDHELDFIRST);
824       CASE (WM_HANDHELDLAST);
825       CASE (WM_AFXFIRST);
826       CASE (WM_AFXLAST);
827       CASE (WM_PENWINFIRST);
828       CASE (WM_PENWINLAST);
829       CASE (WM_APP);
830       CASE (WT_PACKET);
831       CASE (WT_CSRCHANGE);
832       CASE (WT_PROXIMITY);
833 #undef CASE
834     default:
835       if (msg >= WM_HANDHELDFIRST && msg <= WM_HANDHELDLAST)
836         return static_printf ("WM_HANDHELDFIRST+%d", msg - WM_HANDHELDFIRST);
837       else if (msg >= WM_AFXFIRST && msg <= WM_AFXLAST)
838         return static_printf ("WM_AFXFIRST+%d", msg - WM_AFXFIRST);
839       else if (msg >= WM_PENWINFIRST && msg <= WM_PENWINLAST)
840         return static_printf ("WM_PENWINFIRST+%d", msg - WM_PENWINFIRST);
841       else if (msg >= WM_USER && msg <= 0x7FFF)
842         return static_printf ("WM_USER+%d", msg - WM_USER);
843       else if (msg >= 0xC000 && msg <= 0xFFFF)
844         return static_printf ("reg-%#x", msg);
845       else
846         return static_printf ("unk-%#x", msg);
847     }
848   /* NOTREACHED */
849   return NULL;
850 }
851
852 gchar *
853 _gdk_win32_key_to_string (LONG lParam)
854 {
855   char buf[100];
856   gchar *keyname_utf8;
857
858   if (GetKeyNameText (lParam, buf, sizeof (buf)) &&
859       (keyname_utf8 = g_locale_to_utf8 (buf, -1, NULL, NULL, NULL)) != NULL)
860     {
861       gchar *retval = static_printf ("%s", keyname_utf8);
862
863       g_free (keyname_utf8);
864
865       return retval;
866     }
867
868   return static_printf ("unk-%#lx", lParam);
869 }
870       
871 gchar *
872 _gdk_win32_cf_to_string (UINT format)
873 {
874   char buf[100];
875
876   switch (format)
877     {
878 #define CASE(x) case CF_##x: return "CF_" #x
879       CASE (BITMAP);
880       CASE (DIB);
881       CASE (DIBV5);
882       CASE (DIF);
883       CASE (DSPBITMAP);
884       CASE (DSPENHMETAFILE);
885       CASE (DSPMETAFILEPICT);
886       CASE (DSPTEXT);
887       CASE (ENHMETAFILE);
888       CASE (HDROP);
889       CASE (LOCALE);
890       CASE (METAFILEPICT);
891       CASE (OEMTEXT);
892       CASE (OWNERDISPLAY);
893       CASE (PALETTE);
894       CASE (PENDATA);
895       CASE (RIFF);
896       CASE (SYLK);
897       CASE (TEXT);
898       CASE (WAVE);
899       CASE (TIFF);
900       CASE (UNICODETEXT);
901     default:
902       if (format >= CF_GDIOBJFIRST &&
903           format <= CF_GDIOBJLAST)
904         return static_printf ("CF_GDIOBJ%d", format - CF_GDIOBJFIRST);
905       if (format >= CF_PRIVATEFIRST &&
906           format <= CF_PRIVATELAST)
907         return static_printf ("CF_PRIVATE%d", format - CF_PRIVATEFIRST);
908       if (GetClipboardFormatName (format, buf, sizeof (buf)))
909         return static_printf ("'%s'", buf);
910       else
911         return static_printf ("unk-%#lx", format);
912     }
913 }
914       
915 gchar *
916 _gdk_win32_data_to_string (const guchar *data,
917                            int           nbytes)
918 {
919   GString *s = g_string_new ("");
920   int i;
921   gchar *retval;
922
923   for (i = 0; i < nbytes; i++)
924     if (data[i] >=' ' && data[i] <= '~')
925       g_string_append_printf (s, "%c  ", data[i]);
926     else
927       g_string_append_printf (s, "%02X ", data[i]);
928
929   retval = static_printf ("%s", s->str);
930   g_string_free (s, TRUE);
931
932   return retval;
933 }
934
935 gchar *
936 _gdk_win32_rect_to_string (const RECT *rect)
937 {
938   return static_printf ("%ldx%ld@%+ld%+ld",
939                         (rect->right - rect->left), (rect->bottom - rect->top),
940                         rect->left, rect->top);
941 }
942
943 gchar *
944 _gdk_win32_gdkrectangle_to_string (const GdkRectangle *rect)
945 {
946   return static_printf ("%dx%d@%+d%+d",
947                         rect->width, rect->height,
948                         rect->x, rect->y);
949 }
950
951 gchar *
952 _gdk_win32_cairo_region_to_string (const cairo_region_t *rgn)
953 {
954   cairo_rectangle_int_t extents;
955   cairo_region_get_extents (rgn, &extents);
956   return static_printf ("%dx%d@%+d%+d",
957                         extents.width, extents.height,
958                         extents.x, extents.y);
959 }
960
961 gchar *
962 _gdk_win32_window_description (GdkWindow *d)
963 {
964   g_return_val_if_fail (GDK_IS_WINDOW (d), NULL);
965
966   return static_printf ("%s:%p:%dx%dx%d",
967                         G_OBJECT_TYPE_NAME (d),
968                         GDK_WINDOW_HWND (d),
969                         gdk_window_get_width (GDK_WINDOW (d)),
970                         gdk_window_get_height (GDK_WINDOW (d)),
971                         gdk_visual_get_depth (gdk_window_get_visual (GDK_WINDOW (d))));
972 }
973
974 #endif /* G_ENABLE_DEBUG */