]> Pileus Git - ~andy/gtk/blob - gdk/win32/gdkmain-win32.c
Print debugging output only if asked for, not always.
[~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, write to the
17  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18  * Boston, MA 02111-1307, USA.
19  */
20
21 /*
22  * Modified by the GTK+ Team and others 1997-2000.  See the AUTHORS
23  * file for a list of people on the GTK+ Team.  See the ChangeLog
24  * files for a list of changes.  These files are distributed with
25  * GTK+ at ftp://ftp.gtk.org/pub/gtk/. 
26  */
27
28 #include <config.h>
29
30 #include <glib/gprintf.h>
31 #include <stdlib.h>
32 #include <string.h>
33 #include <limits.h>
34 #include <io.h>
35
36 #include "gdk.h"
37 #include "gdkregion-generic.h"
38 #include "gdkkeysyms.h"
39 #include "gdkinternals.h"
40 #include "gdkintl.h"
41 #include "gdkprivate-win32.h"
42 #include "gdkinput-win32.h"
43
44 #include <objbase.h>
45
46 #if defined (__GNUC__) && defined (HAVE_DIMM_H)
47 /* The w32api imm.h clashes a bit with the IE5.5 dimm.h */
48 # define IMEMENUITEMINFOA hidden_IMEMENUITEMINFOA
49 # define IMEMENUITEMINFOW hidden_IMEMENUITEMINFOW
50 #endif
51
52 #include <imm.h>
53
54 static gboolean gdk_synchronize = FALSE;
55
56 static gboolean dummy;
57
58 GOptionEntry _gdk_windowing_args[] = {
59   { "sync", 0, 0, G_OPTION_ARG_NONE, &gdk_synchronize, 
60     /* Description of --sync in --help output */              N_("Don't batch GDI requests"), NULL },
61   { "no-wintab", 0, 0, G_OPTION_ARG_NONE, &_gdk_input_ignore_wintab, 
62     /* Description of --no-wintab in --help output */         N_("Don't use the Wintab API for tablet support"), NULL },
63   { "ignore-wintab", 0, 0, G_OPTION_ARG_NONE, &_gdk_input_ignore_wintab, 
64     /* Description of --ignore-wintab in --help output */     N_("Same as --no-wintab"), NULL },
65   { "use-wintab", 0, 0, G_OPTION_ARG_NONE, &dummy,
66     /* Description of --use-wintab in --help output */     N_("Do use the Wintab API [default]"), NULL },
67   { "max-colors", 0, 0, G_OPTION_ARG_INT, &_gdk_max_colors, 
68     /* Description of --max-colors=COLORS in --help output */ N_("Size of the palette in 8 bit mode"), 
69     /* Placeholder in --max-colors=COLORS in --help output */ N_("COLORS") },
70   { NULL }
71 };
72
73 int __stdcall
74 DllMain (HINSTANCE hinstDLL,
75          DWORD     dwReason,
76          LPVOID    reserved)
77 {
78   _gdk_dll_hinstance = hinstDLL;
79
80   return TRUE;
81 }
82
83 void
84 _gdk_windowing_init (void)
85 {
86   gchar buf[10];
87
88 #ifdef HAVE_WINTAB
89   if (getenv ("GDK_IGNORE_WINTAB") != NULL)
90     _gdk_input_ignore_wintab = TRUE;
91   else if (getenv ("GDK_USE_WINTAB") != NULL)
92     _gdk_input_ignore_wintab = FALSE;
93 #endif
94
95   if (gdk_synchronize)
96     GdiSetBatchLimit (1);
97
98   _gdk_app_hmodule = GetModuleHandle (NULL);
99   _gdk_display_hdc = CreateDC ("DISPLAY", NULL, NULL, NULL);
100   _gdk_root_window = GetDesktopWindow ();
101   _gdk_input_locale = GetKeyboardLayout (0);
102   _gdk_input_locale_is_ime = ImmIsIME (_gdk_input_locale);
103   GetLocaleInfo (MAKELCID (LOWORD (_gdk_input_locale), SORT_DEFAULT),
104                  LOCALE_IDEFAULTANSICODEPAGE,
105                  buf, sizeof (buf));
106   _gdk_input_codepage = atoi (buf);
107   GDK_NOTE (EVENTS, g_print ("input_locale:%p, codepage:%d\n",
108                              _gdk_input_locale, _gdk_input_codepage));
109
110   CoInitialize (NULL);
111
112   _cf_rtf = RegisterClipboardFormat ("Rich Text Format");
113   _cf_utf8_string = RegisterClipboardFormat ("UTF8_STRING");
114
115   _utf8_string = gdk_atom_intern ("UTF8_STRING", FALSE);
116   _text_uri_list = gdk_atom_intern ("text/uri-list", FALSE);
117   _targets = gdk_atom_intern ("TARGETS", FALSE);
118
119   _local_dnd = gdk_atom_intern ("LocalDndSelection", FALSE);
120   _gdk_win32_dropfiles = gdk_atom_intern ("DROPFILES_DND", FALSE);
121   _gdk_ole2_dnd = gdk_atom_intern ("OLE2_DND", FALSE);
122
123   _gdk_selection_property = gdk_atom_intern ("GDK_SELECTION", FALSE);
124
125   _wm_transient_for = gdk_atom_intern ("WM_TRANSIENT_FOR", FALSE);
126
127   _gdk_win32_selection_init ();
128 }
129
130 void
131 _gdk_win32_api_failed (const gchar *where,
132                       gint         line,
133                       const gchar *api)
134 {
135   gchar *msg = g_win32_error_message (GetLastError ());
136   g_warning ("%s:%d: %s failed: %s", where, line, api, msg);
137   g_free (msg);
138 }
139
140 void
141 _gdk_other_api_failed (const gchar *where,
142                       gint         line,
143                       const gchar *api)
144 {
145   g_warning ("%s:%d: %s failed", where, line, api);
146 }
147
148 void
149 _gdk_win32_gdi_failed (const gchar *where,
150                       gint         line,
151                       const gchar *api)
152 {
153   /* On Win9x GDI calls are implemented in 16-bit code and thus
154    * don't set the 32-bit error code, sigh.
155    */
156   if (G_WIN32_IS_NT_BASED ())
157     _gdk_win32_api_failed (where, line, api);
158   else
159     _gdk_other_api_failed (where, line, api);
160 }
161
162 void
163 gdk_set_use_xshm (gboolean use_xshm)
164 {
165   /* Always on */
166 }
167
168 gboolean
169 gdk_get_use_xshm (void)
170 {
171   return TRUE;
172 }
173
174 gint
175 gdk_screen_get_width (GdkScreen *screen)
176 {
177   return GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (_gdk_parent_root)->impl)->width;
178 }
179
180 gint
181 gdk_screen_get_height (GdkScreen *screen)
182 {
183   return GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (_gdk_parent_root)->impl)->height;
184 }
185 gint
186 gdk_screen_get_width_mm (GdkScreen *screen)
187 {
188   return (double) GetDeviceCaps (_gdk_display_hdc, HORZRES) / GetDeviceCaps (_gdk_display_hdc, LOGPIXELSX) * 25.4;
189 }
190
191 gint
192 gdk_screen_get_height_mm (GdkScreen *screen)
193 {
194   return (double) GetDeviceCaps (_gdk_display_hdc, VERTRES) / GetDeviceCaps (_gdk_display_hdc, LOGPIXELSY) * 25.4;
195 }
196
197 void
198 _gdk_windowing_display_set_sm_client_id (GdkDisplay  *display,
199                                          const gchar *sm_client_id)
200 {
201   g_warning("gdk_set_sm_client_id %s", sm_client_id ? sm_client_id : "NULL");
202 }
203
204 void
205 gdk_display_beep (GdkDisplay *display)
206 {
207   g_return_if_fail (display == gdk_display_get_default());
208   Beep(1000, 50);
209 }
210
211 void
212 _gdk_windowing_exit (void)
213 {
214   _gdk_win32_dnd_exit ();
215   CoUninitialize ();
216   DeleteDC (_gdk_display_hdc);
217   _gdk_display_hdc = NULL;
218 }
219
220 gchar *
221 gdk_get_display (void)
222 {
223   return g_strdup (gdk_display_get_name (gdk_display_get_default ()));
224 }
225
226 void
227 gdk_error_trap_push (void)
228 {
229 }
230
231 gint
232 gdk_error_trap_pop (void)
233 {
234   return 0;
235 }
236
237 void
238 gdk_notify_startup_complete (void)
239 {
240 }
241
242 #ifdef G_ENABLE_DEBUG
243
244 /*
245  * Like g_strdup_printf, but to a static buffer. Return value does not
246  * have to be g_free()d. The buffer is of bounded size and reused
247  * cyclically. Thus the return value is valid only until that part of
248  * the buffer happens to get reused. This doesn't matter as this
249  * function's return value is used in debugging output right after the call,
250  * and the return value isn't used after that.
251  */
252 static gchar *
253 static_printf (const gchar *format,
254                          ...)
255 {
256   static gchar buf[10000];
257   gchar *msg;
258   static gchar *bufp = buf;
259   gchar *retval;
260   va_list args;
261
262   va_start (args, format);
263   msg = g_strdup_vprintf (format, args);
264   va_end (args);
265
266   g_assert (strlen (msg) < sizeof (buf));
267
268   if (bufp + strlen (msg) + 1 > buf + sizeof (buf))
269     bufp = buf;
270   retval = bufp;
271
272   strcpy (bufp, msg);
273   bufp += strlen (msg) + 1;
274   g_free (msg);
275
276   return retval;
277 }
278
279 gchar *
280 _gdk_win32_color_to_string (const GdkColor *color)
281 {
282   return static_printf ("(%.04x,%.04x,%.04x):%.06x",
283                         color->red, color->green,
284                         color->blue, color->pixel);
285 }
286
287 void
288 _gdk_win32_print_paletteentries (const PALETTEENTRY *pep,
289                                 const int           nentries)
290 {
291   char buf[20];
292   int i;
293
294   for (i = 0; i < nentries; i++)
295     g_print ("  %3d %02x:  %02x %02x %02x%s\n",
296              i, i,
297              pep[i].peRed, pep[i].peGreen, pep[i].peBlue,
298              (pep[i].peFlags == 0 ? "" :
299               (pep[i].peFlags == PC_EXPLICIT ? " PC_EXPLICIT" :
300                (pep[i].peFlags == PC_NOCOLLAPSE ? " PC_NOCOLLAPSE" :
301                 (pep[i].peFlags == PC_RESERVED ? " PC_RESERVED" :
302                  (g_sprintf (buf, " %d", pep[i].peFlags), buf))))));
303 }
304
305 void
306 _gdk_win32_print_system_palette (void)
307 {
308   PALETTEENTRY *pe;
309   int k;
310
311   k = GetSystemPaletteEntries (_gdk_display_hdc, 0, 0, NULL);
312   pe = g_new (PALETTEENTRY, k);
313   k = GetSystemPaletteEntries (_gdk_display_hdc, 0, k, pe);
314
315   if (!k)
316     g_print ("GetSystemPaletteEntries failed: %s\n",
317              g_win32_error_message (GetLastError ()));
318   else
319     {
320       g_print ("System palette: %d entries\n", k);
321       _gdk_win32_print_paletteentries (pe, k);
322     }
323   g_free (pe);
324 }
325
326 static gint
327 palette_size (HPALETTE hpal)
328 {
329   WORD npal = 0;
330
331   if (!GetObject (hpal, sizeof (npal), &npal))
332     WIN32_GDI_FAILED ("GetObject (HPALETTE)");
333
334   return npal;
335 }
336
337 void
338 _gdk_win32_print_hpalette (HPALETTE hpal)
339 {
340   PALETTEENTRY *pe;
341   gint n, npal;
342
343   npal = palette_size (hpal);
344   pe = g_new (PALETTEENTRY, npal);
345   n = GetPaletteEntries (hpal, 0, npal, pe);
346
347   if (!n)
348     g_print ("HPALETTE %p: GetPaletteEntries failed: %s\n",
349              hpal, g_win32_error_message (GetLastError ()));
350   else
351     {
352       g_print ("HPALETTE %p: %d (%d) entries\n", hpal, n, npal);
353       _gdk_win32_print_paletteentries (pe, n);
354     }
355   g_free (pe);
356 }
357
358 void
359 _gdk_win32_print_dc (HDC hdc)
360 {
361   HGDIOBJ obj;
362   LOGBRUSH logbrush;
363   EXTLOGPEN extlogpen;
364   HRGN hrgn;
365   RECT rect;
366   int flag;
367
368   g_print ("%p:\n", hdc);
369   obj = GetCurrentObject (hdc, OBJ_BRUSH);
370   GetObject (obj, sizeof (LOGBRUSH), &logbrush);
371   g_print ("brush: %s color=%06lx hatch=%p\n",
372            _gdk_win32_lbstyle_to_string (logbrush.lbStyle),
373            logbrush.lbColor, (gpointer) logbrush.lbHatch);
374   obj = GetCurrentObject (hdc, OBJ_PEN);
375   GetObject (obj, sizeof (EXTLOGPEN), &extlogpen);
376   g_print ("pen: %s %s %s %s w=%d %s\n",
377            _gdk_win32_pstype_to_string (extlogpen.elpPenStyle),
378            _gdk_win32_psstyle_to_string (extlogpen.elpPenStyle),
379            _gdk_win32_psendcap_to_string (extlogpen.elpPenStyle),
380            _gdk_win32_psjoin_to_string (extlogpen.elpPenStyle),
381            extlogpen.elpWidth,
382            _gdk_win32_lbstyle_to_string (extlogpen.elpBrushStyle));
383   g_print ("rop2: %s textcolor=%06lx\n",
384            _gdk_win32_rop2_to_string (GetROP2 (hdc)),
385            GetTextColor (hdc));
386   hrgn = CreateRectRgn (0, 0, 0, 0);
387   if ((flag = GetClipRgn (hdc, hrgn)) == -1)
388     WIN32_API_FAILED ("GetClipRgn");
389   else if (flag == 0)
390     g_print ("no clip region\n");
391   else if (flag == 1)
392     {
393       GetRgnBox (hrgn, &rect);
394       g_print ("clip region: %p bbox: %s\n",
395                hrgn, _gdk_win32_rect_to_string (&rect));
396     }
397   DeleteObject (hrgn);
398 }
399
400 gchar *
401 _gdk_win32_cap_style_to_string (GdkCapStyle cap_style)
402 {
403   switch (cap_style)
404     {
405 #define CASE(x) case GDK_CAP_##x: return #x
406     CASE (NOT_LAST);
407     CASE (BUTT);
408     CASE (ROUND);
409     CASE (PROJECTING);
410 #undef CASE
411     default: return static_printf ("illegal_%d", cap_style);
412     }
413   /* NOTREACHED */
414   return NULL;
415 }
416
417 gchar *
418 _gdk_win32_fill_style_to_string (GdkFill fill)
419 {
420   switch (fill)
421     {
422 #define CASE(x) case GDK_##x: return #x
423     CASE (SOLID);
424     CASE (TILED);
425     CASE (STIPPLED);
426     CASE (OPAQUE_STIPPLED);
427 #undef CASE
428     default: return static_printf ("illegal_%d", fill);
429     }
430   /* NOTREACHED */
431   return NULL;
432 }
433
434 gchar *
435 _gdk_win32_function_to_string (GdkFunction function)
436 {
437   switch (function)
438     {
439 #define CASE(x) case GDK_##x: return #x
440     CASE (COPY);
441     CASE (INVERT);
442     CASE (XOR);
443     CASE (CLEAR);
444     CASE (AND);
445     CASE (AND_REVERSE);
446     CASE (AND_INVERT);
447     CASE (NOOP);
448     CASE (OR);
449     CASE (EQUIV);
450     CASE (OR_REVERSE);
451     CASE (COPY_INVERT);
452     CASE (OR_INVERT);
453     CASE (NAND);
454     CASE (SET);
455 #undef CASE
456     default: return static_printf ("illegal_%d", function);
457     }
458   /* NOTREACHED */
459   return NULL; 
460 }
461
462 gchar *
463 _gdk_win32_join_style_to_string (GdkJoinStyle join_style)
464 {
465   switch (join_style)
466     {
467 #define CASE(x) case GDK_JOIN_##x: return #x
468     CASE (MITER);
469     CASE (ROUND);
470     CASE (BEVEL);
471 #undef CASE
472     default: return static_printf ("illegal_%d", join_style);
473     }
474   /* NOTREACHED */
475   return NULL; 
476 }
477
478 gchar *
479 _gdk_win32_line_style_to_string (GdkLineStyle line_style)
480 {
481   switch (line_style)
482     {
483 #define CASE(x) case GDK_LINE_##x: return #x
484     CASE(SOLID);
485     CASE(ON_OFF_DASH);  
486     CASE(DOUBLE_DASH);  
487 #undef CASE
488     default: return static_printf ("illegal_%d", line_style);
489     }
490   /* NOTREACHED */
491   return NULL; 
492 }
493
494 gchar *
495 _gdk_win32_gcvalues_mask_to_string (GdkGCValuesMask mask)
496 {
497   gchar buf[400];
498   gchar *bufp = buf;
499   gchar *s = "";
500
501   buf[0] = '\0';
502
503 #define BIT(x)                                          \
504   if (mask & GDK_GC_##x)                                \
505     (bufp += g_sprintf (bufp, "%s" #x, s), s = "|")
506
507   BIT (FOREGROUND);
508   BIT (BACKGROUND);
509   BIT (FONT);
510   BIT (FUNCTION);
511   BIT (FILL);
512   BIT (TILE);
513   BIT (STIPPLE);
514   BIT (CLIP_MASK);
515   BIT (SUBWINDOW);
516   BIT (TS_X_ORIGIN);
517   BIT (TS_Y_ORIGIN);
518   BIT (CLIP_X_ORIGIN);
519   BIT (CLIP_Y_ORIGIN);
520   BIT (EXPOSURES);
521   BIT (LINE_WIDTH);
522   BIT (LINE_STYLE);
523   BIT (CAP_STYLE);
524   BIT (JOIN_STYLE);
525 #undef BIT
526
527   return static_printf ("%s", buf);  
528 }
529
530 gchar *
531 _gdk_win32_window_state_to_string (GdkWindowState state)
532 {
533   gchar buf[100];
534   gchar *bufp = buf;
535   gchar *s = "";
536
537   buf[0] = '\0';
538
539 #define BIT(x)                                          \
540   if (state & GDK_WINDOW_STATE_ ## x)                   \
541     (bufp += sprintf (bufp, "%s" #x, s), s = "|")
542
543   /* For clarity, also show the complement of WITHDRAWN, i.e. "MAPPED" */
544   if (!(state & GDK_WINDOW_STATE_WITHDRAWN))
545     (bufp += sprintf (bufp, "MAPPED"), s = "|");
546
547   BIT (WITHDRAWN);
548   BIT (ICONIFIED);
549   BIT (MAXIMIZED);
550   BIT (STICKY);
551 #undef BIT
552
553   return static_printf ("%s", buf);  
554 }
555
556 gchar *
557 _gdk_win32_rop2_to_string (int rop2)
558 {
559   switch (rop2)
560     {
561 #define CASE(x) case R2_##x: return #x
562       CASE (BLACK);
563       CASE (COPYPEN);
564       CASE (MASKNOTPEN);
565       CASE (MASKPEN);
566       CASE (MASKPENNOT);
567       CASE (MERGENOTPEN);
568       CASE (MERGEPEN);
569       CASE (MERGEPENNOT);
570       CASE (NOP);
571       CASE (NOT);
572       CASE (NOTCOPYPEN);
573       CASE (NOTMASKPEN);
574       CASE (NOTMERGEPEN);
575       CASE (NOTXORPEN);
576       CASE (WHITE);
577       CASE (XORPEN);
578 #undef CASE
579     default: return static_printf ("illegal_%x", rop2);
580     }
581   /* NOTREACHED */
582   return NULL;
583 }
584
585 gchar *
586 _gdk_win32_lbstyle_to_string (UINT brush_style)
587 {
588   switch (brush_style)
589     {
590 #define CASE(x) case BS_##x: return #x
591       CASE (DIBPATTERN);
592       CASE (DIBPATTERNPT);
593       CASE (HATCHED);
594       CASE (HOLLOW);
595       CASE (PATTERN);
596       CASE (SOLID);
597 #undef CASE
598     default: return static_printf ("illegal_%d", brush_style);
599     }
600   /* NOTREACHED */
601   return NULL;
602 }
603
604 gchar *
605 _gdk_win32_pstype_to_string (DWORD pen_style)
606 {
607   switch (pen_style & PS_TYPE_MASK)
608     {
609     case PS_GEOMETRIC: return "GEOMETRIC";
610     case PS_COSMETIC: return "COSMETIC";
611     default: return static_printf ("illegal_%d", pen_style & PS_TYPE_MASK);
612     }
613   /* NOTREACHED */
614   return NULL;
615 }
616
617 gchar *
618 _gdk_win32_psstyle_to_string (DWORD pen_style)
619 {
620   switch (pen_style & PS_STYLE_MASK)
621     {
622 #define CASE(x) case PS_##x: return #x
623       CASE (DASH);
624       CASE (DASHDOT);
625       CASE (DASHDOTDOT);
626       CASE (DOT);
627       CASE (INSIDEFRAME);
628       CASE (NULL);
629       CASE (SOLID);
630       CASE (USERSTYLE);
631 #undef CASE
632     default: return static_printf ("illegal_%d", pen_style & PS_STYLE_MASK);
633     }
634   /* NOTREACHED */
635   return NULL;
636 }
637
638 gchar *
639 _gdk_win32_psendcap_to_string (DWORD pen_style)
640 {
641   switch (pen_style & PS_ENDCAP_MASK)
642     {
643 #define CASE(x) case PS_ENDCAP_##x: return #x
644       CASE (FLAT);
645       CASE (ROUND);
646       CASE (SQUARE);
647 #undef CASE
648     default: return static_printf ("illegal_%d", pen_style & PS_ENDCAP_MASK);
649     }
650   /* NOTREACHED */
651   return NULL;
652 }
653
654 gchar *
655 _gdk_win32_psjoin_to_string (DWORD pen_style)
656 {
657   switch (pen_style & PS_JOIN_MASK)
658     {
659 #define CASE(x) case PS_JOIN_##x: return #x
660       CASE (BEVEL);
661       CASE (MITER);
662       CASE (ROUND);
663 #undef CASE
664     default: return static_printf ("illegal_%d", pen_style & PS_JOIN_MASK);
665     }
666   /* NOTREACHED */
667   return NULL;
668 }
669
670 gchar *
671 _gdk_win32_message_to_string (UINT msg)
672 {
673   switch (msg)
674     {
675 #define CASE(x) case x: return #x
676       CASE (WM_NULL);
677       CASE (WM_CREATE);
678       CASE (WM_DESTROY);
679       CASE (WM_MOVE);
680       CASE (WM_SIZE);
681       CASE (WM_ACTIVATE);
682       CASE (WM_SETFOCUS);
683       CASE (WM_KILLFOCUS);
684       CASE (WM_ENABLE);
685       CASE (WM_SETREDRAW);
686       CASE (WM_SETTEXT);
687       CASE (WM_GETTEXT);
688       CASE (WM_GETTEXTLENGTH);
689       CASE (WM_PAINT);
690       CASE (WM_CLOSE);
691       CASE (WM_QUERYENDSESSION);
692       CASE (WM_QUERYOPEN);
693       CASE (WM_ENDSESSION);
694       CASE (WM_QUIT);
695       CASE (WM_ERASEBKGND);
696       CASE (WM_SYSCOLORCHANGE);
697       CASE (WM_SHOWWINDOW);
698       CASE (WM_WININICHANGE);
699       CASE (WM_DEVMODECHANGE);
700       CASE (WM_ACTIVATEAPP);
701       CASE (WM_FONTCHANGE);
702       CASE (WM_TIMECHANGE);
703       CASE (WM_CANCELMODE);
704       CASE (WM_SETCURSOR);
705       CASE (WM_MOUSEACTIVATE);
706       CASE (WM_CHILDACTIVATE);
707       CASE (WM_QUEUESYNC);
708       CASE (WM_GETMINMAXINFO);
709       CASE (WM_PAINTICON);
710       CASE (WM_ICONERASEBKGND);
711       CASE (WM_NEXTDLGCTL);
712       CASE (WM_SPOOLERSTATUS);
713       CASE (WM_DRAWITEM);
714       CASE (WM_MEASUREITEM);
715       CASE (WM_DELETEITEM);
716       CASE (WM_VKEYTOITEM);
717       CASE (WM_CHARTOITEM);
718       CASE (WM_SETFONT);
719       CASE (WM_GETFONT);
720       CASE (WM_SETHOTKEY);
721       CASE (WM_GETHOTKEY);
722       CASE (WM_QUERYDRAGICON);
723       CASE (WM_COMPAREITEM);
724       CASE (WM_GETOBJECT);
725       CASE (WM_COMPACTING);
726       CASE (WM_WINDOWPOSCHANGING);
727       CASE (WM_WINDOWPOSCHANGED);
728       CASE (WM_POWER);
729       CASE (WM_COPYDATA);
730       CASE (WM_CANCELJOURNAL);
731       CASE (WM_NOTIFY);
732       CASE (WM_INPUTLANGCHANGEREQUEST);
733       CASE (WM_INPUTLANGCHANGE);
734       CASE (WM_TCARD);
735       CASE (WM_HELP);
736       CASE (WM_USERCHANGED);
737       CASE (WM_NOTIFYFORMAT);
738       CASE (WM_CONTEXTMENU);
739       CASE (WM_STYLECHANGING);
740       CASE (WM_STYLECHANGED);
741       CASE (WM_DISPLAYCHANGE);
742       CASE (WM_GETICON);
743       CASE (WM_SETICON);
744       CASE (WM_NCCREATE);
745       CASE (WM_NCDESTROY);
746       CASE (WM_NCCALCSIZE);
747       CASE (WM_NCHITTEST);
748       CASE (WM_NCPAINT);
749       CASE (WM_NCACTIVATE);
750       CASE (WM_GETDLGCODE);
751       CASE (WM_SYNCPAINT);
752       CASE (WM_NCMOUSEMOVE);
753       CASE (WM_NCLBUTTONDOWN);
754       CASE (WM_NCLBUTTONUP);
755       CASE (WM_NCLBUTTONDBLCLK);
756       CASE (WM_NCRBUTTONDOWN);
757       CASE (WM_NCRBUTTONUP);
758       CASE (WM_NCRBUTTONDBLCLK);
759       CASE (WM_NCMBUTTONDOWN);
760       CASE (WM_NCMBUTTONUP);
761       CASE (WM_NCMBUTTONDBLCLK);
762       CASE (WM_NCXBUTTONDOWN);
763       CASE (WM_NCXBUTTONUP);
764       CASE (WM_NCXBUTTONDBLCLK);
765       CASE (WM_KEYDOWN);
766       CASE (WM_KEYUP);
767       CASE (WM_CHAR);
768       CASE (WM_DEADCHAR);
769       CASE (WM_SYSKEYDOWN);
770       CASE (WM_SYSKEYUP);
771       CASE (WM_SYSCHAR);
772       CASE (WM_SYSDEADCHAR);
773       CASE (WM_KEYLAST);
774       CASE (WM_IME_STARTCOMPOSITION);
775       CASE (WM_IME_ENDCOMPOSITION);
776       CASE (WM_IME_COMPOSITION);
777       CASE (WM_INITDIALOG);
778       CASE (WM_COMMAND);
779       CASE (WM_SYSCOMMAND);
780       CASE (WM_TIMER);
781       CASE (WM_HSCROLL);
782       CASE (WM_VSCROLL);
783       CASE (WM_INITMENU);
784       CASE (WM_INITMENUPOPUP);
785       CASE (WM_MENUSELECT);
786       CASE (WM_MENUCHAR);
787       CASE (WM_ENTERIDLE);
788       CASE (WM_MENURBUTTONUP);
789       CASE (WM_MENUDRAG);
790       CASE (WM_MENUGETOBJECT);
791       CASE (WM_UNINITMENUPOPUP);
792       CASE (WM_MENUCOMMAND);
793       CASE (WM_CHANGEUISTATE);
794       CASE (WM_UPDATEUISTATE);
795       CASE (WM_QUERYUISTATE);
796       CASE (WM_CTLCOLORMSGBOX);
797       CASE (WM_CTLCOLOREDIT);
798       CASE (WM_CTLCOLORLISTBOX);
799       CASE (WM_CTLCOLORBTN);
800       CASE (WM_CTLCOLORDLG);
801       CASE (WM_CTLCOLORSCROLLBAR);
802       CASE (WM_CTLCOLORSTATIC);
803       CASE (WM_MOUSEMOVE);
804       CASE (WM_LBUTTONDOWN);
805       CASE (WM_LBUTTONUP);
806       CASE (WM_LBUTTONDBLCLK);
807       CASE (WM_RBUTTONDOWN);
808       CASE (WM_RBUTTONUP);
809       CASE (WM_RBUTTONDBLCLK);
810       CASE (WM_MBUTTONDOWN);
811       CASE (WM_MBUTTONUP);
812       CASE (WM_MBUTTONDBLCLK);
813       CASE (WM_MOUSEWHEEL);
814       CASE (WM_XBUTTONDOWN);
815       CASE (WM_XBUTTONUP);
816       CASE (WM_XBUTTONDBLCLK);
817       CASE (WM_PARENTNOTIFY);
818       CASE (WM_ENTERMENULOOP);
819       CASE (WM_EXITMENULOOP);
820       CASE (WM_NEXTMENU);
821       CASE (WM_SIZING);
822       CASE (WM_CAPTURECHANGED);
823       CASE (WM_MOVING);
824       CASE (WM_POWERBROADCAST);
825       CASE (WM_DEVICECHANGE);
826       CASE (WM_MDICREATE);
827       CASE (WM_MDIDESTROY);
828       CASE (WM_MDIACTIVATE);
829       CASE (WM_MDIRESTORE);
830       CASE (WM_MDINEXT);
831       CASE (WM_MDIMAXIMIZE);
832       CASE (WM_MDITILE);
833       CASE (WM_MDICASCADE);
834       CASE (WM_MDIICONARRANGE);
835       CASE (WM_MDIGETACTIVE);
836       CASE (WM_MDISETMENU);
837       CASE (WM_ENTERSIZEMOVE);
838       CASE (WM_EXITSIZEMOVE);
839       CASE (WM_DROPFILES);
840       CASE (WM_MDIREFRESHMENU);
841       CASE (WM_IME_SETCONTEXT);
842       CASE (WM_IME_NOTIFY);
843       CASE (WM_IME_CONTROL);
844       CASE (WM_IME_COMPOSITIONFULL);
845       CASE (WM_IME_SELECT);
846       CASE (WM_IME_CHAR);
847       CASE (WM_IME_REQUEST);
848       CASE (WM_IME_KEYDOWN);
849       CASE (WM_IME_KEYUP);
850       CASE (WM_MOUSEHOVER);
851       CASE (WM_MOUSELEAVE);
852       CASE (WM_NCMOUSEHOVER);
853       CASE (WM_NCMOUSELEAVE);
854       CASE (WM_CUT);
855       CASE (WM_COPY);
856       CASE (WM_PASTE);
857       CASE (WM_CLEAR);
858       CASE (WM_UNDO);
859       CASE (WM_RENDERFORMAT);
860       CASE (WM_RENDERALLFORMATS);
861       CASE (WM_DESTROYCLIPBOARD);
862       CASE (WM_DRAWCLIPBOARD);
863       CASE (WM_PAINTCLIPBOARD);
864       CASE (WM_VSCROLLCLIPBOARD);
865       CASE (WM_SIZECLIPBOARD);
866       CASE (WM_ASKCBFORMATNAME);
867       CASE (WM_CHANGECBCHAIN);
868       CASE (WM_HSCROLLCLIPBOARD);
869       CASE (WM_QUERYNEWPALETTE);
870       CASE (WM_PALETTEISCHANGING);
871       CASE (WM_PALETTECHANGED);
872       CASE (WM_HOTKEY);
873       CASE (WM_PRINT);
874       CASE (WM_PRINTCLIENT);
875       CASE (WM_APPCOMMAND);
876       CASE (WM_HANDHELDFIRST);
877       CASE (WM_HANDHELDLAST);
878       CASE (WM_AFXFIRST);
879       CASE (WM_AFXLAST);
880       CASE (WM_PENWINFIRST);
881       CASE (WM_PENWINLAST);
882       CASE (WM_APP);
883 #ifdef HAVE_WINTAB
884       CASE (WT_PACKET);
885       CASE (WT_CSRCHANGE);
886       CASE (WT_PROXIMITY);
887 #endif
888 #undef CASE
889     default:
890       if (msg >= WM_HANDHELDFIRST && msg <= WM_HANDHELDLAST)
891         return static_printf ("WM_HANDHELDFIRST+%d", msg - WM_HANDHELDFIRST);
892       else if (msg >= WM_AFXFIRST && msg <= WM_AFXLAST)
893         return static_printf ("WM_AFXFIRST+%d", msg - WM_AFXFIRST);
894       else if (msg >= WM_PENWINFIRST && msg <= WM_PENWINLAST)
895         return static_printf ("WM_PENWINFIRST+%d", msg - WM_PENWINFIRST);
896       else if (msg >= WM_USER && msg <= 0x7FFF)
897         return static_printf ("WM_USER+%d", msg - WM_USER);
898       else if (msg >= 0xC000 && msg <= 0xFFFF)
899         return static_printf ("reg-%#x", msg);
900       else
901         return static_printf ("unk-%#x", msg);
902     }
903   /* NOTREACHED */
904   return NULL;
905 }
906
907 gchar *
908 _gdk_win32_key_to_string (LONG lParam)
909 {
910   char buf[100];
911   gchar *keyname_utf8;
912
913   if (GetKeyNameText (lParam, buf, sizeof (buf)) &&
914       (keyname_utf8 = g_locale_to_utf8 (buf, -1, NULL, NULL, NULL)) != NULL)
915     {
916       gchar *retval = static_printf ("%s", keyname_utf8);
917
918       g_free (keyname_utf8);
919
920       return retval;
921     }
922
923   return static_printf ("unk-%#lx", lParam);
924 }
925       
926
927 gchar *
928 _gdk_win32_rect_to_string (const RECT *rect)
929 {
930   return static_printf ("%ldx%ld@%+ld%+ld",
931                         (rect->right - rect->left), (rect->bottom - rect->top),
932                         rect->left, rect->top);
933 }
934
935 gchar *
936 _gdk_win32_gdkrectangle_to_string (const GdkRectangle *rect)
937 {
938   return static_printf ("%dx%d@%+d%+d",
939                         rect->width, rect->height,
940                         rect->x, rect->y);
941 }
942
943 gchar *
944 _gdk_win32_gdkregion_to_string (const GdkRegion *rgn)
945 {
946   return static_printf ("%dx%d@%+d%+d",
947                         (rgn->extents.x2 - rgn->extents.x1),
948                         (rgn->extents.y2 - rgn->extents.y1),
949                         rgn->extents.x1, rgn->extents.y1);
950 }
951
952 gchar *
953 _gdk_win32_drawable_description (GdkDrawable *d)
954 {
955   gint width, height, depth;
956
957   gdk_drawable_get_size (d, &width, &height);
958   depth = gdk_drawable_get_depth (d);
959
960   return static_printf
961     ("%s:%p:%dx%dx%d",
962      G_OBJECT_TYPE_NAME (d),
963      GDK_DRAWABLE_HANDLE (d),
964      width, height, depth);
965 }
966
967 #endif /* G_ENABLE_DEBUG */