]> Pileus Git - ~andy/gtk/blob - gdk/win32/gdkevents-win32.c
Plug same refcount leaks as in the X11 backend.
[~andy/gtk] / gdk / win32 / gdkevents-win32.c
1 /* GDK - The GIMP Drawing Kit
2  * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
3  * Copyright (C) 1998-1999 Tor Lillqvist
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Library 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  * Library General Public License for more details.
14  *
15  * You should have received a copy of the GNU Library 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-1999.  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 /* Cannot use TrackMouseEvent, as the stupid WM_MOUSELEAVE message
31  * doesn't tell us where the mouse has gone. Thus we cannot use it to
32  * generate a correct GdkNotifyType. Pity, as using TrackMouseEvent
33  * otherwise would make it possible to reliably generate
34  * GDK_LEAVE_NOTIFY events, which would help get rid of those pesky
35  * tooltips sometimes popping up in the wrong place.
36  */
37 /* define USE_TRACKMOUSEEVENT */
38
39 #include <stdio.h>
40
41 #include "gdk.h"
42 #include "gdkinternals.h"
43 #include "gdkprivate-win32.h"
44
45 #include "gdkkeysyms.h"
46
47 #include "gdkinputprivate.h"
48
49 #include <objbase.h>
50 #include <imm.h>
51
52 #ifdef HAVE_DIMM_H
53 #include <dimm.h>
54 #else
55 #include "surrogate-dimm.h"
56 #endif
57
58 #ifdef HAVE_WINTAB
59 #include <wintab.h>
60 #endif
61
62 #define PING() printf("%s: %d\n",__FILE__,__LINE__),fflush(stdout)
63
64 #define WINDOW_PRIVATE(wp) GDK_WINDOW_WIN32DATA (wp)
65
66 typedef struct _GdkIOClosure GdkIOClosure;
67 typedef struct _GdkEventPrivate GdkEventPrivate;
68
69 typedef enum
70 {
71   /* Following flag is set for events on the event queue during
72    * translation and cleared afterwards.
73    */
74   GDK_EVENT_PENDING = 1 << 0
75 } GdkEventFlags;
76
77 struct _GdkIOClosure
78 {
79   GdkInputFunction function;
80   GdkInputCondition condition;
81   GdkDestroyNotify notify;
82   gpointer data;
83 };
84
85 struct _GdkEventPrivate
86 {
87   GdkEvent event;
88   guint    flags;
89 };
90
91 /* 
92  * Private function declarations
93  */
94
95 static GdkFilterReturn
96                  gdk_event_apply_filters(MSG      *xevent,
97                                          GdkEvent *event,
98                                          GList    *filters);
99 static gboolean  gdk_event_translate    (GdkEvent *event, 
100                                          MSG      *xevent,
101                                          gboolean *ret_val_flagp,
102                                          gint     *ret_valp);
103 static gboolean  gdk_event_prepare      (gpointer  source_data, 
104                                          GTimeVal *current_time,
105                                          gint     *timeout,
106                                          gpointer  user_data);
107 static gboolean  gdk_event_check        (gpointer  source_data,
108                                          GTimeVal *current_time,
109                                          gpointer  user_data);
110 static gboolean  gdk_event_dispatch     (gpointer  source_data,
111                                          GTimeVal *current_time,
112                                          gpointer  user_data);
113
114 static void      gdk_synthesize_click   (GdkEvent     *event, 
115                                          gint          nclicks);
116
117 /* Private variable declarations
118  */
119
120 static GdkWindow *p_grab_window = NULL; /* Window that currently
121                                          * holds the pointer grab
122                                          */
123
124 static GdkWindow *k_grab_window = NULL; /* Window the holds the
125                                          * keyboard grab
126                                          */
127
128 static GList *client_filters;   /* Filters for client messages */
129
130 static gboolean p_grab_automatic;
131 static GdkEventMask p_grab_mask;
132 static gboolean p_grab_owner_events, k_grab_owner_events;
133 static HCURSOR p_grab_cursor;
134
135 static GList *client_filters;               /* Filters for client messages */
136
137 static GSourceFuncs event_funcs = {
138   gdk_event_prepare,
139   gdk_event_check,
140   gdk_event_dispatch,
141   (GDestroyNotify)g_free
142 };
143
144 GPollFD event_poll_fd;
145
146 static GdkWindow *curWnd = NULL;
147 static HWND active = NULL;
148 static gint curX, curY;
149 static gdouble curXroot, curYroot;
150 static UINT gdk_ping_msg;
151 static gboolean ignore_WM_CHAR = FALSE;
152 static gboolean is_AltGr_key = FALSE;
153
154 static IActiveIMMApp *paimmapp = NULL;
155 static IActiveIMMMessagePumpOwner *paimmmpo = NULL;
156
157 typedef BOOL (WINAPI *PFN_TrackMouseEvent) (LPTRACKMOUSEEVENT);
158 static PFN_TrackMouseEvent p_TrackMouseEvent = NULL;
159
160 static gboolean use_IME_COMPOSITION = FALSE;
161
162 LRESULT CALLBACK 
163 gdk_WindowProc (HWND hWnd,
164                 UINT message,
165                 WPARAM wParam,
166                 LPARAM lParam)
167 {
168   GdkEvent event;
169   GdkEvent *eventp;
170   MSG msg;
171   DWORD pos;
172   LRESULT lres;
173   gint ret_val;
174   gboolean ret_val_flag;
175
176   GDK_NOTE (EVENTS, g_print ("gdk_WindowProc: %#x %#.03x\n", hWnd, message));
177
178   msg.hwnd = hWnd;
179   msg.message = message;
180   msg.wParam = wParam;
181   msg.lParam = lParam;
182   msg.time = GetTickCount ();
183   pos = GetMessagePos ();
184   msg.pt.x = LOWORD (pos);
185   msg.pt.y = HIWORD (pos);
186
187   ((GdkEventPrivate *)&event)->flags |= GDK_EVENT_PENDING;
188   if (gdk_event_translate (&event, &msg, &ret_val_flag, &ret_val))
189     {
190       ((GdkEventPrivate *)&event)->flags &= ~GDK_EVENT_PENDING;
191 #if 1
192       if (event.any.type == GDK_CONFIGURE)
193         {
194           /* Compress configure events */
195           GList *list = gdk_queued_events;
196
197           while (list != NULL
198                  && (((GdkEvent *)list->data)->any.type != GDK_CONFIGURE
199                      || ((GdkEvent *)list->data)->any.window != event.any.window))
200             list = list->next;
201           if (list != NULL)
202             {
203               GDK_NOTE (EVENTS, g_print ("... compressing an CONFIGURE event\n"));
204
205               *((GdkEvent *)list->data) = event;
206               gdk_drawable_unref (event.any.window);
207               /* Wake up WaitMessage */
208               PostMessage (NULL, gdk_ping_msg, 0, 0);
209               return FALSE;
210             }
211         }
212       else if (event.any.type == GDK_EXPOSE)
213         {
214           /* Compress expose events */
215           GList *list = gdk_queued_events;
216
217           while (list != NULL
218                  && (((GdkEvent *)list->data)->any.type != GDK_EXPOSE
219                      || ((GdkEvent *)list->data)->any.window != event.any.window))
220             list = list->next;
221           if (list != NULL)
222             {
223               GdkRectangle u;
224
225               GDK_NOTE (EVENTS, g_print ("... compressing an EXPOSE event\n"));
226               gdk_rectangle_union (&event.expose.area,
227                                    &((GdkEvent *)list->data)->expose.area,
228                                    &u);
229               ((GdkEvent *)list->data)->expose.area = u;
230               gdk_drawable_unref (event.any.window);
231 #if 0
232               /* Wake up WaitMessage */
233               PostMessage (NULL, gdk_ping_msg, 0, 0);
234 #endif
235               return FALSE;
236             }
237         }
238 #endif
239       eventp = gdk_event_new ();
240       *eventp = event;
241
242       /* Philippe Colantoni <colanton@aris.ss.uci.edu> suggests this
243        * in order to handle events while opaque resizing neatly.  I
244        * don't want it as default. Set the
245        * GDK_EVENT_FUNC_FROM_WINDOW_PROC env var to get this
246        * behaviour.
247        */
248       if (gdk_event_func_from_window_proc && gdk_event_func)
249         {
250           GDK_THREADS_ENTER ();
251           
252           (*gdk_event_func) (eventp, gdk_event_data);
253           gdk_event_free (eventp);
254           
255           GDK_THREADS_LEAVE ();
256         }
257       else
258         {
259           gdk_event_queue_append (eventp);
260 #if 1
261           /* Wake up WaitMessage */
262           PostMessage (NULL, gdk_ping_msg, 0, 0);
263 #endif
264         }
265       
266       if (ret_val_flag)
267         return ret_val;
268       else
269         return FALSE;
270     }
271
272   if (ret_val_flag)
273     return ret_val;
274   else
275     {
276       if (paimmapp == NULL
277           || (*paimmapp->lpVtbl->OnDefWindowProc) (paimmapp, hWnd, message, wParam, lParam, &lres) == S_FALSE)
278         return DefWindowProc (hWnd, message, wParam, lParam);
279       else
280         return lres;
281     }
282 }
283
284 void 
285 gdk_events_init (void)
286 {
287   HRESULT hres;
288   HMODULE user32, imm32;
289   HINSTANCE commctrl32;
290
291   if (g_pipe_readable_msg == 0)
292     g_pipe_readable_msg = RegisterWindowMessage ("g-pipe-readable");
293   GDK_NOTE (EVENTS, g_print ("g-pipe-readable = %#.03x\n",
294                              g_pipe_readable_msg));
295
296   gdk_ping_msg = RegisterWindowMessage ("gdk-ping");
297   GDK_NOTE (EVENTS, g_print ("gdk-ping = %#.03x\n",
298                              gdk_ping_msg));
299
300   g_source_add (GDK_PRIORITY_EVENTS, TRUE, &event_funcs, NULL, NULL, NULL);
301
302   event_poll_fd.fd = G_WIN32_MSG_HANDLE;
303   event_poll_fd.events = G_IO_IN;
304   
305   g_main_add_poll (&event_poll_fd, GDK_PRIORITY_EVENTS);
306
307   hres = CoCreateInstance (&CLSID_CActiveIMM,
308                            NULL,
309                            CLSCTX_ALL,
310                            &IID_IActiveIMMApp,
311                            (LPVOID *) &paimmapp);
312   
313   if (hres == S_OK)
314     {
315       GDK_NOTE (EVENTS, g_print ("IActiveIMMApp created %#x\n",
316                                  paimmapp));
317       (*paimmapp->lpVtbl->Activate) (paimmapp, TRUE);
318       
319       hres = (*paimmapp->lpVtbl->QueryInterface) (paimmapp, &IID_IActiveIMMMessagePumpOwner, &paimmmpo);
320       GDK_NOTE (EVENTS, g_print ("IActiveIMMMessagePumpOwner created %#x\n",
321                                  paimmmpo));
322       (paimmmpo->lpVtbl->Start) (paimmmpo);
323     }
324
325 #ifdef USE_TRACKMOUSEEVENT
326   user32 = GetModuleHandle ("user32.dll");
327   if ((p_TrackMouseEvent = GetProcAddress (user32, "TrackMouseEvent")) == NULL)
328     {
329       if ((commctrl32 = LoadLibrary ("commctrl32.dll")) != NULL)
330         p_TrackMouseEvent = (PFN_TrackMouseEvent)
331           GetProcAddress (commctrl32, "_TrackMouseEvent");
332     }
333   if (p_TrackMouseEvent != NULL)
334     GDK_NOTE (EVENTS, g_print ("Using TrackMouseEvent to detect leave events\n"));
335 #endif
336   if (IS_WIN_NT (windows_version) && (windows_version & 0xFF) == 5)
337     {
338       /* On Win2k (Beta 3, at least) WM_IME_CHAR doesn't seem to work
339        * correctly for non-Unicode applications. Handle
340        * WM_IME_COMPOSITION with GCS_RESULTSTR instead, fetch the
341        * Unicode char from the IME with ImmGetCompositionStringW().
342        */
343       use_IME_COMPOSITION = TRUE;
344     }
345 }
346
347 /*
348  *--------------------------------------------------------------
349  * gdk_events_pending
350  *
351  *   Returns if events are pending on the queue.
352  *
353  * Arguments:
354  *
355  * Results:
356  *   Returns TRUE if events are pending
357  *
358  * Side effects:
359  *
360  *--------------------------------------------------------------
361  */
362
363 gboolean
364 gdk_events_pending (void)
365 {
366   MSG msg;
367
368   return (gdk_event_queue_find_first() || PeekMessage (&msg, NULL, 0, 0, PM_NOREMOVE));
369 }
370
371 /*
372  *--------------------------------------------------------------
373  * gdk_event_get_graphics_expose
374  *
375  *   Waits for a GraphicsExpose or NoExpose event
376  *
377  * Arguments:
378  *
379  * Results: 
380  *   For GraphicsExpose events, returns a pointer to the event
381  *   converted into a GdkEvent Otherwise, returns NULL.
382  *
383  * Side effects:
384  *
385  *-------------------------------------------------------------- */
386
387 GdkEvent*
388 gdk_event_get_graphics_expose (GdkWindow *window)
389 {
390   MSG xevent;
391   GdkEvent *event;
392
393   g_return_val_if_fail (window != NULL, NULL);
394   
395   GDK_NOTE (EVENTS, g_print ("gdk_event_get_graphics_expose\n"));
396
397 #if 0 /* ??? */
398   /* Some nasty bugs here, just return NULL for now. */
399   return NULL;
400 #else
401   if (PeekMessage (&xevent, GDK_DRAWABLE_XID (window), WM_PAINT, WM_PAINT, PM_REMOVE))
402     {
403       event = gdk_event_new ();
404       
405       if (gdk_event_translate (event, &xevent, NULL, NULL))
406         return event;
407       else
408         gdk_event_free (event);
409     }
410   
411   return NULL;  
412 #endif
413 }
414
415 static char *
416 event_mask_string (GdkEventMask mask)
417 {
418   static char bfr[500];
419   char *p = bfr;
420
421   *p = '\0';
422 #define BIT(x) \
423   if (mask & GDK_##x##_MASK) \
424     p += sprintf (p, "%s" #x, (p > bfr ? " " : ""))
425   BIT(EXPOSURE);
426   BIT(POINTER_MOTION);
427   BIT(POINTER_MOTION_HINT);
428   BIT(BUTTON_MOTION);
429   BIT(BUTTON1_MOTION);
430   BIT(BUTTON2_MOTION);
431   BIT(BUTTON3_MOTION);
432   BIT(BUTTON_PRESS);
433   BIT(BUTTON_RELEASE);
434   BIT(KEY_PRESS);
435   BIT(KEY_RELEASE);
436   BIT(ENTER_NOTIFY);
437   BIT(LEAVE_NOTIFY);
438   BIT(FOCUS_CHANGE);
439   BIT(STRUCTURE);
440   BIT(PROPERTY_CHANGE);
441   BIT(VISIBILITY_NOTIFY);
442   BIT(PROXIMITY_IN);
443   BIT(PROXIMITY_OUT);
444   BIT(SUBSTRUCTURE);
445   BIT(SCROLL);
446 #undef BIT
447
448   return bfr;
449 }
450
451 /*
452  *--------------------------------------------------------------
453  * gdk_pointer_grab
454  *
455  *   Grabs the pointer to a specific window
456  *
457  * Arguments:
458  *   "window" is the window which will receive the grab
459  *   "owner_events" specifies whether events will be reported as is,
460  *     or relative to "window"
461  *   "event_mask" masks only interesting events
462  *   "confine_to" limits the cursor movement to the specified window
463  *   "cursor" changes the cursor for the duration of the grab
464  *   "time" specifies the time
465  *
466  * Results:
467  *
468  * Side effects:
469  *   requires a corresponding call to gdk_pointer_ungrab
470  *
471  *--------------------------------------------------------------
472  */
473
474 gint
475 gdk_pointer_grab (GdkWindow *     window,
476                   gboolean        owner_events,
477                   GdkEventMask    event_mask,
478                   GdkWindow *     confine_to,
479                   GdkCursor *     cursor,
480                   guint32         time)
481 {
482   HWND xwindow;
483   HWND xconfine_to;
484   HCURSOR xcursor;
485   GdkCursorPrivate *cursor_private;
486   gint return_val;
487
488   g_return_val_if_fail (window != NULL, 0);
489   g_return_val_if_fail (GDK_IS_WINDOW (window), 0);
490   g_return_val_if_fail (confine_to == NULL || GDK_IS_WINDOW (confine_to), 0);
491   
492   cursor_private = (GdkCursorPrivate*) cursor;
493   
494   xwindow = GDK_DRAWABLE_XID (window);
495   
496   if (!confine_to || GDK_DRAWABLE_DESTROYED (confine_to))
497     xconfine_to = NULL;
498   else
499     xconfine_to = GDK_DRAWABLE_XID (confine_to);
500   
501   if (!cursor)
502     xcursor = NULL;
503   else
504     xcursor = cursor_private->xcursor;
505   
506   if (gdk_input_vtable.grab_pointer)
507     return_val = gdk_input_vtable.grab_pointer (window,
508                                                 owner_events,
509                                                 event_mask,
510                                                 confine_to,
511                                                 time);
512   else
513     return_val = Success;
514   
515   if (return_val == Success)
516     {
517       if (!GDK_DRAWABLE_DESTROYED (window))
518       {
519         GDK_NOTE (EVENTS, g_print ("gdk_pointer_grab: %#x %s %#x %s\n",
520                                    xwindow,
521                                    (owner_events ? "TRUE" : "FALSE"),
522                                    xcursor,
523                                    event_mask_string (event_mask)));
524         p_grab_mask = event_mask;
525         p_grab_owner_events = (owner_events != 0);
526         p_grab_automatic = FALSE;
527
528 #if 0 /* Menus don't work if we use mouse capture. Pity, because many other
529        * things work better with mouse capture.
530        */
531         SetCapture (xwindow);
532 #endif
533         return_val = GrabSuccess;
534       }
535       else
536         return_val = AlreadyGrabbed;
537     }
538   
539   if (return_val == GrabSuccess)
540     {
541       p_grab_window = window;
542       p_grab_cursor = xcursor;
543     }
544   
545   return return_val;
546 }
547
548 /*
549  *--------------------------------------------------------------
550  * gdk_pointer_ungrab
551  *
552  *   Releases any pointer grab
553  *
554  * Arguments:
555  *
556  * Results:
557  *
558  * Side effects:
559  *
560  *--------------------------------------------------------------
561  */
562
563 void
564 gdk_pointer_ungrab (guint32 time)
565 {
566   if (gdk_input_vtable.ungrab_pointer)
567     gdk_input_vtable.ungrab_pointer (time);
568 #if 0
569   if (GetCapture () != NULL)
570     ReleaseCapture ();
571 #endif
572   GDK_NOTE (EVENTS, g_print ("gdk_pointer_ungrab\n"));
573
574   p_grab_window = NULL;
575 }
576
577 /*
578  *--------------------------------------------------------------
579  * gdk_pointer_is_grabbed
580  *
581  *   Tell wether there is an active x pointer grab in effect
582  *
583  * Arguments:
584  *
585  * Results:
586  *
587  * Side effects:
588  *
589  *--------------------------------------------------------------
590  */
591
592 gboolean
593 gdk_pointer_is_grabbed (void)
594 {
595   return p_grab_window != NULL;
596 }
597
598 /*
599  *--------------------------------------------------------------
600  * gdk_keyboard_grab
601  *
602  *   Grabs the keyboard to a specific window
603  *
604  * Arguments:
605  *   "window" is the window which will receive the grab
606  *   "owner_events" specifies whether events will be reported as is,
607  *     or relative to "window"
608  *   "time" specifies the time
609  *
610  * Results:
611  *
612  * Side effects:
613  *   requires a corresponding call to gdk_keyboard_ungrab
614  *
615  *--------------------------------------------------------------
616  */
617
618 gint
619 gdk_keyboard_grab (GdkWindow *     window,
620                    gboolean        owner_events,
621                    guint32         time)
622 {
623   gint return_val;
624   
625   g_return_val_if_fail (window != NULL, 0);
626   g_return_val_if_fail (GDK_IS_WINDOW (window), 0);
627   
628   GDK_NOTE (EVENTS, g_print ("gdk_keyboard_grab %#x\n",
629                              GDK_DRAWABLE_XID (window)));
630
631   if (!GDK_DRAWABLE_DESTROYED (window))
632     {
633       k_grab_owner_events = owner_events != 0;
634       return_val = GrabSuccess;
635     }
636   else
637     return_val = AlreadyGrabbed;
638
639   if (return_val == GrabSuccess)
640     k_grab_window = window;
641   
642   return return_val;
643 }
644
645 /*
646  *--------------------------------------------------------------
647  * gdk_keyboard_ungrab
648  *
649  *   Releases any keyboard grab
650  *
651  * Arguments:
652  *
653  * Results:
654  *
655  * Side effects:
656  *
657  *--------------------------------------------------------------
658  */
659
660 void
661 gdk_keyboard_ungrab (guint32 time)
662 {
663   GDK_NOTE (EVENTS, g_print ("gdk_keyboard_ungrab\n"));
664
665   k_grab_window = NULL;
666 }
667
668 static void
669 gdk_io_destroy (gpointer data)
670 {
671   GdkIOClosure *closure = data;
672
673   if (closure->notify)
674     closure->notify (closure->data);
675
676   g_free (closure);
677 }
678
679 static gboolean  
680 gdk_io_invoke (GIOChannel   *source,
681                GIOCondition  condition,
682                gpointer      data)
683 {
684   GdkIOClosure *closure = data;
685   GdkInputCondition gdk_cond = 0;
686
687   if (condition & (G_IO_IN | G_IO_PRI))
688     gdk_cond |= GDK_INPUT_READ;
689   if (condition & G_IO_OUT)
690     gdk_cond |= GDK_INPUT_WRITE;
691   if (condition & (G_IO_ERR | G_IO_HUP | G_IO_NVAL))
692     gdk_cond |= GDK_INPUT_EXCEPTION;
693
694   if (closure->condition & gdk_cond)
695     closure->function (closure->data, g_io_channel_unix_get_fd (source), gdk_cond);
696
697   return TRUE;
698 }
699
700 static GdkFilterReturn
701 gdk_event_apply_filters (MSG      *xevent,
702                          GdkEvent *event,
703                          GList    *filters)
704 {
705   GdkEventFilter *filter;
706   GList *tmp_list;
707   GdkFilterReturn result;
708   
709   tmp_list = filters;
710   
711   while (tmp_list)
712     {
713       filter = (GdkEventFilter *) tmp_list->data;
714       
715       result = (*filter->function) (xevent, event, filter->data);
716       if (result !=  GDK_FILTER_CONTINUE)
717         return result;
718       
719       tmp_list = tmp_list->next;
720     }
721   
722   return GDK_FILTER_CONTINUE;
723 }
724
725 void 
726 gdk_add_client_message_filter (GdkAtom       message_type,
727                                GdkFilterFunc func,
728                                gpointer      data)
729 {
730   GdkClientFilter *filter = g_new (GdkClientFilter, 1);
731
732   filter->type = message_type;
733   filter->function = func;
734   filter->data = data;
735   
736   client_filters = g_list_prepend (client_filters, filter);
737 }
738
739 /* Thanks to Markus G. Kuhn <mkuhn@acm.org> for the ksysym<->Unicode
740  * mapping functions, from the xterm sources.
741  */
742
743 #if 0 /* Keyval-to-Unicode isn't actually needed  */
744
745 struct k2u {
746   unsigned short keysym;
747   unsigned short ucs;
748 } k2utab[] = {
749   { 0x01a1, 0x0104 }, /*                     Aogonek Ą LATIN CAPITAL LETTER A WITH OGONEK */
750   { 0x01a2, 0x02d8 }, /*                       breve ˘ BREVE */
751   { 0x01a3, 0x0141 }, /*                     Lstroke Ł LATIN CAPITAL LETTER L WITH STROKE */
752   { 0x01a5, 0x013d }, /*                      Lcaron Ľ LATIN CAPITAL LETTER L WITH CARON */
753   { 0x01a6, 0x015a }, /*                      Sacute Ś LATIN CAPITAL LETTER S WITH ACUTE */
754   { 0x01a9, 0x0160 }, /*                      Scaron Š LATIN CAPITAL LETTER S WITH CARON */
755   { 0x01aa, 0x015e }, /*                    Scedilla Ş LATIN CAPITAL LETTER S WITH CEDILLA */
756   { 0x01ab, 0x0164 }, /*                      Tcaron Ť LATIN CAPITAL LETTER T WITH CARON */
757   { 0x01ac, 0x0179 }, /*                      Zacute Ź LATIN CAPITAL LETTER Z WITH ACUTE */
758   { 0x01ae, 0x017d }, /*                      Zcaron Ž LATIN CAPITAL LETTER Z WITH CARON */
759   { 0x01af, 0x017b }, /*                   Zabovedot Ż LATIN CAPITAL LETTER Z WITH DOT ABOVE */
760   { 0x01b1, 0x0105 }, /*                     aogonek ą LATIN SMALL LETTER A WITH OGONEK */
761   { 0x01b2, 0x02db }, /*                      ogonek ˛ OGONEK */
762   { 0x01b3, 0x0142 }, /*                     lstroke ł LATIN SMALL LETTER L WITH STROKE */
763   { 0x01b5, 0x013e }, /*                      lcaron ľ LATIN SMALL LETTER L WITH CARON */
764   { 0x01b6, 0x015b }, /*                      sacute ś LATIN SMALL LETTER S WITH ACUTE */
765   { 0x01b7, 0x02c7 }, /*                       caron ˇ CARON */
766   { 0x01b9, 0x0161 }, /*                      scaron š LATIN SMALL LETTER S WITH CARON */
767   { 0x01ba, 0x015f }, /*                    scedilla ş LATIN SMALL LETTER S WITH CEDILLA */
768   { 0x01bb, 0x0165 }, /*                      tcaron ť LATIN SMALL LETTER T WITH CARON */
769   { 0x01bc, 0x017a }, /*                      zacute ź LATIN SMALL LETTER Z WITH ACUTE */
770   { 0x01bd, 0x02dd }, /*                 doubleacute ˝ DOUBLE ACUTE ACCENT */
771   { 0x01be, 0x017e }, /*                      zcaron ž LATIN SMALL LETTER Z WITH CARON */
772   { 0x01bf, 0x017c }, /*                   zabovedot ż LATIN SMALL LETTER Z WITH DOT ABOVE */
773   { 0x01c0, 0x0154 }, /*                      Racute Ŕ LATIN CAPITAL LETTER R WITH ACUTE */
774   { 0x01c3, 0x0102 }, /*                      Abreve Ă LATIN CAPITAL LETTER A WITH BREVE */
775   { 0x01c5, 0x0139 }, /*                      Lacute Ĺ LATIN CAPITAL LETTER L WITH ACUTE */
776   { 0x01c6, 0x0106 }, /*                      Cacute Ć LATIN CAPITAL LETTER C WITH ACUTE */
777   { 0x01c8, 0x010c }, /*                      Ccaron Č LATIN CAPITAL LETTER C WITH CARON */
778   { 0x01ca, 0x0118 }, /*                     Eogonek Ę LATIN CAPITAL LETTER E WITH OGONEK */
779   { 0x01cc, 0x011a }, /*                      Ecaron Ě LATIN CAPITAL LETTER E WITH CARON */
780   { 0x01cf, 0x010e }, /*                      Dcaron Ď LATIN CAPITAL LETTER D WITH CARON */
781   { 0x01d0, 0x0110 }, /*                     Dstroke Đ LATIN CAPITAL LETTER D WITH STROKE */
782   { 0x01d1, 0x0143 }, /*                      Nacute Ń LATIN CAPITAL LETTER N WITH ACUTE */
783   { 0x01d2, 0x0147 }, /*                      Ncaron Ň LATIN CAPITAL LETTER N WITH CARON */
784   { 0x01d5, 0x0150 }, /*                Odoubleacute Ő LATIN CAPITAL LETTER O WITH DOUBLE ACUTE */
785   { 0x01d8, 0x0158 }, /*                      Rcaron Ř LATIN CAPITAL LETTER R WITH CARON */
786   { 0x01d9, 0x016e }, /*                       Uring Ů LATIN CAPITAL LETTER U WITH RING ABOVE */
787   { 0x01db, 0x0170 }, /*                Udoubleacute Ű LATIN CAPITAL LETTER U WITH DOUBLE ACUTE */
788   { 0x01de, 0x0162 }, /*                    Tcedilla Ţ LATIN CAPITAL LETTER T WITH CEDILLA */
789   { 0x01e0, 0x0155 }, /*                      racute ŕ LATIN SMALL LETTER R WITH ACUTE */
790   { 0x01e3, 0x0103 }, /*                      abreve ă LATIN SMALL LETTER A WITH BREVE */
791   { 0x01e5, 0x013a }, /*                      lacute ĺ LATIN SMALL LETTER L WITH ACUTE */
792   { 0x01e6, 0x0107 }, /*                      cacute ć LATIN SMALL LETTER C WITH ACUTE */
793   { 0x01e8, 0x010d }, /*                      ccaron č LATIN SMALL LETTER C WITH CARON */
794   { 0x01ea, 0x0119 }, /*                     eogonek ę LATIN SMALL LETTER E WITH OGONEK */
795   { 0x01ec, 0x011b }, /*                      ecaron ě LATIN SMALL LETTER E WITH CARON */
796   { 0x01ef, 0x010f }, /*                      dcaron ď LATIN SMALL LETTER D WITH CARON */
797   { 0x01f0, 0x0111 }, /*                     dstroke đ LATIN SMALL LETTER D WITH STROKE */
798   { 0x01f1, 0x0144 }, /*                      nacute ń LATIN SMALL LETTER N WITH ACUTE */
799   { 0x01f2, 0x0148 }, /*                      ncaron ň LATIN SMALL LETTER N WITH CARON */
800   { 0x01f5, 0x0151 }, /*                odoubleacute ő LATIN SMALL LETTER O WITH DOUBLE ACUTE */
801   { 0x01f8, 0x0159 }, /*                      rcaron ř LATIN SMALL LETTER R WITH CARON */
802   { 0x01f9, 0x016f }, /*                       uring ů LATIN SMALL LETTER U WITH RING ABOVE */
803   { 0x01fb, 0x0171 }, /*                udoubleacute ű LATIN SMALL LETTER U WITH DOUBLE ACUTE */
804   { 0x01fe, 0x0163 }, /*                    tcedilla ţ LATIN SMALL LETTER T WITH CEDILLA */
805   { 0x01ff, 0x02d9 }, /*                    abovedot ˙ DOT ABOVE */
806   { 0x02a1, 0x0126 }, /*                     Hstroke Ħ LATIN CAPITAL LETTER H WITH STROKE */
807   { 0x02a6, 0x0124 }, /*                 Hcircumflex Ĥ LATIN CAPITAL LETTER H WITH CIRCUMFLEX */
808   { 0x02a9, 0x0130 }, /*                   Iabovedot İ LATIN CAPITAL LETTER I WITH DOT ABOVE */
809   { 0x02ab, 0x011e }, /*                      Gbreve Ğ LATIN CAPITAL LETTER G WITH BREVE */
810   { 0x02ac, 0x0134 }, /*                 Jcircumflex Ĵ LATIN CAPITAL LETTER J WITH CIRCUMFLEX */
811   { 0x02b1, 0x0127 }, /*                     hstroke ħ LATIN SMALL LETTER H WITH STROKE */
812   { 0x02b6, 0x0125 }, /*                 hcircumflex ĥ LATIN SMALL LETTER H WITH CIRCUMFLEX */
813   { 0x02b9, 0x0131 }, /*                    idotless ı LATIN SMALL LETTER DOTLESS I */
814   { 0x02bb, 0x011f }, /*                      gbreve ğ LATIN SMALL LETTER G WITH BREVE */
815   { 0x02bc, 0x0135 }, /*                 jcircumflex ĵ LATIN SMALL LETTER J WITH CIRCUMFLEX */
816   { 0x02c5, 0x010a }, /*                   Cabovedot Ċ LATIN CAPITAL LETTER C WITH DOT ABOVE */
817   { 0x02c6, 0x0108 }, /*                 Ccircumflex Ĉ LATIN CAPITAL LETTER C WITH CIRCUMFLEX */
818   { 0x02d5, 0x0120 }, /*                   Gabovedot Ġ LATIN CAPITAL LETTER G WITH DOT ABOVE */
819   { 0x02d8, 0x011c }, /*                 Gcircumflex Ĝ LATIN CAPITAL LETTER G WITH CIRCUMFLEX */
820   { 0x02dd, 0x016c }, /*                      Ubreve Ŭ LATIN CAPITAL LETTER U WITH BREVE */
821   { 0x02de, 0x015c }, /*                 Scircumflex Ŝ LATIN CAPITAL LETTER S WITH CIRCUMFLEX */
822   { 0x02e5, 0x010b }, /*                   cabovedot ċ LATIN SMALL LETTER C WITH DOT ABOVE */
823   { 0x02e6, 0x0109 }, /*                 ccircumflex ĉ LATIN SMALL LETTER C WITH CIRCUMFLEX */
824   { 0x02f5, 0x0121 }, /*                   gabovedot ġ LATIN SMALL LETTER G WITH DOT ABOVE */
825   { 0x02f8, 0x011d }, /*                 gcircumflex ĝ LATIN SMALL LETTER G WITH CIRCUMFLEX */
826   { 0x02fd, 0x016d }, /*                      ubreve ŭ LATIN SMALL LETTER U WITH BREVE */
827   { 0x02fe, 0x015d }, /*                 scircumflex ŝ LATIN SMALL LETTER S WITH CIRCUMFLEX */
828   { 0x03a2, 0x0138 }, /*                         kra ĸ LATIN SMALL LETTER KRA */
829   { 0x03a3, 0x0156 }, /*                    Rcedilla Ŗ LATIN CAPITAL LETTER R WITH CEDILLA */
830   { 0x03a5, 0x0128 }, /*                      Itilde Ĩ LATIN CAPITAL LETTER I WITH TILDE */
831   { 0x03a6, 0x013b }, /*                    Lcedilla Ļ LATIN CAPITAL LETTER L WITH CEDILLA */
832   { 0x03aa, 0x0112 }, /*                     Emacron Ē LATIN CAPITAL LETTER E WITH MACRON */
833   { 0x03ab, 0x0122 }, /*                    Gcedilla Ģ LATIN CAPITAL LETTER G WITH CEDILLA */
834   { 0x03ac, 0x0166 }, /*                      Tslash Ŧ LATIN CAPITAL LETTER T WITH STROKE */
835   { 0x03b3, 0x0157 }, /*                    rcedilla ŗ LATIN SMALL LETTER R WITH CEDILLA */
836   { 0x03b5, 0x0129 }, /*                      itilde ĩ LATIN SMALL LETTER I WITH TILDE */
837   { 0x03b6, 0x013c }, /*                    lcedilla ļ LATIN SMALL LETTER L WITH CEDILLA */
838   { 0x03ba, 0x0113 }, /*                     emacron ē LATIN SMALL LETTER E WITH MACRON */
839   { 0x03bb, 0x0123 }, /*                    gcedilla ģ LATIN SMALL LETTER G WITH CEDILLA */
840   { 0x03bc, 0x0167 }, /*                      tslash ŧ LATIN SMALL LETTER T WITH STROKE */
841   { 0x03bd, 0x014a }, /*                         ENG Ŋ LATIN CAPITAL LETTER ENG */
842   { 0x03bf, 0x014b }, /*                         eng ŋ LATIN SMALL LETTER ENG */
843   { 0x03c0, 0x0100 }, /*                     Amacron Ā LATIN CAPITAL LETTER A WITH MACRON */
844   { 0x03c7, 0x012e }, /*                     Iogonek Į LATIN CAPITAL LETTER I WITH OGONEK */
845   { 0x03cc, 0x0116 }, /*                   Eabovedot Ė LATIN CAPITAL LETTER E WITH DOT ABOVE */
846   { 0x03cf, 0x012a }, /*                     Imacron Ī LATIN CAPITAL LETTER I WITH MACRON */
847   { 0x03d1, 0x0145 }, /*                    Ncedilla Ņ LATIN CAPITAL LETTER N WITH CEDILLA */
848   { 0x03d2, 0x014c }, /*                     Omacron Ō LATIN CAPITAL LETTER O WITH MACRON */
849   { 0x03d3, 0x0136 }, /*                    Kcedilla Ķ LATIN CAPITAL LETTER K WITH CEDILLA */
850   { 0x03d9, 0x0172 }, /*                     Uogonek Ų LATIN CAPITAL LETTER U WITH OGONEK */
851   { 0x03dd, 0x0168 }, /*                      Utilde Ũ LATIN CAPITAL LETTER U WITH TILDE */
852   { 0x03de, 0x016a }, /*                     Umacron Ū LATIN CAPITAL LETTER U WITH MACRON */
853   { 0x03e0, 0x0101 }, /*                     amacron ā LATIN SMALL LETTER A WITH MACRON */
854   { 0x03e7, 0x012f }, /*                     iogonek į LATIN SMALL LETTER I WITH OGONEK */
855   { 0x03ec, 0x0117 }, /*                   eabovedot ė LATIN SMALL LETTER E WITH DOT ABOVE */
856   { 0x03ef, 0x012b }, /*                     imacron ī LATIN SMALL LETTER I WITH MACRON */
857   { 0x03f1, 0x0146 }, /*                    ncedilla ņ LATIN SMALL LETTER N WITH CEDILLA */
858   { 0x03f2, 0x014d }, /*                     omacron ō LATIN SMALL LETTER O WITH MACRON */
859   { 0x03f3, 0x0137 }, /*                    kcedilla ķ LATIN SMALL LETTER K WITH CEDILLA */
860   { 0x03f9, 0x0173 }, /*                     uogonek ų LATIN SMALL LETTER U WITH OGONEK */
861   { 0x03fd, 0x0169 }, /*                      utilde ũ LATIN SMALL LETTER U WITH TILDE */
862   { 0x03fe, 0x016b }, /*                     umacron ū LATIN SMALL LETTER U WITH MACRON */
863   { 0x047e, 0x203e }, /*                    overline ‾ OVERLINE */
864   { 0x04a1, 0x3002 }, /*               kana_fullstop 。 IDEOGRAPHIC FULL STOP */
865   { 0x04a2, 0x300c }, /*         kana_openingbracket 「 LEFT CORNER BRACKET */
866   { 0x04a3, 0x300d }, /*         kana_closingbracket 」 RIGHT CORNER BRACKET */
867   { 0x04a4, 0x3001 }, /*                  kana_comma 、 IDEOGRAPHIC COMMA */
868   { 0x04a5, 0x30fb }, /*            kana_conjunctive ・ KATAKANA MIDDLE DOT */
869   { 0x04a6, 0x30f2 }, /*                     kana_WO ヲ KATAKANA LETTER WO */
870   { 0x04a7, 0x30a1 }, /*                      kana_a ァ KATAKANA LETTER SMALL A */
871   { 0x04a8, 0x30a3 }, /*                      kana_i ィ KATAKANA LETTER SMALL I */
872   { 0x04a9, 0x30a5 }, /*                      kana_u ゥ KATAKANA LETTER SMALL U */
873   { 0x04aa, 0x30a7 }, /*                      kana_e ェ KATAKANA LETTER SMALL E */
874   { 0x04ab, 0x30a9 }, /*                      kana_o ォ KATAKANA LETTER SMALL O */
875   { 0x04ac, 0x30e3 }, /*                     kana_ya ャ KATAKANA LETTER SMALL YA */
876   { 0x04ad, 0x30e5 }, /*                     kana_yu ュ KATAKANA LETTER SMALL YU */
877   { 0x04ae, 0x30e7 }, /*                     kana_yo ョ KATAKANA LETTER SMALL YO */
878   { 0x04af, 0x30c3 }, /*                    kana_tsu ッ KATAKANA LETTER SMALL TU */
879   { 0x04b0, 0x30fc }, /*              prolongedsound ー KATAKANA-HIRAGANA PROLONGED SOUND MARK */
880   { 0x04b1, 0x30a2 }, /*                      kana_A ア KATAKANA LETTER A */
881   { 0x04b2, 0x30a4 }, /*                      kana_I イ KATAKANA LETTER I */
882   { 0x04b3, 0x30a6 }, /*                      kana_U ウ KATAKANA LETTER U */
883   { 0x04b4, 0x30a8 }, /*                      kana_E エ KATAKANA LETTER E */
884   { 0x04b5, 0x30aa }, /*                      kana_O オ KATAKANA LETTER O */
885   { 0x04b6, 0x30ab }, /*                     kana_KA カ KATAKANA LETTER KA */
886   { 0x04b7, 0x30ad }, /*                     kana_KI キ KATAKANA LETTER KI */
887   { 0x04b8, 0x30af }, /*                     kana_KU ク KATAKANA LETTER KU */
888   { 0x04b9, 0x30b1 }, /*                     kana_KE ケ KATAKANA LETTER KE */
889   { 0x04ba, 0x30b3 }, /*                     kana_KO コ KATAKANA LETTER KO */
890   { 0x04bb, 0x30b5 }, /*                     kana_SA サ KATAKANA LETTER SA */
891   { 0x04bc, 0x30b7 }, /*                    kana_SHI シ KATAKANA LETTER SI */
892   { 0x04bd, 0x30b9 }, /*                     kana_SU ス KATAKANA LETTER SU */
893   { 0x04be, 0x30bb }, /*                     kana_SE セ KATAKANA LETTER SE */
894   { 0x04bf, 0x30bd }, /*                     kana_SO ソ KATAKANA LETTER SO */
895   { 0x04c0, 0x30bf }, /*                     kana_TA タ KATAKANA LETTER TA */
896   { 0x04c1, 0x30c1 }, /*                    kana_CHI チ KATAKANA LETTER TI */
897   { 0x04c2, 0x30c4 }, /*                    kana_TSU ツ KATAKANA LETTER TU */
898   { 0x04c3, 0x30c6 }, /*                     kana_TE テ KATAKANA LETTER TE */
899   { 0x04c4, 0x30c8 }, /*                     kana_TO ト KATAKANA LETTER TO */
900   { 0x04c5, 0x30ca }, /*                     kana_NA ナ KATAKANA LETTER NA */
901   { 0x04c6, 0x30cb }, /*                     kana_NI ニ KATAKANA LETTER NI */
902   { 0x04c7, 0x30cc }, /*                     kana_NU ヌ KATAKANA LETTER NU */
903   { 0x04c8, 0x30cd }, /*                     kana_NE ネ KATAKANA LETTER NE */
904   { 0x04c9, 0x30ce }, /*                     kana_NO ノ KATAKANA LETTER NO */
905   { 0x04ca, 0x30cf }, /*                     kana_HA ハ KATAKANA LETTER HA */
906   { 0x04cb, 0x30d2 }, /*                     kana_HI ヒ KATAKANA LETTER HI */
907   { 0x04cc, 0x30d5 }, /*                     kana_FU フ KATAKANA LETTER HU */
908   { 0x04cd, 0x30d8 }, /*                     kana_HE ヘ KATAKANA LETTER HE */
909   { 0x04ce, 0x30db }, /*                     kana_HO ホ KATAKANA LETTER HO */
910   { 0x04cf, 0x30de }, /*                     kana_MA マ KATAKANA LETTER MA */
911   { 0x04d0, 0x30df }, /*                     kana_MI ミ KATAKANA LETTER MI */
912   { 0x04d1, 0x30e0 }, /*                     kana_MU ム KATAKANA LETTER MU */
913   { 0x04d2, 0x30e1 }, /*                     kana_ME メ KATAKANA LETTER ME */
914   { 0x04d3, 0x30e2 }, /*                     kana_MO モ KATAKANA LETTER MO */
915   { 0x04d4, 0x30e4 }, /*                     kana_YA ヤ KATAKANA LETTER YA */
916   { 0x04d5, 0x30e6 }, /*                     kana_YU ユ KATAKANA LETTER YU */
917   { 0x04d6, 0x30e8 }, /*                     kana_YO ヨ KATAKANA LETTER YO */
918   { 0x04d7, 0x30e9 }, /*                     kana_RA ラ KATAKANA LETTER RA */
919   { 0x04d8, 0x30ea }, /*                     kana_RI リ KATAKANA LETTER RI */
920   { 0x04d9, 0x30eb }, /*                     kana_RU ル KATAKANA LETTER RU */
921   { 0x04da, 0x30ec }, /*                     kana_RE レ KATAKANA LETTER RE */
922   { 0x04db, 0x30ed }, /*                     kana_RO ロ KATAKANA LETTER RO */
923   { 0x04dc, 0x30ef }, /*                     kana_WA ワ KATAKANA LETTER WA */
924   { 0x04dd, 0x30f3 }, /*                      kana_N ン KATAKANA LETTER N */
925   { 0x04de, 0x309b }, /*                 voicedsound ゛ KATAKANA-HIRAGANA VOICED SOUND MARK */
926   { 0x04df, 0x309c }, /*             semivoicedsound ゜ KATAKANA-HIRAGANA SEMI-VOICED SOUND MARK */
927   { 0x05ac, 0x060c }, /*                Arabic_comma ، ARABIC COMMA */
928   { 0x05bb, 0x061b }, /*            Arabic_semicolon ؛ ARABIC SEMICOLON */
929   { 0x05bf, 0x061f }, /*        Arabic_question_mark ؟ ARABIC QUESTION MARK */
930   { 0x05c1, 0x0621 }, /*                Arabic_hamza ء ARABIC LETTER HAMZA */
931   { 0x05c2, 0x0622 }, /*          Arabic_maddaonalef آ ARABIC LETTER ALEF WITH MADDA ABOVE */
932   { 0x05c3, 0x0623 }, /*          Arabic_hamzaonalef أ ARABIC LETTER ALEF WITH HAMZA ABOVE */
933   { 0x05c4, 0x0624 }, /*           Arabic_hamzaonwaw ؤ ARABIC LETTER WAW WITH HAMZA ABOVE */
934   { 0x05c5, 0x0625 }, /*       Arabic_hamzaunderalef إ ARABIC LETTER ALEF WITH HAMZA BELOW */
935   { 0x05c6, 0x0626 }, /*           Arabic_hamzaonyeh ئ ARABIC LETTER YEH WITH HAMZA ABOVE */
936   { 0x05c7, 0x0627 }, /*                 Arabic_alef ا ARABIC LETTER ALEF */
937   { 0x05c8, 0x0628 }, /*                  Arabic_beh ب ARABIC LETTER BEH */
938   { 0x05c9, 0x0629 }, /*           Arabic_tehmarbuta ة ARABIC LETTER TEH MARBUTA */
939   { 0x05ca, 0x062a }, /*                  Arabic_teh ت ARABIC LETTER TEH */
940   { 0x05cb, 0x062b }, /*                 Arabic_theh ث ARABIC LETTER THEH */
941   { 0x05cc, 0x062c }, /*                 Arabic_jeem ج ARABIC LETTER JEEM */
942   { 0x05cd, 0x062d }, /*                  Arabic_hah ح ARABIC LETTER HAH */
943   { 0x05ce, 0x062e }, /*                 Arabic_khah خ ARABIC LETTER KHAH */
944   { 0x05cf, 0x062f }, /*                  Arabic_dal د ARABIC LETTER DAL */
945   { 0x05d0, 0x0630 }, /*                 Arabic_thal ذ ARABIC LETTER THAL */
946   { 0x05d1, 0x0631 }, /*                   Arabic_ra ر ARABIC LETTER REH */
947   { 0x05d2, 0x0632 }, /*                 Arabic_zain ز ARABIC LETTER ZAIN */
948   { 0x05d3, 0x0633 }, /*                 Arabic_seen س ARABIC LETTER SEEN */
949   { 0x05d4, 0x0634 }, /*                Arabic_sheen ش ARABIC LETTER SHEEN */
950   { 0x05d5, 0x0635 }, /*                  Arabic_sad ص ARABIC LETTER SAD */
951   { 0x05d6, 0x0636 }, /*                  Arabic_dad ض ARABIC LETTER DAD */
952   { 0x05d7, 0x0637 }, /*                  Arabic_tah ط ARABIC LETTER TAH */
953   { 0x05d8, 0x0638 }, /*                  Arabic_zah ظ ARABIC LETTER ZAH */
954   { 0x05d9, 0x0639 }, /*                  Arabic_ain ع ARABIC LETTER AIN */
955   { 0x05da, 0x063a }, /*                Arabic_ghain غ ARABIC LETTER GHAIN */
956   { 0x05e0, 0x0640 }, /*              Arabic_tatweel ـ ARABIC TATWEEL */
957   { 0x05e1, 0x0641 }, /*                  Arabic_feh ف ARABIC LETTER FEH */
958   { 0x05e2, 0x0642 }, /*                  Arabic_qaf ق ARABIC LETTER QAF */
959   { 0x05e3, 0x0643 }, /*                  Arabic_kaf ك ARABIC LETTER KAF */
960   { 0x05e4, 0x0644 }, /*                  Arabic_lam ل ARABIC LETTER LAM */
961   { 0x05e5, 0x0645 }, /*                 Arabic_meem م ARABIC LETTER MEEM */
962   { 0x05e6, 0x0646 }, /*                 Arabic_noon ن ARABIC LETTER NOON */
963   { 0x05e7, 0x0647 }, /*                   Arabic_ha ه ARABIC LETTER HEH */
964   { 0x05e8, 0x0648 }, /*                  Arabic_waw و ARABIC LETTER WAW */
965   { 0x05e9, 0x0649 }, /*          Arabic_alefmaksura ى ARABIC LETTER ALEF MAKSURA */
966   { 0x05ea, 0x064a }, /*                  Arabic_yeh ي ARABIC LETTER YEH */
967   { 0x05eb, 0x064b }, /*             Arabic_fathatan ً ARABIC FATHATAN */
968   { 0x05ec, 0x064c }, /*             Arabic_dammatan ٌ ARABIC DAMMATAN */
969   { 0x05ed, 0x064d }, /*             Arabic_kasratan ٍ ARABIC KASRATAN */
970   { 0x05ee, 0x064e }, /*                Arabic_fatha َ ARABIC FATHA */
971   { 0x05ef, 0x064f }, /*                Arabic_damma ُ ARABIC DAMMA */
972   { 0x05f0, 0x0650 }, /*                Arabic_kasra ِ ARABIC KASRA */
973   { 0x05f1, 0x0651 }, /*               Arabic_shadda ّ ARABIC SHADDA */
974   { 0x05f2, 0x0652 }, /*                Arabic_sukun ْ ARABIC SUKUN */
975   { 0x06a1, 0x0452 }, /*                 Serbian_dje ђ CYRILLIC SMALL LETTER DJE */
976   { 0x06a2, 0x0453 }, /*               Macedonia_gje ѓ CYRILLIC SMALL LETTER GJE */
977   { 0x06a3, 0x0451 }, /*                 Cyrillic_io ё CYRILLIC SMALL LETTER IO */
978   { 0x06a4, 0x0454 }, /*                Ukrainian_ie є CYRILLIC SMALL LETTER UKRAINIAN IE */
979   { 0x06a5, 0x0455 }, /*               Macedonia_dse ѕ CYRILLIC SMALL LETTER DZE */
980   { 0x06a6, 0x0456 }, /*                 Ukrainian_i і CYRILLIC SMALL LETTER BYELORUSSIAN-UKRAINIAN I */
981   { 0x06a7, 0x0457 }, /*                Ukrainian_yi ї CYRILLIC SMALL LETTER YI */
982   { 0x06a8, 0x0458 }, /*                 Cyrillic_je ј CYRILLIC SMALL LETTER JE */
983   { 0x06a9, 0x0459 }, /*                Cyrillic_lje љ CYRILLIC SMALL LETTER LJE */
984   { 0x06aa, 0x045a }, /*                Cyrillic_nje њ CYRILLIC SMALL LETTER NJE */
985   { 0x06ab, 0x045b }, /*                Serbian_tshe ћ CYRILLIC SMALL LETTER TSHE */
986   { 0x06ac, 0x045c }, /*               Macedonia_kje ќ CYRILLIC SMALL LETTER KJE */
987   { 0x06ae, 0x045e }, /*         Byelorussian_shortu ў CYRILLIC SMALL LETTER SHORT U */
988   { 0x06af, 0x045f }, /*               Cyrillic_dzhe џ CYRILLIC SMALL LETTER DZHE */
989   { 0x06b0, 0x2116 }, /*                  numerosign № NUMERO SIGN */
990   { 0x06b1, 0x0402 }, /*                 Serbian_DJE Ђ CYRILLIC CAPITAL LETTER DJE */
991   { 0x06b2, 0x0403 }, /*               Macedonia_GJE Ѓ CYRILLIC CAPITAL LETTER GJE */
992   { 0x06b3, 0x0401 }, /*                 Cyrillic_IO Ё CYRILLIC CAPITAL LETTER IO */
993   { 0x06b4, 0x0404 }, /*                Ukrainian_IE Є CYRILLIC CAPITAL LETTER UKRAINIAN IE */
994   { 0x06b5, 0x0405 }, /*               Macedonia_DSE Ѕ CYRILLIC CAPITAL LETTER DZE */
995   { 0x06b6, 0x0406 }, /*                 Ukrainian_I І CYRILLIC CAPITAL LETTER BYELORUSSIAN-UKRAINIAN I */
996   { 0x06b7, 0x0407 }, /*                Ukrainian_YI Ї CYRILLIC CAPITAL LETTER YI */
997   { 0x06b8, 0x0408 }, /*                 Cyrillic_JE Ј CYRILLIC CAPITAL LETTER JE */
998   { 0x06b9, 0x0409 }, /*                Cyrillic_LJE Љ CYRILLIC CAPITAL LETTER LJE */
999   { 0x06ba, 0x040a }, /*                Cyrillic_NJE Њ CYRILLIC CAPITAL LETTER NJE */
1000   { 0x06bb, 0x040b }, /*                Serbian_TSHE Ћ CYRILLIC CAPITAL LETTER TSHE */
1001   { 0x06bc, 0x040c }, /*               Macedonia_KJE Ќ CYRILLIC CAPITAL LETTER KJE */
1002   { 0x06be, 0x040e }, /*         Byelorussian_SHORTU Ў CYRILLIC CAPITAL LETTER SHORT U */
1003   { 0x06bf, 0x040f }, /*               Cyrillic_DZHE Џ CYRILLIC CAPITAL LETTER DZHE */
1004   { 0x06c0, 0x044e }, /*                 Cyrillic_yu ю CYRILLIC SMALL LETTER YU */
1005   { 0x06c1, 0x0430 }, /*                  Cyrillic_a а CYRILLIC SMALL LETTER A */
1006   { 0x06c2, 0x0431 }, /*                 Cyrillic_be б CYRILLIC SMALL LETTER BE */
1007   { 0x06c3, 0x0446 }, /*                Cyrillic_tse ц CYRILLIC SMALL LETTER TSE */
1008   { 0x06c4, 0x0434 }, /*                 Cyrillic_de д CYRILLIC SMALL LETTER DE */
1009   { 0x06c5, 0x0435 }, /*                 Cyrillic_ie е CYRILLIC SMALL LETTER IE */
1010   { 0x06c6, 0x0444 }, /*                 Cyrillic_ef ф CYRILLIC SMALL LETTER EF */
1011   { 0x06c7, 0x0433 }, /*                Cyrillic_ghe г CYRILLIC SMALL LETTER GHE */
1012   { 0x06c8, 0x0445 }, /*                 Cyrillic_ha х CYRILLIC SMALL LETTER HA */
1013   { 0x06c9, 0x0438 }, /*                  Cyrillic_i и CYRILLIC SMALL LETTER I */
1014   { 0x06ca, 0x0439 }, /*             Cyrillic_shorti й CYRILLIC SMALL LETTER SHORT I */
1015   { 0x06cb, 0x043a }, /*                 Cyrillic_ka к CYRILLIC SMALL LETTER KA */
1016   { 0x06cc, 0x043b }, /*                 Cyrillic_el л CYRILLIC SMALL LETTER EL */
1017   { 0x06cd, 0x043c }, /*                 Cyrillic_em м CYRILLIC SMALL LETTER EM */
1018   { 0x06ce, 0x043d }, /*                 Cyrillic_en н CYRILLIC SMALL LETTER EN */
1019   { 0x06cf, 0x043e }, /*                  Cyrillic_o о CYRILLIC SMALL LETTER O */
1020   { 0x06d0, 0x043f }, /*                 Cyrillic_pe п CYRILLIC SMALL LETTER PE */
1021   { 0x06d1, 0x044f }, /*                 Cyrillic_ya я CYRILLIC SMALL LETTER YA */
1022   { 0x06d2, 0x0440 }, /*                 Cyrillic_er р CYRILLIC SMALL LETTER ER */
1023   { 0x06d3, 0x0441 }, /*                 Cyrillic_es с CYRILLIC SMALL LETTER ES */
1024   { 0x06d4, 0x0442 }, /*                 Cyrillic_te т CYRILLIC SMALL LETTER TE */
1025   { 0x06d5, 0x0443 }, /*                  Cyrillic_u у CYRILLIC SMALL LETTER U */
1026   { 0x06d6, 0x0436 }, /*                Cyrillic_zhe ж CYRILLIC SMALL LETTER ZHE */
1027   { 0x06d7, 0x0432 }, /*                 Cyrillic_ve в CYRILLIC SMALL LETTER VE */
1028   { 0x06d8, 0x044c }, /*           Cyrillic_softsign ь CYRILLIC SMALL LETTER SOFT SIGN */
1029   { 0x06d9, 0x044b }, /*               Cyrillic_yeru ы CYRILLIC SMALL LETTER YERU */
1030   { 0x06da, 0x0437 }, /*                 Cyrillic_ze з CYRILLIC SMALL LETTER ZE */
1031   { 0x06db, 0x0448 }, /*                Cyrillic_sha ш CYRILLIC SMALL LETTER SHA */
1032   { 0x06dc, 0x044d }, /*                  Cyrillic_e э CYRILLIC SMALL LETTER E */
1033   { 0x06dd, 0x0449 }, /*              Cyrillic_shcha щ CYRILLIC SMALL LETTER SHCHA */
1034   { 0x06de, 0x0447 }, /*                Cyrillic_che ч CYRILLIC SMALL LETTER CHE */
1035   { 0x06df, 0x044a }, /*           Cyrillic_hardsign ъ CYRILLIC SMALL LETTER HARD SIGN */
1036   { 0x06e0, 0x042e }, /*                 Cyrillic_YU Ю CYRILLIC CAPITAL LETTER YU */
1037   { 0x06e1, 0x0410 }, /*                  Cyrillic_A А CYRILLIC CAPITAL LETTER A */
1038   { 0x06e2, 0x0411 }, /*                 Cyrillic_BE Б CYRILLIC CAPITAL LETTER BE */
1039   { 0x06e3, 0x0426 }, /*                Cyrillic_TSE Ц CYRILLIC CAPITAL LETTER TSE */
1040   { 0x06e4, 0x0414 }, /*                 Cyrillic_DE Д CYRILLIC CAPITAL LETTER DE */
1041   { 0x06e5, 0x0415 }, /*                 Cyrillic_IE Е CYRILLIC CAPITAL LETTER IE */
1042   { 0x06e6, 0x0424 }, /*                 Cyrillic_EF Ф CYRILLIC CAPITAL LETTER EF */
1043   { 0x06e7, 0x0413 }, /*                Cyrillic_GHE Г CYRILLIC CAPITAL LETTER GHE */
1044   { 0x06e8, 0x0425 }, /*                 Cyrillic_HA Х CYRILLIC CAPITAL LETTER HA */
1045   { 0x06e9, 0x0418 }, /*                  Cyrillic_I И CYRILLIC CAPITAL LETTER I */
1046   { 0x06ea, 0x0419 }, /*             Cyrillic_SHORTI Й CYRILLIC CAPITAL LETTER SHORT I */
1047   { 0x06eb, 0x041a }, /*                 Cyrillic_KA К CYRILLIC CAPITAL LETTER KA */
1048   { 0x06ec, 0x041b }, /*                 Cyrillic_EL Л CYRILLIC CAPITAL LETTER EL */
1049   { 0x06ed, 0x041c }, /*                 Cyrillic_EM М CYRILLIC CAPITAL LETTER EM */
1050   { 0x06ee, 0x041d }, /*                 Cyrillic_EN Н CYRILLIC CAPITAL LETTER EN */
1051   { 0x06ef, 0x041e }, /*                  Cyrillic_O О CYRILLIC CAPITAL LETTER O */
1052   { 0x06f0, 0x041f }, /*                 Cyrillic_PE П CYRILLIC CAPITAL LETTER PE */
1053   { 0x06f1, 0x042f }, /*                 Cyrillic_YA Я CYRILLIC CAPITAL LETTER YA */
1054   { 0x06f2, 0x0420 }, /*                 Cyrillic_ER Р CYRILLIC CAPITAL LETTER ER */
1055   { 0x06f3, 0x0421 }, /*                 Cyrillic_ES С CYRILLIC CAPITAL LETTER ES */
1056   { 0x06f4, 0x0422 }, /*                 Cyrillic_TE Т CYRILLIC CAPITAL LETTER TE */
1057   { 0x06f5, 0x0423 }, /*                  Cyrillic_U У CYRILLIC CAPITAL LETTER U */
1058   { 0x06f6, 0x0416 }, /*                Cyrillic_ZHE Ж CYRILLIC CAPITAL LETTER ZHE */
1059   { 0x06f7, 0x0412 }, /*                 Cyrillic_VE В CYRILLIC CAPITAL LETTER VE */
1060   { 0x06f8, 0x042c }, /*           Cyrillic_SOFTSIGN Ь CYRILLIC CAPITAL LETTER SOFT SIGN */
1061   { 0x06f9, 0x042b }, /*               Cyrillic_YERU Ы CYRILLIC CAPITAL LETTER YERU */
1062   { 0x06fa, 0x0417 }, /*                 Cyrillic_ZE З CYRILLIC CAPITAL LETTER ZE */
1063   { 0x06fb, 0x0428 }, /*                Cyrillic_SHA Ш CYRILLIC CAPITAL LETTER SHA */
1064   { 0x06fc, 0x042d }, /*                  Cyrillic_E Э CYRILLIC CAPITAL LETTER E */
1065   { 0x06fd, 0x0429 }, /*              Cyrillic_SHCHA Щ CYRILLIC CAPITAL LETTER SHCHA */
1066   { 0x06fe, 0x0427 }, /*                Cyrillic_CHE Ч CYRILLIC CAPITAL LETTER CHE */
1067   { 0x06ff, 0x042a }, /*           Cyrillic_HARDSIGN Ъ CYRILLIC CAPITAL LETTER HARD SIGN */
1068   { 0x07a1, 0x0386 }, /*           Greek_ALPHAaccent Ά GREEK CAPITAL LETTER ALPHA WITH TONOS */
1069   { 0x07a2, 0x0388 }, /*         Greek_EPSILONaccent Έ GREEK CAPITAL LETTER EPSILON WITH TONOS */
1070   { 0x07a3, 0x0389 }, /*             Greek_ETAaccent Ή GREEK CAPITAL LETTER ETA WITH TONOS */
1071   { 0x07a4, 0x038a }, /*            Greek_IOTAaccent Ί GREEK CAPITAL LETTER IOTA WITH TONOS */
1072   { 0x07a5, 0x03aa }, /*         Greek_IOTAdiaeresis Ϊ GREEK CAPITAL LETTER IOTA WITH DIALYTIKA */
1073   { 0x07a7, 0x038c }, /*         Greek_OMICRONaccent Ό GREEK CAPITAL LETTER OMICRON WITH TONOS */
1074   { 0x07a8, 0x038e }, /*         Greek_UPSILONaccent Ύ GREEK CAPITAL LETTER UPSILON WITH TONOS */
1075   { 0x07a9, 0x03ab }, /*       Greek_UPSILONdieresis Ϋ GREEK CAPITAL LETTER UPSILON WITH DIALYTIKA */
1076   { 0x07ab, 0x038f }, /*           Greek_OMEGAaccent Ώ GREEK CAPITAL LETTER OMEGA WITH TONOS */
1077   { 0x07ae, 0x0385 }, /*        Greek_accentdieresis ΅ GREEK DIALYTIKA TONOS */
1078   { 0x07af, 0x2015 }, /*              Greek_horizbar ― HORIZONTAL BAR */
1079   { 0x07b1, 0x03ac }, /*           Greek_alphaaccent ά GREEK SMALL LETTER ALPHA WITH TONOS */
1080   { 0x07b2, 0x03ad }, /*         Greek_epsilonaccent έ GREEK SMALL LETTER EPSILON WITH TONOS */
1081   { 0x07b3, 0x03ae }, /*             Greek_etaaccent ή GREEK SMALL LETTER ETA WITH TONOS */
1082   { 0x07b4, 0x03af }, /*            Greek_iotaaccent ί GREEK SMALL LETTER IOTA WITH TONOS */
1083   { 0x07b5, 0x03ca }, /*          Greek_iotadieresis ϊ GREEK SMALL LETTER IOTA WITH DIALYTIKA */
1084   { 0x07b6, 0x0390 }, /*    Greek_iotaaccentdieresis ΐ GREEK SMALL LETTER IOTA WITH DIALYTIKA AND TONOS */
1085   { 0x07b7, 0x03cc }, /*         Greek_omicronaccent ό GREEK SMALL LETTER OMICRON WITH TONOS */
1086   { 0x07b8, 0x03cd }, /*         Greek_upsilonaccent ύ GREEK SMALL LETTER UPSILON WITH TONOS */
1087   { 0x07b9, 0x03cb }, /*       Greek_upsilondieresis ϋ GREEK SMALL LETTER UPSILON WITH DIALYTIKA */
1088   { 0x07ba, 0x03b0 }, /* Greek_upsilonaccentdieresis ΰ GREEK SMALL LETTER UPSILON WITH DIALYTIKA AND TONOS */
1089   { 0x07bb, 0x03ce }, /*           Greek_omegaaccent ώ GREEK SMALL LETTER OMEGA WITH TONOS */
1090   { 0x07c1, 0x0391 }, /*                 Greek_ALPHA Α GREEK CAPITAL LETTER ALPHA */
1091   { 0x07c2, 0x0392 }, /*                  Greek_BETA Β GREEK CAPITAL LETTER BETA */
1092   { 0x07c3, 0x0393 }, /*                 Greek_GAMMA Γ GREEK CAPITAL LETTER GAMMA */
1093   { 0x07c4, 0x0394 }, /*                 Greek_DELTA Δ GREEK CAPITAL LETTER DELTA */
1094   { 0x07c5, 0x0395 }, /*               Greek_EPSILON Ε GREEK CAPITAL LETTER EPSILON */
1095   { 0x07c6, 0x0396 }, /*                  Greek_ZETA Ζ GREEK CAPITAL LETTER ZETA */
1096   { 0x07c7, 0x0397 }, /*                   Greek_ETA Η GREEK CAPITAL LETTER ETA */
1097   { 0x07c8, 0x0398 }, /*                 Greek_THETA Θ GREEK CAPITAL LETTER THETA */
1098   { 0x07c9, 0x0399 }, /*                  Greek_IOTA Ι GREEK CAPITAL LETTER IOTA */
1099   { 0x07ca, 0x039a }, /*                 Greek_KAPPA Κ GREEK CAPITAL LETTER KAPPA */
1100   { 0x07cb, 0x039b }, /*                Greek_LAMBDA Λ GREEK CAPITAL LETTER LAMDA */
1101   { 0x07cc, 0x039c }, /*                    Greek_MU Μ GREEK CAPITAL LETTER MU */
1102   { 0x07cd, 0x039d }, /*                    Greek_NU Ν GREEK CAPITAL LETTER NU */
1103   { 0x07ce, 0x039e }, /*                    Greek_XI Ξ GREEK CAPITAL LETTER XI */
1104   { 0x07cf, 0x039f }, /*               Greek_OMICRON Ο GREEK CAPITAL LETTER OMICRON */
1105   { 0x07d0, 0x03a0 }, /*                    Greek_PI Π GREEK CAPITAL LETTER PI */
1106   { 0x07d1, 0x03a1 }, /*                   Greek_RHO Ρ GREEK CAPITAL LETTER RHO */
1107   { 0x07d2, 0x03a3 }, /*                 Greek_SIGMA Σ GREEK CAPITAL LETTER SIGMA */
1108   { 0x07d4, 0x03a4 }, /*                   Greek_TAU Τ GREEK CAPITAL LETTER TAU */
1109   { 0x07d5, 0x03a5 }, /*               Greek_UPSILON Υ GREEK CAPITAL LETTER UPSILON */
1110   { 0x07d6, 0x03a6 }, /*                   Greek_PHI Φ GREEK CAPITAL LETTER PHI */
1111   { 0x07d7, 0x03a7 }, /*                   Greek_CHI Χ GREEK CAPITAL LETTER CHI */
1112   { 0x07d8, 0x03a8 }, /*                   Greek_PSI Ψ GREEK CAPITAL LETTER PSI */
1113   { 0x07d9, 0x03a9 }, /*                 Greek_OMEGA Ω GREEK CAPITAL LETTER OMEGA */
1114   { 0x07e1, 0x03b1 }, /*                 Greek_alpha α GREEK SMALL LETTER ALPHA */
1115   { 0x07e2, 0x03b2 }, /*                  Greek_beta β GREEK SMALL LETTER BETA */
1116   { 0x07e3, 0x03b3 }, /*                 Greek_gamma γ GREEK SMALL LETTER GAMMA */
1117   { 0x07e4, 0x03b4 }, /*                 Greek_delta δ GREEK SMALL LETTER DELTA */
1118   { 0x07e5, 0x03b5 }, /*               Greek_epsilon ε GREEK SMALL LETTER EPSILON */
1119   { 0x07e6, 0x03b6 }, /*                  Greek_zeta ζ GREEK SMALL LETTER ZETA */
1120   { 0x07e7, 0x03b7 }, /*                   Greek_eta η GREEK SMALL LETTER ETA */
1121   { 0x07e8, 0x03b8 }, /*                 Greek_theta θ GREEK SMALL LETTER THETA */
1122   { 0x07e9, 0x03b9 }, /*                  Greek_iota ι GREEK SMALL LETTER IOTA */
1123   { 0x07ea, 0x03ba }, /*                 Greek_kappa κ GREEK SMALL LETTER KAPPA */
1124   { 0x07eb, 0x03bb }, /*                Greek_lambda λ GREEK SMALL LETTER LAMDA */
1125   { 0x07ec, 0x03bc }, /*                    Greek_mu μ GREEK SMALL LETTER MU */
1126   { 0x07ed, 0x03bd }, /*                    Greek_nu ν GREEK SMALL LETTER NU */
1127   { 0x07ee, 0x03be }, /*                    Greek_xi ξ GREEK SMALL LETTER XI */
1128   { 0x07ef, 0x03bf }, /*               Greek_omicron ο GREEK SMALL LETTER OMICRON */
1129   { 0x07f0, 0x03c0 }, /*                    Greek_pi π GREEK SMALL LETTER PI */
1130   { 0x07f1, 0x03c1 }, /*                   Greek_rho ρ GREEK SMALL LETTER RHO */
1131   { 0x07f2, 0x03c3 }, /*                 Greek_sigma σ GREEK SMALL LETTER SIGMA */
1132   { 0x07f3, 0x03c2 }, /*       Greek_finalsmallsigma ς GREEK SMALL LETTER FINAL SIGMA */
1133   { 0x07f4, 0x03c4 }, /*                   Greek_tau τ GREEK SMALL LETTER TAU */
1134   { 0x07f5, 0x03c5 }, /*               Greek_upsilon υ GREEK SMALL LETTER UPSILON */
1135   { 0x07f6, 0x03c6 }, /*                   Greek_phi φ GREEK SMALL LETTER PHI */
1136   { 0x07f7, 0x03c7 }, /*                   Greek_chi χ GREEK SMALL LETTER CHI */
1137   { 0x07f8, 0x03c8 }, /*                   Greek_psi ψ GREEK SMALL LETTER PSI */
1138   { 0x07f9, 0x03c9 }, /*                 Greek_omega ω GREEK SMALL LETTER OMEGA */
1139 /*  0x08a1                               leftradical ? ??? */
1140 /*  0x08a2                            topleftradical ? ??? */
1141 /*  0x08a3                            horizconnector ? ??? */
1142   { 0x08a4, 0x2320 }, /*                 topintegral ⌠ TOP HALF INTEGRAL */
1143   { 0x08a5, 0x2321 }, /*                 botintegral ⌡ BOTTOM HALF INTEGRAL */
1144   { 0x08a6, 0x2502 }, /*               vertconnector │ BOX DRAWINGS LIGHT VERTICAL */
1145 /*  0x08a7                          topleftsqbracket ? ??? */
1146 /*  0x08a8                          botleftsqbracket ? ??? */
1147 /*  0x08a9                         toprightsqbracket ? ??? */
1148 /*  0x08aa                         botrightsqbracket ? ??? */
1149 /*  0x08ab                             topleftparens ? ??? */
1150 /*  0x08ac                             botleftparens ? ??? */
1151 /*  0x08ad                            toprightparens ? ??? */
1152 /*  0x08ae                            botrightparens ? ??? */
1153 /*  0x08af                      leftmiddlecurlybrace ? ??? */
1154 /*  0x08b0                     rightmiddlecurlybrace ? ??? */
1155 /*  0x08b1                          topleftsummation ? ??? */
1156 /*  0x08b2                          botleftsummation ? ??? */
1157 /*  0x08b3                 topvertsummationconnector ? ??? */
1158 /*  0x08b4                 botvertsummationconnector ? ??? */
1159 /*  0x08b5                         toprightsummation ? ??? */
1160 /*  0x08b6                         botrightsummation ? ??? */
1161 /*  0x08b7                      rightmiddlesummation ? ??? */
1162   { 0x08bc, 0x2264 }, /*               lessthanequal ≤ LESS-THAN OR EQUAL TO */
1163   { 0x08bd, 0x2260 }, /*                    notequal ≠ NOT EQUAL TO */
1164   { 0x08be, 0x2265 }, /*            greaterthanequal ≥ GREATER-THAN OR EQUAL TO */
1165   { 0x08bf, 0x222b }, /*                    integral ∫ INTEGRAL */
1166   { 0x08c0, 0x2234 }, /*                   therefore ∴ THEREFORE */
1167   { 0x08c1, 0x221d }, /*                   variation ∝ PROPORTIONAL TO */
1168   { 0x08c2, 0x221e }, /*                    infinity ∞ INFINITY */
1169   { 0x08c5, 0x2207 }, /*                       nabla ∇ NABLA */
1170   { 0x08c8, 0x2245 }, /*                 approximate ≅ APPROXIMATELY EQUAL TO */
1171 /*  0x08c9                              similarequal ? ??? */
1172   { 0x08cd, 0x21d4 }, /*                    ifonlyif ⇔ LEFT RIGHT DOUBLE ARROW */
1173   { 0x08ce, 0x21d2 }, /*                     implies ⇒ RIGHTWARDS DOUBLE ARROW */
1174   { 0x08cf, 0x2261 }, /*                   identical ≡ IDENTICAL TO */
1175   { 0x08d6, 0x221a }, /*                     radical √ SQUARE ROOT */
1176   { 0x08da, 0x2282 }, /*                  includedin ⊂ SUBSET OF */
1177   { 0x08db, 0x2283 }, /*                    includes ⊃ SUPERSET OF */
1178   { 0x08dc, 0x2229 }, /*                intersection ∩ INTERSECTION */
1179   { 0x08dd, 0x222a }, /*                       union ∪ UNION */
1180   { 0x08de, 0x2227 }, /*                  logicaland ∧ LOGICAL AND */
1181   { 0x08df, 0x2228 }, /*                   logicalor ∨ LOGICAL OR */
1182   { 0x08ef, 0x2202 }, /*           partialderivative ∂ PARTIAL DIFFERENTIAL */
1183   { 0x08f6, 0x0192 }, /*                    function ƒ LATIN SMALL LETTER F WITH HOOK */
1184   { 0x08fb, 0x2190 }, /*                   leftarrow ← LEFTWARDS ARROW */
1185   { 0x08fc, 0x2191 }, /*                     uparrow ↑ UPWARDS ARROW */
1186   { 0x08fd, 0x2192 }, /*                  rightarrow → RIGHTWARDS ARROW */
1187   { 0x08fe, 0x2193 }, /*                   downarrow ↓ DOWNWARDS ARROW */
1188   { 0x09df, 0x2422 }, /*                       blank ␢ BLANK SYMBOL */
1189   { 0x09e0, 0x25c6 }, /*                soliddiamond ◆ BLACK DIAMOND */
1190   { 0x09e1, 0x2592 }, /*                checkerboard ▒ MEDIUM SHADE */
1191   { 0x09e2, 0x2409 }, /*                          ht ␉ SYMBOL FOR HORIZONTAL TABULATION */
1192   { 0x09e3, 0x240c }, /*                          ff ␌ SYMBOL FOR FORM FEED */
1193   { 0x09e4, 0x240d }, /*                          cr ␍ SYMBOL FOR CARRIAGE RETURN */
1194   { 0x09e5, 0x240a }, /*                          lf ␊ SYMBOL FOR LINE FEED */
1195   { 0x09e8, 0x2424 }, /*                          nl ␤ SYMBOL FOR NEWLINE */
1196   { 0x09e9, 0x240b }, /*                          vt ␋ SYMBOL FOR VERTICAL TABULATION */
1197   { 0x09ea, 0x2518 }, /*              lowrightcorner ┘ BOX DRAWINGS LIGHT UP AND LEFT */
1198   { 0x09eb, 0x2510 }, /*               uprightcorner ┐ BOX DRAWINGS LIGHT DOWN AND LEFT */
1199   { 0x09ec, 0x250c }, /*                upleftcorner ┌ BOX DRAWINGS LIGHT DOWN AND RIGHT */
1200   { 0x09ed, 0x2514 }, /*               lowleftcorner └ BOX DRAWINGS LIGHT UP AND RIGHT */
1201   { 0x09ee, 0x253c }, /*               crossinglines ┼ BOX DRAWINGS LIGHT VERTICAL AND HORIZONTAL */
1202 /*  0x09ef                            horizlinescan1 ? ??? */
1203 /*  0x09f0                            horizlinescan3 ? ??? */
1204   { 0x09f1, 0x2500 }, /*              horizlinescan5 ─ BOX DRAWINGS LIGHT HORIZONTAL */
1205 /*  0x09f2                            horizlinescan7 ? ??? */
1206 /*  0x09f3                            horizlinescan9 ? ??? */
1207   { 0x09f4, 0x251c }, /*                       leftt ├ BOX DRAWINGS LIGHT VERTICAL AND RIGHT */
1208   { 0x09f5, 0x2524 }, /*                      rightt ┤ BOX DRAWINGS LIGHT VERTICAL AND LEFT */
1209   { 0x09f6, 0x2534 }, /*                        bott ┴ BOX DRAWINGS LIGHT UP AND HORIZONTAL */
1210   { 0x09f7, 0x252c }, /*                        topt ┬ BOX DRAWINGS LIGHT DOWN AND HORIZONTAL */
1211   { 0x09f8, 0x2502 }, /*                     vertbar │ BOX DRAWINGS LIGHT VERTICAL */
1212   { 0x0aa1, 0x2003 }, /*                     emspace   EM SPACE */
1213   { 0x0aa2, 0x2002 }, /*                     enspace   EN SPACE */
1214   { 0x0aa3, 0x2004 }, /*                    em3space   THREE-PER-EM SPACE */
1215   { 0x0aa4, 0x2005 }, /*                    em4space   FOUR-PER-EM SPACE */
1216   { 0x0aa5, 0x2007 }, /*                  digitspace   FIGURE SPACE */
1217   { 0x0aa6, 0x2008 }, /*                  punctspace   PUNCTUATION SPACE */
1218   { 0x0aa7, 0x2009 }, /*                   thinspace   THIN SPACE */
1219   { 0x0aa8, 0x200a }, /*                   hairspace   HAIR SPACE */
1220   { 0x0aa9, 0x2014 }, /*                      emdash — EM DASH */
1221   { 0x0aaa, 0x2013 }, /*                      endash – EN DASH */
1222 /*  0x0aac                               signifblank ? ??? */
1223   { 0x0aae, 0x2026 }, /*                    ellipsis … HORIZONTAL ELLIPSIS */
1224 /*  0x0aaf                           doubbaselinedot ? ??? */
1225   { 0x0ab0, 0x2153 }, /*                    onethird ⅓ VULGAR FRACTION ONE THIRD */
1226   { 0x0ab1, 0x2154 }, /*                   twothirds ⅔ VULGAR FRACTION TWO THIRDS */
1227   { 0x0ab2, 0x2155 }, /*                    onefifth ⅕ VULGAR FRACTION ONE FIFTH */
1228   { 0x0ab3, 0x2156 }, /*                   twofifths ⅖ VULGAR FRACTION TWO FIFTHS */
1229   { 0x0ab4, 0x2157 }, /*                 threefifths ⅗ VULGAR FRACTION THREE FIFTHS */
1230   { 0x0ab5, 0x2158 }, /*                  fourfifths ⅘ VULGAR FRACTION FOUR FIFTHS */
1231   { 0x0ab6, 0x2159 }, /*                    onesixth ⅙ VULGAR FRACTION ONE SIXTH */
1232   { 0x0ab7, 0x215a }, /*                  fivesixths ⅚ VULGAR FRACTION FIVE SIXTHS */
1233   { 0x0ab8, 0x2105 }, /*                      careof ℅ CARE OF */
1234   { 0x0abb, 0x2012 }, /*                     figdash ‒ FIGURE DASH */
1235   { 0x0abc, 0x2329 }, /*            leftanglebracket 〈 LEFT-POINTING ANGLE BRACKET */
1236   { 0x0abd, 0x002e }, /*                decimalpoint . FULL STOP */
1237   { 0x0abe, 0x232a }, /*           rightanglebracket 〉 RIGHT-POINTING ANGLE BRACKET */
1238 /*  0x0abf                                    marker ? ??? */
1239   { 0x0ac3, 0x215b }, /*                   oneeighth ⅛ VULGAR FRACTION ONE EIGHTH */
1240   { 0x0ac4, 0x215c }, /*                threeeighths ⅜ VULGAR FRACTION THREE EIGHTHS */
1241   { 0x0ac5, 0x215d }, /*                 fiveeighths ⅝ VULGAR FRACTION FIVE EIGHTHS */
1242   { 0x0ac6, 0x215e }, /*                seveneighths ⅞ VULGAR FRACTION SEVEN EIGHTHS */
1243   { 0x0ac9, 0x2122 }, /*                   trademark ™ TRADE MARK SIGN */
1244   { 0x0aca, 0x2613 }, /*               signaturemark ☓ SALTIRE */
1245 /*  0x0acb                         trademarkincircle ? ??? */
1246   { 0x0acc, 0x25c1 }, /*            leftopentriangle ◁ WHITE LEFT-POINTING TRIANGLE */
1247   { 0x0acd, 0x25b7 }, /*           rightopentriangle ▷ WHITE RIGHT-POINTING TRIANGLE */
1248   { 0x0ace, 0x25cb }, /*                emopencircle ○ WHITE CIRCLE */
1249   { 0x0acf, 0x25a1 }, /*             emopenrectangle □ WHITE SQUARE */
1250   { 0x0ad0, 0x2018 }, /*         leftsinglequotemark ‘ LEFT SINGLE QUOTATION MARK */
1251   { 0x0ad1, 0x2019 }, /*        rightsinglequotemark ’ RIGHT SINGLE QUOTATION MARK */
1252   { 0x0ad2, 0x201c }, /*         leftdoublequotemark “ LEFT DOUBLE QUOTATION MARK */
1253   { 0x0ad3, 0x201d }, /*        rightdoublequotemark ” RIGHT DOUBLE QUOTATION MARK */
1254   { 0x0ad4, 0x211e }, /*                prescription ℞ PRESCRIPTION TAKE */
1255   { 0x0ad6, 0x2032 }, /*                     minutes ′ PRIME */
1256   { 0x0ad7, 0x2033 }, /*                     seconds ″ DOUBLE PRIME */
1257   { 0x0ad9, 0x271d }, /*                  latincross ✝ LATIN CROSS */
1258 /*  0x0ada                                  hexagram ? ??? */
1259   { 0x0adb, 0x25ac }, /*            filledrectbullet ▬ BLACK RECTANGLE */
1260   { 0x0adc, 0x25c0 }, /*         filledlefttribullet ◀ BLACK LEFT-POINTING TRIANGLE */
1261   { 0x0add, 0x25b6 }, /*        filledrighttribullet ▶ BLACK RIGHT-POINTING TRIANGLE */
1262   { 0x0ade, 0x25cf }, /*              emfilledcircle ● BLACK CIRCLE */
1263   { 0x0adf, 0x25a0 }, /*                emfilledrect ■ BLACK SQUARE */
1264   { 0x0ae0, 0x25e6 }, /*            enopencircbullet ◦ WHITE BULLET */
1265   { 0x0ae1, 0x25ab }, /*          enopensquarebullet ▫ WHITE SMALL SQUARE */
1266   { 0x0ae2, 0x25ad }, /*              openrectbullet ▭ WHITE RECTANGLE */
1267   { 0x0ae3, 0x25b3 }, /*             opentribulletup △ WHITE UP-POINTING TRIANGLE */
1268   { 0x0ae4, 0x25bd }, /*           opentribulletdown ▽ WHITE DOWN-POINTING TRIANGLE */
1269   { 0x0ae5, 0x2606 }, /*                    openstar ☆ WHITE STAR */
1270   { 0x0ae6, 0x2022 }, /*          enfilledcircbullet • BULLET */
1271   { 0x0ae7, 0x25aa }, /*            enfilledsqbullet ▪ BLACK SMALL SQUARE */
1272   { 0x0ae8, 0x25b2 }, /*           filledtribulletup ▲ BLACK UP-POINTING TRIANGLE */
1273   { 0x0ae9, 0x25bc }, /*         filledtribulletdown ▼ BLACK DOWN-POINTING TRIANGLE */
1274   { 0x0aea, 0x261c }, /*                 leftpointer ☜ WHITE LEFT POINTING INDEX */
1275   { 0x0aeb, 0x261e }, /*                rightpointer ☞ WHITE RIGHT POINTING INDEX */
1276   { 0x0aec, 0x2663 }, /*                        club ♣ BLACK CLUB SUIT */
1277   { 0x0aed, 0x2666 }, /*                     diamond ♦ BLACK DIAMOND SUIT */
1278   { 0x0aee, 0x2665 }, /*                       heart ♥ BLACK HEART SUIT */
1279   { 0x0af0, 0x2720 }, /*                maltesecross ✠ MALTESE CROSS */
1280   { 0x0af1, 0x2020 }, /*                      dagger † DAGGER */
1281   { 0x0af2, 0x2021 }, /*                doubledagger ‡ DOUBLE DAGGER */
1282   { 0x0af3, 0x2713 }, /*                   checkmark ✓ CHECK MARK */
1283   { 0x0af4, 0x2717 }, /*                 ballotcross ✗ BALLOT X */
1284   { 0x0af5, 0x266f }, /*                musicalsharp ♯ MUSIC SHARP SIGN */
1285   { 0x0af6, 0x266d }, /*                 musicalflat ♭ MUSIC FLAT SIGN */
1286   { 0x0af7, 0x2642 }, /*                  malesymbol ♂ MALE SIGN */
1287   { 0x0af8, 0x2640 }, /*                femalesymbol ♀ FEMALE SIGN */
1288   { 0x0af9, 0x260e }, /*                   telephone ☎ BLACK TELEPHONE */
1289   { 0x0afa, 0x2315 }, /*           telephonerecorder ⌕ TELEPHONE RECORDER */
1290   { 0x0afb, 0x2117 }, /*         phonographcopyright ℗ SOUND RECORDING COPYRIGHT */
1291   { 0x0afc, 0x2038 }, /*                       caret ‸ CARET */
1292   { 0x0afd, 0x201a }, /*          singlelowquotemark ‚ SINGLE LOW-9 QUOTATION MARK */
1293   { 0x0afe, 0x201e }, /*          doublelowquotemark „ DOUBLE LOW-9 QUOTATION MARK */
1294 /*  0x0aff                                    cursor ? ??? */
1295   { 0x0ba3, 0x003c }, /*                   leftcaret < LESS-THAN SIGN */
1296   { 0x0ba6, 0x003e }, /*                  rightcaret > GREATER-THAN SIGN */
1297   { 0x0ba8, 0x2228 }, /*                   downcaret ∨ LOGICAL OR */
1298   { 0x0ba9, 0x2227 }, /*                     upcaret ∧ LOGICAL AND */
1299   { 0x0bc0, 0x00af }, /*                     overbar ¯ MACRON */
1300   { 0x0bc2, 0x22a4 }, /*                    downtack ⊤ DOWN TACK */
1301   { 0x0bc3, 0x2229 }, /*                      upshoe ∩ INTERSECTION */
1302   { 0x0bc4, 0x230a }, /*                   downstile ⌊ LEFT FLOOR */
1303   { 0x0bc6, 0x005f }, /*                    underbar _ LOW LINE */
1304   { 0x0bca, 0x2218 }, /*                         jot ∘ RING OPERATOR */
1305   { 0x0bcc, 0x2395 }, /*                        quad ⎕ APL FUNCTIONAL SYMBOL QUAD (Unicode 3.0) */
1306   { 0x0bce, 0x22a5 }, /*                      uptack ⊥ UP TACK */
1307   { 0x0bcf, 0x25cb }, /*                      circle ○ WHITE CIRCLE */
1308   { 0x0bd3, 0x2308 }, /*                     upstile ⌈ LEFT CEILING */
1309   { 0x0bd6, 0x222a }, /*                    downshoe ∪ UNION */
1310   { 0x0bd8, 0x2283 }, /*                   rightshoe ⊃ SUPERSET OF */
1311   { 0x0bda, 0x2282 }, /*                    leftshoe ⊂ SUBSET OF */
1312   { 0x0bdc, 0x22a3 }, /*                    lefttack ⊣ LEFT TACK */
1313   { 0x0bfc, 0x22a2 }, /*                   righttack ⊢ RIGHT TACK */
1314   { 0x0cdf, 0x2017 }, /*        hebrew_doublelowline ‗ DOUBLE LOW LINE */
1315   { 0x0ce0, 0x05d0 }, /*                hebrew_aleph א HEBREW LETTER ALEF */
1316   { 0x0ce1, 0x05d1 }, /*                  hebrew_bet ב HEBREW LETTER BET */
1317   { 0x0ce2, 0x05d2 }, /*                hebrew_gimel ג HEBREW LETTER GIMEL */
1318   { 0x0ce3, 0x05d3 }, /*                hebrew_dalet ד HEBREW LETTER DALET */
1319   { 0x0ce4, 0x05d4 }, /*                   hebrew_he ה HEBREW LETTER HE */
1320   { 0x0ce5, 0x05d5 }, /*                  hebrew_waw ו HEBREW LETTER VAV */
1321   { 0x0ce6, 0x05d6 }, /*                 hebrew_zain ז HEBREW LETTER ZAYIN */
1322   { 0x0ce7, 0x05d7 }, /*                 hebrew_chet ח HEBREW LETTER HET */
1323   { 0x0ce8, 0x05d8 }, /*                  hebrew_tet ט HEBREW LETTER TET */
1324   { 0x0ce9, 0x05d9 }, /*                  hebrew_yod י HEBREW LETTER YOD */
1325   { 0x0cea, 0x05da }, /*            hebrew_finalkaph ך HEBREW LETTER FINAL KAF */
1326   { 0x0ceb, 0x05db }, /*                 hebrew_kaph כ HEBREW LETTER KAF */
1327   { 0x0cec, 0x05dc }, /*                hebrew_lamed ל HEBREW LETTER LAMED */
1328   { 0x0ced, 0x05dd }, /*             hebrew_finalmem ם HEBREW LETTER FINAL MEM */
1329   { 0x0cee, 0x05de }, /*                  hebrew_mem מ HEBREW LETTER MEM */
1330   { 0x0cef, 0x05df }, /*             hebrew_finalnun ן HEBREW LETTER FINAL NUN */
1331   { 0x0cf0, 0x05e0 }, /*                  hebrew_nun נ HEBREW LETTER NUN */
1332   { 0x0cf1, 0x05e1 }, /*               hebrew_samech ס HEBREW LETTER SAMEKH */
1333   { 0x0cf2, 0x05e2 }, /*                 hebrew_ayin ע HEBREW LETTER AYIN */
1334   { 0x0cf3, 0x05e3 }, /*              hebrew_finalpe ף HEBREW LETTER FINAL PE */
1335   { 0x0cf4, 0x05e4 }, /*                   hebrew_pe פ HEBREW LETTER PE */
1336   { 0x0cf5, 0x05e5 }, /*            hebrew_finalzade ץ HEBREW LETTER FINAL TSADI */
1337   { 0x0cf6, 0x05e6 }, /*                 hebrew_zade צ HEBREW LETTER TSADI */
1338   { 0x0cf7, 0x05e7 }, /*                 hebrew_qoph ק HEBREW LETTER QOF */
1339   { 0x0cf8, 0x05e8 }, /*                 hebrew_resh ר HEBREW LETTER RESH */
1340   { 0x0cf9, 0x05e9 }, /*                 hebrew_shin ש HEBREW LETTER SHIN */
1341   { 0x0cfa, 0x05ea }, /*                  hebrew_taw ת HEBREW LETTER TAV */
1342   { 0x0da1, 0x0e01 }, /*                  Thai_kokai ก THAI CHARACTER KO KAI */
1343   { 0x0da2, 0x0e02 }, /*                Thai_khokhai ข THAI CHARACTER KHO KHAI */
1344   { 0x0da3, 0x0e03 }, /*               Thai_khokhuat ฃ THAI CHARACTER KHO KHUAT */
1345   { 0x0da4, 0x0e04 }, /*               Thai_khokhwai ค THAI CHARACTER KHO KHWAI */
1346   { 0x0da5, 0x0e05 }, /*                Thai_khokhon ฅ THAI CHARACTER KHO KHON */
1347   { 0x0da6, 0x0e06 }, /*             Thai_khorakhang ฆ THAI CHARACTER KHO RAKHANG */
1348   { 0x0da7, 0x0e07 }, /*                 Thai_ngongu ง THAI CHARACTER NGO NGU */
1349   { 0x0da8, 0x0e08 }, /*                Thai_chochan จ THAI CHARACTER CHO CHAN */
1350   { 0x0da9, 0x0e09 }, /*               Thai_choching ฉ THAI CHARACTER CHO CHING */
1351   { 0x0daa, 0x0e0a }, /*               Thai_chochang ช THAI CHARACTER CHO CHANG */
1352   { 0x0dab, 0x0e0b }, /*                   Thai_soso ซ THAI CHARACTER SO SO */
1353   { 0x0dac, 0x0e0c }, /*                Thai_chochoe ฌ THAI CHARACTER CHO CHOE */
1354   { 0x0dad, 0x0e0d }, /*                 Thai_yoying ญ THAI CHARACTER YO YING */
1355   { 0x0dae, 0x0e0e }, /*                Thai_dochada ฎ THAI CHARACTER DO CHADA */
1356   { 0x0daf, 0x0e0f }, /*                Thai_topatak ฏ THAI CHARACTER TO PATAK */
1357   { 0x0db0, 0x0e10 }, /*                Thai_thothan ฐ THAI CHARACTER THO THAN */
1358   { 0x0db1, 0x0e11 }, /*          Thai_thonangmontho ฑ THAI CHARACTER THO NANGMONTHO */
1359   { 0x0db2, 0x0e12 }, /*             Thai_thophuthao ฒ THAI CHARACTER THO PHUTHAO */
1360   { 0x0db3, 0x0e13 }, /*                  Thai_nonen ณ THAI CHARACTER NO NEN */
1361   { 0x0db4, 0x0e14 }, /*                  Thai_dodek ด THAI CHARACTER DO DEK */
1362   { 0x0db5, 0x0e15 }, /*                  Thai_totao ต THAI CHARACTER TO TAO */
1363   { 0x0db6, 0x0e16 }, /*               Thai_thothung ถ THAI CHARACTER THO THUNG */
1364   { 0x0db7, 0x0e17 }, /*              Thai_thothahan ท THAI CHARACTER THO THAHAN */
1365   { 0x0db8, 0x0e18 }, /*               Thai_thothong ธ THAI CHARACTER THO THONG */
1366   { 0x0db9, 0x0e19 }, /*                   Thai_nonu น THAI CHARACTER NO NU */
1367   { 0x0dba, 0x0e1a }, /*               Thai_bobaimai บ THAI CHARACTER BO BAIMAI */
1368   { 0x0dbb, 0x0e1b }, /*                  Thai_popla ป THAI CHARACTER PO PLA */
1369   { 0x0dbc, 0x0e1c }, /*               Thai_phophung ผ THAI CHARACTER PHO PHUNG */
1370   { 0x0dbd, 0x0e1d }, /*                   Thai_fofa ฝ THAI CHARACTER FO FA */
1371   { 0x0dbe, 0x0e1e }, /*                Thai_phophan พ THAI CHARACTER PHO PHAN */
1372   { 0x0dbf, 0x0e1f }, /*                  Thai_fofan ฟ THAI CHARACTER FO FAN */
1373   { 0x0dc0, 0x0e20 }, /*             Thai_phosamphao ภ THAI CHARACTER PHO SAMPHAO */
1374   { 0x0dc1, 0x0e21 }, /*                   Thai_moma ม THAI CHARACTER MO MA */
1375   { 0x0dc2, 0x0e22 }, /*                  Thai_yoyak ย THAI CHARACTER YO YAK */
1376   { 0x0dc3, 0x0e23 }, /*                  Thai_rorua ร THAI CHARACTER RO RUA */
1377   { 0x0dc4, 0x0e24 }, /*                     Thai_ru ฤ THAI CHARACTER RU */
1378   { 0x0dc5, 0x0e25 }, /*                 Thai_loling ล THAI CHARACTER LO LING */
1379   { 0x0dc6, 0x0e26 }, /*                     Thai_lu ฦ THAI CHARACTER LU */
1380   { 0x0dc7, 0x0e27 }, /*                 Thai_wowaen ว THAI CHARACTER WO WAEN */
1381   { 0x0dc8, 0x0e28 }, /*                 Thai_sosala ศ THAI CHARACTER SO SALA */
1382   { 0x0dc9, 0x0e29 }, /*                 Thai_sorusi ษ THAI CHARACTER SO RUSI */
1383   { 0x0dca, 0x0e2a }, /*                  Thai_sosua ส THAI CHARACTER SO SUA */
1384   { 0x0dcb, 0x0e2b }, /*                  Thai_hohip ห THAI CHARACTER HO HIP */
1385   { 0x0dcc, 0x0e2c }, /*                Thai_lochula ฬ THAI CHARACTER LO CHULA */
1386   { 0x0dcd, 0x0e2d }, /*                   Thai_oang อ THAI CHARACTER O ANG */
1387   { 0x0dce, 0x0e2e }, /*               Thai_honokhuk ฮ THAI CHARACTER HO NOKHUK */
1388   { 0x0dcf, 0x0e2f }, /*              Thai_paiyannoi ฯ THAI CHARACTER PAIYANNOI */
1389   { 0x0dd0, 0x0e30 }, /*                  Thai_saraa ะ THAI CHARACTER SARA A */
1390   { 0x0dd1, 0x0e31 }, /*             Thai_maihanakat ั THAI CHARACTER MAI HAN-AKAT */
1391   { 0x0dd2, 0x0e32 }, /*                 Thai_saraaa า THAI CHARACTER SARA AA */
1392   { 0x0dd3, 0x0e33 }, /*                 Thai_saraam ำ THAI CHARACTER SARA AM */
1393   { 0x0dd4, 0x0e34 }, /*                  Thai_sarai ิ THAI CHARACTER SARA I */
1394   { 0x0dd5, 0x0e35 }, /*                 Thai_saraii ี THAI CHARACTER SARA II */
1395   { 0x0dd6, 0x0e36 }, /*                 Thai_saraue ึ THAI CHARACTER SARA UE */
1396   { 0x0dd7, 0x0e37 }, /*                Thai_sarauee ื THAI CHARACTER SARA UEE */
1397   { 0x0dd8, 0x0e38 }, /*                  Thai_sarau ุ THAI CHARACTER SARA U */
1398   { 0x0dd9, 0x0e39 }, /*                 Thai_sarauu ู THAI CHARACTER SARA UU */
1399   { 0x0dda, 0x0e3a }, /*                Thai_phinthu ฺ THAI CHARACTER PHINTHU */
1400   { 0x0dde, 0x0e3e }, /*      Thai_maihanakat_maitho ฾ ??? */
1401   { 0x0ddf, 0x0e3f }, /*                   Thai_baht ฿ THAI CURRENCY SYMBOL BAHT */
1402   { 0x0de0, 0x0e40 }, /*                  Thai_sarae เ THAI CHARACTER SARA E */
1403   { 0x0de1, 0x0e41 }, /*                 Thai_saraae แ THAI CHARACTER SARA AE */
1404   { 0x0de2, 0x0e42 }, /*                  Thai_sarao โ THAI CHARACTER SARA O */
1405   { 0x0de3, 0x0e43 }, /*          Thai_saraaimaimuan ใ THAI CHARACTER SARA AI MAIMUAN */
1406   { 0x0de4, 0x0e44 }, /*         Thai_saraaimaimalai ไ THAI CHARACTER SARA AI MAIMALAI */
1407   { 0x0de5, 0x0e45 }, /*            Thai_lakkhangyao ๅ THAI CHARACTER LAKKHANGYAO */
1408   { 0x0de6, 0x0e46 }, /*               Thai_maiyamok ๆ THAI CHARACTER MAIYAMOK */
1409   { 0x0de7, 0x0e47 }, /*              Thai_maitaikhu ็ THAI CHARACTER MAITAIKHU */
1410   { 0x0de8, 0x0e48 }, /*                  Thai_maiek ่ THAI CHARACTER MAI EK */
1411   { 0x0de9, 0x0e49 }, /*                 Thai_maitho ้ THAI CHARACTER MAI THO */
1412   { 0x0dea, 0x0e4a }, /*                 Thai_maitri ๊ THAI CHARACTER MAI TRI */
1413   { 0x0deb, 0x0e4b }, /*            Thai_maichattawa ๋ THAI CHARACTER MAI CHATTAWA */
1414   { 0x0dec, 0x0e4c }, /*            Thai_thanthakhat ์ THAI CHARACTER THANTHAKHAT */
1415   { 0x0ded, 0x0e4d }, /*               Thai_nikhahit ํ THAI CHARACTER NIKHAHIT */
1416   { 0x0df0, 0x0e50 }, /*                 Thai_leksun ๐ THAI DIGIT ZERO */
1417   { 0x0df1, 0x0e51 }, /*                Thai_leknung ๑ THAI DIGIT ONE */
1418   { 0x0df2, 0x0e52 }, /*                Thai_leksong ๒ THAI DIGIT TWO */
1419   { 0x0df3, 0x0e53 }, /*                 Thai_leksam ๓ THAI DIGIT THREE */
1420   { 0x0df4, 0x0e54 }, /*                  Thai_leksi ๔ THAI DIGIT FOUR */
1421   { 0x0df5, 0x0e55 }, /*                  Thai_lekha ๕ THAI DIGIT FIVE */
1422   { 0x0df6, 0x0e56 }, /*                 Thai_lekhok ๖ THAI DIGIT SIX */
1423   { 0x0df7, 0x0e57 }, /*                Thai_lekchet ๗ THAI DIGIT SEVEN */
1424   { 0x0df8, 0x0e58 }, /*                Thai_lekpaet ๘ THAI DIGIT EIGHT */
1425   { 0x0df9, 0x0e59 }, /*                 Thai_lekkao ๙ THAI DIGIT NINE */
1426   { 0x0ea1, 0x3131 }, /*               Hangul_Kiyeog ㄱ HANGUL LETTER KIYEOK */
1427   { 0x0ea2, 0x3132 }, /*          Hangul_SsangKiyeog ㄲ HANGUL LETTER SSANGKIYEOK */
1428   { 0x0ea3, 0x3133 }, /*           Hangul_KiyeogSios ㄳ HANGUL LETTER KIYEOK-SIOS */
1429   { 0x0ea4, 0x3134 }, /*                Hangul_Nieun ㄴ HANGUL LETTER NIEUN */
1430   { 0x0ea5, 0x3135 }, /*           Hangul_NieunJieuj ㄵ HANGUL LETTER NIEUN-CIEUC */
1431   { 0x0ea6, 0x3136 }, /*           Hangul_NieunHieuh ㄶ HANGUL LETTER NIEUN-HIEUH */
1432   { 0x0ea7, 0x3137 }, /*               Hangul_Dikeud ㄷ HANGUL LETTER TIKEUT */
1433   { 0x0ea8, 0x3138 }, /*          Hangul_SsangDikeud ㄸ HANGUL LETTER SSANGTIKEUT */
1434   { 0x0ea9, 0x3139 }, /*                Hangul_Rieul ㄹ HANGUL LETTER RIEUL */
1435   { 0x0eaa, 0x313a }, /*          Hangul_RieulKiyeog ㄺ HANGUL LETTER RIEUL-KIYEOK */
1436   { 0x0eab, 0x313b }, /*           Hangul_RieulMieum ㄻ HANGUL LETTER RIEUL-MIEUM */
1437   { 0x0eac, 0x313c }, /*           Hangul_RieulPieub ㄼ HANGUL LETTER RIEUL-PIEUP */
1438   { 0x0ead, 0x313d }, /*            Hangul_RieulSios ㄽ HANGUL LETTER RIEUL-SIOS */
1439   { 0x0eae, 0x313e }, /*           Hangul_RieulTieut ㄾ HANGUL LETTER RIEUL-THIEUTH */
1440   { 0x0eaf, 0x313f }, /*          Hangul_RieulPhieuf ㄿ HANGUL LETTER RIEUL-PHIEUPH */
1441   { 0x0eb0, 0x3140 }, /*           Hangul_RieulHieuh ㅀ HANGUL LETTER RIEUL-HIEUH */
1442   { 0x0eb1, 0x3141 }, /*                Hangul_Mieum ㅁ HANGUL LETTER MIEUM */
1443   { 0x0eb2, 0x3142 }, /*                Hangul_Pieub ㅂ HANGUL LETTER PIEUP */
1444   { 0x0eb3, 0x3143 }, /*           Hangul_SsangPieub ㅃ HANGUL LETTER SSANGPIEUP */
1445   { 0x0eb4, 0x3144 }, /*            Hangul_PieubSios ㅄ HANGUL LETTER PIEUP-SIOS */
1446   { 0x0eb5, 0x3145 }, /*                 Hangul_Sios ㅅ HANGUL LETTER SIOS */
1447   { 0x0eb6, 0x3146 }, /*            Hangul_SsangSios ㅆ HANGUL LETTER SSANGSIOS */
1448   { 0x0eb7, 0x3147 }, /*                Hangul_Ieung ㅇ HANGUL LETTER IEUNG */
1449   { 0x0eb8, 0x3148 }, /*                Hangul_Jieuj ㅈ HANGUL LETTER CIEUC */
1450   { 0x0eb9, 0x3149 }, /*           Hangul_SsangJieuj ㅉ HANGUL LETTER SSANGCIEUC */
1451   { 0x0eba, 0x314a }, /*                Hangul_Cieuc ㅊ HANGUL LETTER CHIEUCH */
1452   { 0x0ebb, 0x314b }, /*               Hangul_Khieuq ㅋ HANGUL LETTER KHIEUKH */
1453   { 0x0ebc, 0x314c }, /*                Hangul_Tieut ㅌ HANGUL LETTER THIEUTH */
1454   { 0x0ebd, 0x314d }, /*               Hangul_Phieuf ㅍ HANGUL LETTER PHIEUPH */
1455   { 0x0ebe, 0x314e }, /*                Hangul_Hieuh ㅎ HANGUL LETTER HIEUH */
1456   { 0x0ebf, 0x314f }, /*                    Hangul_A ㅏ HANGUL LETTER A */
1457   { 0x0ec0, 0x3150 }, /*                   Hangul_AE ㅐ HANGUL LETTER AE */
1458   { 0x0ec1, 0x3151 }, /*                   Hangul_YA ㅑ HANGUL LETTER YA */
1459   { 0x0ec2, 0x3152 }, /*                  Hangul_YAE ㅒ HANGUL LETTER YAE */
1460   { 0x0ec3, 0x3153 }, /*                   Hangul_EO ㅓ HANGUL LETTER EO */
1461   { 0x0ec4, 0x3154 }, /*                    Hangul_E ㅔ HANGUL LETTER E */
1462   { 0x0ec5, 0x3155 }, /*                  Hangul_YEO ㅕ HANGUL LETTER YEO */
1463   { 0x0ec6, 0x3156 }, /*                   Hangul_YE ㅖ HANGUL LETTER YE */
1464   { 0x0ec7, 0x3157 }, /*                    Hangul_O ㅗ HANGUL LETTER O */
1465   { 0x0ec8, 0x3158 }, /*                   Hangul_WA ㅘ HANGUL LETTER WA */
1466   { 0x0ec9, 0x3159 }, /*                  Hangul_WAE ㅙ HANGUL LETTER WAE */
1467   { 0x0eca, 0x315a }, /*                   Hangul_OE ㅚ HANGUL LETTER OE */
1468   { 0x0ecb, 0x315b }, /*                   Hangul_YO ㅛ HANGUL LETTER YO */
1469   { 0x0ecc, 0x315c }, /*                    Hangul_U ㅜ HANGUL LETTER U */
1470   { 0x0ecd, 0x315d }, /*                  Hangul_WEO ㅝ HANGUL LETTER WEO */
1471   { 0x0ece, 0x315e }, /*                   Hangul_WE ㅞ HANGUL LETTER WE */
1472   { 0x0ecf, 0x315f }, /*                   Hangul_WI ㅟ HANGUL LETTER WI */
1473   { 0x0ed0, 0x3160 }, /*                   Hangul_YU ㅠ HANGUL LETTER YU */
1474   { 0x0ed1, 0x3161 }, /*                   Hangul_EU ㅡ HANGUL LETTER EU */
1475   { 0x0ed2, 0x3162 }, /*                   Hangul_YI ㅢ HANGUL LETTER YI */
1476   { 0x0ed3, 0x3163 }, /*                    Hangul_I ㅣ HANGUL LETTER I */
1477   { 0x0ed4, 0x11a8 }, /*             Hangul_J_Kiyeog ᆨ HANGUL JONGSEONG KIYEOK */
1478   { 0x0ed5, 0x11a9 }, /*        Hangul_J_SsangKiyeog ᆩ HANGUL JONGSEONG SSANGKIYEOK */
1479   { 0x0ed6, 0x11aa }, /*         Hangul_J_KiyeogSios ᆪ HANGUL JONGSEONG KIYEOK-SIOS */
1480   { 0x0ed7, 0x11ab }, /*              Hangul_J_Nieun ᆫ HANGUL JONGSEONG NIEUN */
1481   { 0x0ed8, 0x11ac }, /*         Hangul_J_NieunJieuj ᆬ HANGUL JONGSEONG NIEUN-CIEUC */
1482   { 0x0ed9, 0x11ad }, /*         Hangul_J_NieunHieuh ᆭ HANGUL JONGSEONG NIEUN-HIEUH */
1483   { 0x0eda, 0x11ae }, /*             Hangul_J_Dikeud ᆮ HANGUL JONGSEONG TIKEUT */
1484   { 0x0edb, 0x11af }, /*              Hangul_J_Rieul ᆯ HANGUL JONGSEONG RIEUL */
1485   { 0x0edc, 0x11b0 }, /*        Hangul_J_RieulKiyeog ᆰ HANGUL JONGSEONG RIEUL-KIYEOK */
1486   { 0x0edd, 0x11b1 }, /*         Hangul_J_RieulMieum ᆱ HANGUL JONGSEONG RIEUL-MIEUM */
1487   { 0x0ede, 0x11b2 }, /*         Hangul_J_RieulPieub ᆲ HANGUL JONGSEONG RIEUL-PIEUP */
1488   { 0x0edf, 0x11b3 }, /*          Hangul_J_RieulSios ᆳ HANGUL JONGSEONG RIEUL-SIOS */
1489   { 0x0ee0, 0x11b4 }, /*         Hangul_J_RieulTieut ᆴ HANGUL JONGSEONG RIEUL-THIEUTH */
1490   { 0x0ee1, 0x11b5 }, /*        Hangul_J_RieulPhieuf ᆵ HANGUL JONGSEONG RIEUL-PHIEUPH */
1491   { 0x0ee2, 0x11b6 }, /*         Hangul_J_RieulHieuh ᆶ HANGUL JONGSEONG RIEUL-HIEUH */
1492   { 0x0ee3, 0x11b7 }, /*              Hangul_J_Mieum ᆷ HANGUL JONGSEONG MIEUM */
1493   { 0x0ee4, 0x11b8 }, /*              Hangul_J_Pieub ᆸ HANGUL JONGSEONG PIEUP */
1494   { 0x0ee5, 0x11b9 }, /*          Hangul_J_PieubSios ᆹ HANGUL JONGSEONG PIEUP-SIOS */
1495   { 0x0ee6, 0x11ba }, /*               Hangul_J_Sios ᆺ HANGUL JONGSEONG SIOS */
1496   { 0x0ee7, 0x11bb }, /*          Hangul_J_SsangSios ᆻ HANGUL JONGSEONG SSANGSIOS */
1497   { 0x0ee8, 0x11bc }, /*              Hangul_J_Ieung ᆼ HANGUL JONGSEONG IEUNG */
1498   { 0x0ee9, 0x11bd }, /*              Hangul_J_Jieuj ᆽ HANGUL JONGSEONG CIEUC */
1499   { 0x0eea, 0x11be }, /*              Hangul_J_Cieuc ᆾ HANGUL JONGSEONG CHIEUCH */
1500   { 0x0eeb, 0x11bf }, /*             Hangul_J_Khieuq ᆿ HANGUL JONGSEONG KHIEUKH */
1501   { 0x0eec, 0x11c0 }, /*              Hangul_J_Tieut ᇀ HANGUL JONGSEONG THIEUTH */
1502   { 0x0eed, 0x11c1 }, /*             Hangul_J_Phieuf ᇁ HANGUL JONGSEONG PHIEUPH */
1503   { 0x0eee, 0x11c2 }, /*              Hangul_J_Hieuh ᇂ HANGUL JONGSEONG HIEUH */
1504   { 0x0eef, 0x316d }, /*     Hangul_RieulYeorinHieuh ㅭ HANGUL LETTER RIEUL-YEORINHIEUH */
1505   { 0x0ef0, 0x3171 }, /*    Hangul_SunkyeongeumMieum ㅱ HANGUL LETTER KAPYEOUNMIEUM */
1506   { 0x0ef1, 0x3178 }, /*    Hangul_SunkyeongeumPieub ㅸ HANGUL LETTER KAPYEOUNPIEUP */
1507   { 0x0ef2, 0x317f }, /*              Hangul_PanSios ㅿ HANGUL LETTER PANSIOS */
1508 /*  0x0ef3                  Hangul_KkogjiDalrinIeung ? ??? */
1509   { 0x0ef4, 0x3184 }, /*   Hangul_SunkyeongeumPhieuf ㆄ HANGUL LETTER KAPYEOUNPHIEUPH */
1510   { 0x0ef5, 0x3186 }, /*          Hangul_YeorinHieuh ㆆ HANGUL LETTER YEORINHIEUH */
1511   { 0x0ef6, 0x318d }, /*                Hangul_AraeA ㆍ HANGUL LETTER ARAEA */
1512   { 0x0ef7, 0x318e }, /*               Hangul_AraeAE ㆎ HANGUL LETTER ARAEAE */
1513   { 0x0ef8, 0x11eb }, /*            Hangul_J_PanSios ᇫ HANGUL JONGSEONG PANSIOS */
1514 /*  0x0ef9                Hangul_J_KkogjiDalrinIeung ? ??? */
1515   { 0x0efa, 0x11f9 }, /*        Hangul_J_YeorinHieuh ᇹ HANGUL JONGSEONG YEORINHIEUH */
1516   { 0x0eff, 0x20a9 }, /*                  Korean_Won ₩ WON SIGN */
1517   { 0x13bc, 0x0152 }, /*                          OE Œ LATIN CAPITAL LIGATURE OE */
1518   { 0x13bd, 0x0153 }, /*                          oe œ LATIN SMALL LIGATURE OE */
1519   { 0x13be, 0x0178 }, /*                  Ydiaeresis Ÿ LATIN CAPITAL LETTER Y WITH DIAERESIS */
1520   { 0x20a0, 0x20a0 }, /*                     EcuSign ₠ EURO-CURRENCY SIGN */
1521   { 0x20a1, 0x20a1 }, /*                   ColonSign ₡ COLON SIGN */
1522   { 0x20a2, 0x20a2 }, /*                CruzeiroSign ₢ CRUZEIRO SIGN */
1523   { 0x20a3, 0x20a3 }, /*                  FFrancSign ₣ FRENCH FRANC SIGN */
1524   { 0x20a4, 0x20a4 }, /*                    LiraSign ₤ LIRA SIGN */
1525   { 0x20a5, 0x20a5 }, /*                    MillSign ₥ MILL SIGN */
1526   { 0x20a6, 0x20a6 }, /*                   NairaSign ₦ NAIRA SIGN */
1527   { 0x20a7, 0x20a7 }, /*                  PesetaSign ₧ PESETA SIGN */
1528   { 0x20a8, 0x20a8 }, /*                   RupeeSign ₨ RUPEE SIGN */
1529   { 0x20a9, 0x20a9 }, /*                     WonSign ₩ WON SIGN */
1530   { 0x20aa, 0x20aa }, /*               NewSheqelSign ₪ NEW SHEQEL SIGN */
1531   { 0x20ab, 0x20ab }, /*                    DongSign ₫ DONG SIGN */
1532   { 0x20ac, 0x20ac }, /*                    EuroSign € EURO SIGN */
1533 };
1534
1535 static guint
1536 keyval_to_unicode (guint keysym)
1537 {
1538   int min = 0;
1539   int max = sizeof(k2utab) / sizeof(k2utab[0]) - 1;
1540   int mid;
1541
1542   /* First check for Latin-1 characters (1:1 mapping) */
1543   if ((keysym >= 0x0020 && keysym <= 0x007e) ||
1544       (keysym >= 0x00a0 && keysym <= 0x00ff))
1545     return keysym;
1546
1547   /* Also check for directly encoded 24-bit UCS characters */
1548   if ((keysym & 0xff000000) == 0x01000000)
1549     return keysym & 0x00ffffff;
1550
1551   /* binary search in table */
1552   while (max >= min) {
1553     mid = (min + max) / 2;
1554     if (k2utab[mid].keysym < keysym)
1555       min = mid + 1;
1556     else if (k2utab[mid].keysym > keysym)
1557       max = mid - 1;
1558     else {
1559       /* found it */
1560       return k2utab[mid].ucs;
1561     }
1562   }
1563   
1564   /* No matching Unicode value found */
1565   return -1;
1566 }
1567
1568 #endif /* 0 */
1569
1570 struct u2k {
1571   unsigned short keysym;
1572   unsigned short ucs;
1573 } u2ktab[] = {
1574   { 0x0abd, 0x002e }, /*                decimalpoint . FULL STOP */
1575   { 0x0ba3, 0x003c }, /*                   leftcaret < LESS-THAN SIGN */
1576   { 0x0ba6, 0x003e }, /*                  rightcaret > GREATER-THAN SIGN */
1577   { 0x0bc6, 0x005f }, /*                    underbar _ LOW LINE */
1578   { 0x0bc0, 0x00af }, /*                     overbar ¯ MACRON */
1579   { 0x03c0, 0x0100 }, /*                     Amacron Ā LATIN CAPITAL LETTER A WITH MACRON */
1580   { 0x03e0, 0x0101 }, /*                     amacron ā LATIN SMALL LETTER A WITH MACRON */
1581   { 0x01c3, 0x0102 }, /*                      Abreve Ă LATIN CAPITAL LETTER A WITH BREVE */
1582   { 0x01e3, 0x0103 }, /*                      abreve ă LATIN SMALL LETTER A WITH BREVE */
1583   { 0x01a1, 0x0104 }, /*                     Aogonek Ą LATIN CAPITAL LETTER A WITH OGONEK */
1584   { 0x01b1, 0x0105 }, /*                     aogonek ą LATIN SMALL LETTER A WITH OGONEK */
1585   { 0x01c6, 0x0106 }, /*                      Cacute Ć LATIN CAPITAL LETTER C WITH ACUTE */
1586   { 0x01e6, 0x0107 }, /*                      cacute ć LATIN SMALL LETTER C WITH ACUTE */
1587   { 0x02c6, 0x0108 }, /*                 Ccircumflex Ĉ LATIN CAPITAL LETTER C WITH CIRCUMFLEX */
1588   { 0x02e6, 0x0109 }, /*                 ccircumflex ĉ LATIN SMALL LETTER C WITH CIRCUMFLEX */
1589   { 0x02c5, 0x010a }, /*                   Cabovedot Ċ LATIN CAPITAL LETTER C WITH DOT ABOVE */
1590   { 0x02e5, 0x010b }, /*                   cabovedot ċ LATIN SMALL LETTER C WITH DOT ABOVE */
1591   { 0x01c8, 0x010c }, /*                      Ccaron Č LATIN CAPITAL LETTER C WITH CARON */
1592   { 0x01e8, 0x010d }, /*                      ccaron č LATIN SMALL LETTER C WITH CARON */
1593   { 0x01cf, 0x010e }, /*                      Dcaron Ď LATIN CAPITAL LETTER D WITH CARON */
1594   { 0x01ef, 0x010f }, /*                      dcaron ď LATIN SMALL LETTER D WITH CARON */
1595   { 0x01d0, 0x0110 }, /*                     Dstroke Đ LATIN CAPITAL LETTER D WITH STROKE */
1596   { 0x01f0, 0x0111 }, /*                     dstroke đ LATIN SMALL LETTER D WITH STROKE */
1597   { 0x03aa, 0x0112 }, /*                     Emacron Ē LATIN CAPITAL LETTER E WITH MACRON */
1598   { 0x03ba, 0x0113 }, /*                     emacron ē LATIN SMALL LETTER E WITH MACRON */
1599   { 0x03cc, 0x0116 }, /*                   Eabovedot Ė LATIN CAPITAL LETTER E WITH DOT ABOVE */
1600   { 0x03ec, 0x0117 }, /*                   eabovedot ė LATIN SMALL LETTER E WITH DOT ABOVE */
1601   { 0x01ca, 0x0118 }, /*                     Eogonek Ę LATIN CAPITAL LETTER E WITH OGONEK */
1602   { 0x01ea, 0x0119 }, /*                     eogonek ę LATIN SMALL LETTER E WITH OGONEK */
1603   { 0x01cc, 0x011a }, /*                      Ecaron Ě LATIN CAPITAL LETTER E WITH CARON */
1604   { 0x01ec, 0x011b }, /*                      ecaron ě LATIN SMALL LETTER E WITH CARON */
1605   { 0x02d8, 0x011c }, /*                 Gcircumflex Ĝ LATIN CAPITAL LETTER G WITH CIRCUMFLEX */
1606   { 0x02f8, 0x011d }, /*                 gcircumflex ĝ LATIN SMALL LETTER G WITH CIRCUMFLEX */
1607   { 0x02ab, 0x011e }, /*                      Gbreve Ğ LATIN CAPITAL LETTER G WITH BREVE */
1608   { 0x02bb, 0x011f }, /*                      gbreve ğ LATIN SMALL LETTER G WITH BREVE */
1609   { 0x02d5, 0x0120 }, /*                   Gabovedot Ġ LATIN CAPITAL LETTER G WITH DOT ABOVE */
1610   { 0x02f5, 0x0121 }, /*                   gabovedot ġ LATIN SMALL LETTER G WITH DOT ABOVE */
1611   { 0x03ab, 0x0122 }, /*                    Gcedilla Ģ LATIN CAPITAL LETTER G WITH CEDILLA */
1612   { 0x03bb, 0x0123 }, /*                    gcedilla ģ LATIN SMALL LETTER G WITH CEDILLA */
1613   { 0x02a6, 0x0124 }, /*                 Hcircumflex Ĥ LATIN CAPITAL LETTER H WITH CIRCUMFLEX */
1614   { 0x02b6, 0x0125 }, /*                 hcircumflex ĥ LATIN SMALL LETTER H WITH CIRCUMFLEX */
1615   { 0x02a1, 0x0126 }, /*                     Hstroke Ħ LATIN CAPITAL LETTER H WITH STROKE */
1616   { 0x02b1, 0x0127 }, /*                     hstroke ħ LATIN SMALL LETTER H WITH STROKE */
1617   { 0x03a5, 0x0128 }, /*                      Itilde Ĩ LATIN CAPITAL LETTER I WITH TILDE */
1618   { 0x03b5, 0x0129 }, /*                      itilde ĩ LATIN SMALL LETTER I WITH TILDE */
1619   { 0x03cf, 0x012a }, /*                     Imacron Ī LATIN CAPITAL LETTER I WITH MACRON */
1620   { 0x03ef, 0x012b }, /*                     imacron ī LATIN SMALL LETTER I WITH MACRON */
1621   { 0x03c7, 0x012e }, /*                     Iogonek Į LATIN CAPITAL LETTER I WITH OGONEK */
1622   { 0x03e7, 0x012f }, /*                     iogonek į LATIN SMALL LETTER I WITH OGONEK */
1623   { 0x02a9, 0x0130 }, /*                   Iabovedot İ LATIN CAPITAL LETTER I WITH DOT ABOVE */
1624   { 0x02b9, 0x0131 }, /*                    idotless ı LATIN SMALL LETTER DOTLESS I */
1625   { 0x02ac, 0x0134 }, /*                 Jcircumflex Ĵ LATIN CAPITAL LETTER J WITH CIRCUMFLEX */
1626   { 0x02bc, 0x0135 }, /*                 jcircumflex ĵ LATIN SMALL LETTER J WITH CIRCUMFLEX */
1627   { 0x03d3, 0x0136 }, /*                    Kcedilla Ķ LATIN CAPITAL LETTER K WITH CEDILLA */
1628   { 0x03f3, 0x0137 }, /*                    kcedilla ķ LATIN SMALL LETTER K WITH CEDILLA */
1629   { 0x03a2, 0x0138 }, /*                         kra ĸ LATIN SMALL LETTER KRA */
1630   { 0x01c5, 0x0139 }, /*                      Lacute Ĺ LATIN CAPITAL LETTER L WITH ACUTE */
1631   { 0x01e5, 0x013a }, /*                      lacute ĺ LATIN SMALL LETTER L WITH ACUTE */
1632   { 0x03a6, 0x013b }, /*                    Lcedilla Ļ LATIN CAPITAL LETTER L WITH CEDILLA */
1633   { 0x03b6, 0x013c }, /*                    lcedilla ļ LATIN SMALL LETTER L WITH CEDILLA */
1634   { 0x01a5, 0x013d }, /*                      Lcaron Ľ LATIN CAPITAL LETTER L WITH CARON */
1635   { 0x01b5, 0x013e }, /*                      lcaron ľ LATIN SMALL LETTER L WITH CARON */
1636   { 0x01a3, 0x0141 }, /*                     Lstroke Ł LATIN CAPITAL LETTER L WITH STROKE */
1637   { 0x01b3, 0x0142 }, /*                     lstroke ł LATIN SMALL LETTER L WITH STROKE */
1638   { 0x01d1, 0x0143 }, /*                      Nacute Ń LATIN CAPITAL LETTER N WITH ACUTE */
1639   { 0x01f1, 0x0144 }, /*                      nacute ń LATIN SMALL LETTER N WITH ACUTE */
1640   { 0x03d1, 0x0145 }, /*                    Ncedilla Ņ LATIN CAPITAL LETTER N WITH CEDILLA */
1641   { 0x03f1, 0x0146 }, /*                    ncedilla ņ LATIN SMALL LETTER N WITH CEDILLA */
1642   { 0x01d2, 0x0147 }, /*                      Ncaron Ň LATIN CAPITAL LETTER N WITH CARON */
1643   { 0x01f2, 0x0148 }, /*                      ncaron ň LATIN SMALL LETTER N WITH CARON */
1644   { 0x03bd, 0x014a }, /*                         ENG Ŋ LATIN CAPITAL LETTER ENG */
1645   { 0x03bf, 0x014b }, /*                         eng ŋ LATIN SMALL LETTER ENG */
1646   { 0x03d2, 0x014c }, /*                     Omacron Ō LATIN CAPITAL LETTER O WITH MACRON */
1647   { 0x03f2, 0x014d }, /*                     omacron ō LATIN SMALL LETTER O WITH MACRON */
1648   { 0x01d5, 0x0150 }, /*                Odoubleacute Ő LATIN CAPITAL LETTER O WITH DOUBLE ACUTE */
1649   { 0x01f5, 0x0151 }, /*                odoubleacute ő LATIN SMALL LETTER O WITH DOUBLE ACUTE */
1650   { 0x13bc, 0x0152 }, /*                          OE Œ LATIN CAPITAL LIGATURE OE */
1651   { 0x13bd, 0x0153 }, /*                          oe œ LATIN SMALL LIGATURE OE */
1652   { 0x01c0, 0x0154 }, /*                      Racute Ŕ LATIN CAPITAL LETTER R WITH ACUTE */
1653   { 0x01e0, 0x0155 }, /*                      racute ŕ LATIN SMALL LETTER R WITH ACUTE */
1654   { 0x03a3, 0x0156 }, /*                    Rcedilla Ŗ LATIN CAPITAL LETTER R WITH CEDILLA */
1655   { 0x03b3, 0x0157 }, /*                    rcedilla ŗ LATIN SMALL LETTER R WITH CEDILLA */
1656   { 0x01d8, 0x0158 }, /*                      Rcaron Ř LATIN CAPITAL LETTER R WITH CARON */
1657   { 0x01f8, 0x0159 }, /*                      rcaron ř LATIN SMALL LETTER R WITH CARON */
1658   { 0x01a6, 0x015a }, /*                      Sacute Ś LATIN CAPITAL LETTER S WITH ACUTE */
1659   { 0x01b6, 0x015b }, /*                      sacute ś LATIN SMALL LETTER S WITH ACUTE */
1660   { 0x02de, 0x015c }, /*                 Scircumflex Ŝ LATIN CAPITAL LETTER S WITH CIRCUMFLEX */
1661   { 0x02fe, 0x015d }, /*                 scircumflex ŝ LATIN SMALL LETTER S WITH CIRCUMFLEX */
1662   { 0x01aa, 0x015e }, /*                    Scedilla Ş LATIN CAPITAL LETTER S WITH CEDILLA */
1663   { 0x01ba, 0x015f }, /*                    scedilla ş LATIN SMALL LETTER S WITH CEDILLA */
1664   { 0x01a9, 0x0160 }, /*                      Scaron Š LATIN CAPITAL LETTER S WITH CARON */
1665   { 0x01b9, 0x0161 }, /*                      scaron š LATIN SMALL LETTER S WITH CARON */
1666   { 0x01de, 0x0162 }, /*                    Tcedilla Ţ LATIN CAPITAL LETTER T WITH CEDILLA */
1667   { 0x01fe, 0x0163 }, /*                    tcedilla ţ LATIN SMALL LETTER T WITH CEDILLA */
1668   { 0x01ab, 0x0164 }, /*                      Tcaron Ť LATIN CAPITAL LETTER T WITH CARON */
1669   { 0x01bb, 0x0165 }, /*                      tcaron ť LATIN SMALL LETTER T WITH CARON */
1670   { 0x03ac, 0x0166 }, /*                      Tslash Ŧ LATIN CAPITAL LETTER T WITH STROKE */
1671   { 0x03bc, 0x0167 }, /*                      tslash ŧ LATIN SMALL LETTER T WITH STROKE */
1672   { 0x03dd, 0x0168 }, /*                      Utilde Ũ LATIN CAPITAL LETTER U WITH TILDE */
1673   { 0x03fd, 0x0169 }, /*                      utilde ũ LATIN SMALL LETTER U WITH TILDE */
1674   { 0x03de, 0x016a }, /*                     Umacron Ū LATIN CAPITAL LETTER U WITH MACRON */
1675   { 0x03fe, 0x016b }, /*                     umacron ū LATIN SMALL LETTER U WITH MACRON */
1676   { 0x02dd, 0x016c }, /*                      Ubreve Ŭ LATIN CAPITAL LETTER U WITH BREVE */
1677   { 0x02fd, 0x016d }, /*                      ubreve ŭ LATIN SMALL LETTER U WITH BREVE */
1678   { 0x01d9, 0x016e }, /*                       Uring Ů LATIN CAPITAL LETTER U WITH RING ABOVE */
1679   { 0x01f9, 0x016f }, /*                       uring ů LATIN SMALL LETTER U WITH RING ABOVE */
1680   { 0x01db, 0x0170 }, /*                Udoubleacute Ű LATIN CAPITAL LETTER U WITH DOUBLE ACUTE */
1681   { 0x01fb, 0x0171 }, /*                udoubleacute ű LATIN SMALL LETTER U WITH DOUBLE ACUTE */
1682   { 0x03d9, 0x0172 }, /*                     Uogonek Ų LATIN CAPITAL LETTER U WITH OGONEK */
1683   { 0x03f9, 0x0173 }, /*                     uogonek ų LATIN SMALL LETTER U WITH OGONEK */
1684   { 0x13be, 0x0178 }, /*                  Ydiaeresis Ÿ LATIN CAPITAL LETTER Y WITH DIAERESIS */
1685   { 0x01ac, 0x0179 }, /*                      Zacute Ź LATIN CAPITAL LETTER Z WITH ACUTE */
1686   { 0x01bc, 0x017a }, /*                      zacute ź LATIN SMALL LETTER Z WITH ACUTE */
1687   { 0x01af, 0x017b }, /*                   Zabovedot Ż LATIN CAPITAL LETTER Z WITH DOT ABOVE */
1688   { 0x01bf, 0x017c }, /*                   zabovedot ż LATIN SMALL LETTER Z WITH DOT ABOVE */
1689   { 0x01ae, 0x017d }, /*                      Zcaron Ž LATIN CAPITAL LETTER Z WITH CARON */
1690   { 0x01be, 0x017e }, /*                      zcaron ž LATIN SMALL LETTER Z WITH CARON */
1691   { 0x08f6, 0x0192 }, /*                    function ƒ LATIN SMALL LETTER F WITH HOOK */
1692   { 0x01b7, 0x02c7 }, /*                       caron ˇ CARON */
1693   { 0x01a2, 0x02d8 }, /*                       breve ˘ BREVE */
1694   { 0x01ff, 0x02d9 }, /*                    abovedot ˙ DOT ABOVE */
1695   { 0x01b2, 0x02db }, /*                      ogonek ˛ OGONEK */
1696   { 0x01bd, 0x02dd }, /*                 doubleacute ˝ DOUBLE ACUTE ACCENT */
1697   { 0x07ae, 0x0385 }, /*        Greek_accentdieresis ΅ GREEK DIALYTIKA TONOS */
1698   { 0x07a1, 0x0386 }, /*           Greek_ALPHAaccent Ά GREEK CAPITAL LETTER ALPHA WITH TONOS */
1699   { 0x07a2, 0x0388 }, /*         Greek_EPSILONaccent Έ GREEK CAPITAL LETTER EPSILON WITH TONOS */
1700   { 0x07a3, 0x0389 }, /*             Greek_ETAaccent Ή GREEK CAPITAL LETTER ETA WITH TONOS */
1701   { 0x07a4, 0x038a }, /*            Greek_IOTAaccent Ί GREEK CAPITAL LETTER IOTA WITH TONOS */
1702   { 0x07a7, 0x038c }, /*         Greek_OMICRONaccent Ό GREEK CAPITAL LETTER OMICRON WITH TONOS */
1703   { 0x07a8, 0x038e }, /*         Greek_UPSILONaccent Ύ GREEK CAPITAL LETTER UPSILON WITH TONOS */
1704   { 0x07ab, 0x038f }, /*           Greek_OMEGAaccent Ώ GREEK CAPITAL LETTER OMEGA WITH TONOS */
1705   { 0x07b6, 0x0390 }, /*    Greek_iotaaccentdieresis ΐ GREEK SMALL LETTER IOTA WITH DIALYTIKA AND TONOS */
1706   { 0x07c1, 0x0391 }, /*                 Greek_ALPHA Α GREEK CAPITAL LETTER ALPHA */
1707   { 0x07c2, 0x0392 }, /*                  Greek_BETA Β GREEK CAPITAL LETTER BETA */
1708   { 0x07c3, 0x0393 }, /*                 Greek_GAMMA Γ GREEK CAPITAL LETTER GAMMA */
1709   { 0x07c4, 0x0394 }, /*                 Greek_DELTA Δ GREEK CAPITAL LETTER DELTA */
1710   { 0x07c5, 0x0395 }, /*               Greek_EPSILON Ε GREEK CAPITAL LETTER EPSILON */
1711   { 0x07c6, 0x0396 }, /*                  Greek_ZETA Ζ GREEK CAPITAL LETTER ZETA */
1712   { 0x07c7, 0x0397 }, /*                   Greek_ETA Η GREEK CAPITAL LETTER ETA */
1713   { 0x07c8, 0x0398 }, /*                 Greek_THETA Θ GREEK CAPITAL LETTER THETA */
1714   { 0x07c9, 0x0399 }, /*                  Greek_IOTA Ι GREEK CAPITAL LETTER IOTA */
1715   { 0x07ca, 0x039a }, /*                 Greek_KAPPA Κ GREEK CAPITAL LETTER KAPPA */
1716   { 0x07cb, 0x039b }, /*                Greek_LAMBDA Λ GREEK CAPITAL LETTER LAMDA */
1717   { 0x07cc, 0x039c }, /*                    Greek_MU Μ GREEK CAPITAL LETTER MU */
1718   { 0x07cd, 0x039d }, /*                    Greek_NU Ν GREEK CAPITAL LETTER NU */
1719   { 0x07ce, 0x039e }, /*                    Greek_XI Ξ GREEK CAPITAL LETTER XI */
1720   { 0x07cf, 0x039f }, /*               Greek_OMICRON Ο GREEK CAPITAL LETTER OMICRON */
1721   { 0x07d0, 0x03a0 }, /*                    Greek_PI Π GREEK CAPITAL LETTER PI */
1722   { 0x07d1, 0x03a1 }, /*                   Greek_RHO Ρ GREEK CAPITAL LETTER RHO */
1723   { 0x07d2, 0x03a3 }, /*                 Greek_SIGMA Σ GREEK CAPITAL LETTER SIGMA */
1724   { 0x07d4, 0x03a4 }, /*                   Greek_TAU Τ GREEK CAPITAL LETTER TAU */
1725   { 0x07d5, 0x03a5 }, /*               Greek_UPSILON Υ GREEK CAPITAL LETTER UPSILON */
1726   { 0x07d6, 0x03a6 }, /*                   Greek_PHI Φ GREEK CAPITAL LETTER PHI */
1727   { 0x07d7, 0x03a7 }, /*                   Greek_CHI Χ GREEK CAPITAL LETTER CHI */
1728   { 0x07d8, 0x03a8 }, /*                   Greek_PSI Ψ GREEK CAPITAL LETTER PSI */
1729   { 0x07d9, 0x03a9 }, /*                 Greek_OMEGA Ω GREEK CAPITAL LETTER OMEGA */
1730   { 0x07a5, 0x03aa }, /*         Greek_IOTAdiaeresis Ϊ GREEK CAPITAL LETTER IOTA WITH DIALYTIKA */
1731   { 0x07a9, 0x03ab }, /*       Greek_UPSILONdieresis Ϋ GREEK CAPITAL LETTER UPSILON WITH DIALYTIKA */
1732   { 0x07b1, 0x03ac }, /*           Greek_alphaaccent ά GREEK SMALL LETTER ALPHA WITH TONOS */
1733   { 0x07b2, 0x03ad }, /*         Greek_epsilonaccent έ GREEK SMALL LETTER EPSILON WITH TONOS */
1734   { 0x07b3, 0x03ae }, /*             Greek_etaaccent ή GREEK SMALL LETTER ETA WITH TONOS */
1735   { 0x07b4, 0x03af }, /*            Greek_iotaaccent ί GREEK SMALL LETTER IOTA WITH TONOS */
1736   { 0x07ba, 0x03b0 }, /* Greek_upsilonaccentdieresis ΰ GREEK SMALL LETTER UPSILON WITH DIALYTIKA AND TONOS */
1737   { 0x07e1, 0x03b1 }, /*                 Greek_alpha α GREEK SMALL LETTER ALPHA */
1738   { 0x07e2, 0x03b2 }, /*                  Greek_beta β GREEK SMALL LETTER BETA */
1739   { 0x07e3, 0x03b3 }, /*                 Greek_gamma γ GREEK SMALL LETTER GAMMA */
1740   { 0x07e4, 0x03b4 }, /*                 Greek_delta δ GREEK SMALL LETTER DELTA */
1741   { 0x07e5, 0x03b5 }, /*               Greek_epsilon ε GREEK SMALL LETTER EPSILON */
1742   { 0x07e6, 0x03b6 }, /*                  Greek_zeta ζ GREEK SMALL LETTER ZETA */
1743   { 0x07e7, 0x03b7 }, /*                   Greek_eta η GREEK SMALL LETTER ETA */
1744   { 0x07e8, 0x03b8 }, /*                 Greek_theta θ GREEK SMALL LETTER THETA */
1745   { 0x07e9, 0x03b9 }, /*                  Greek_iota ι GREEK SMALL LETTER IOTA */
1746   { 0x07ea, 0x03ba }, /*                 Greek_kappa κ GREEK SMALL LETTER KAPPA */
1747   { 0x07eb, 0x03bb }, /*                Greek_lambda λ GREEK SMALL LETTER LAMDA */
1748   { 0x07ec, 0x03bc }, /*                    Greek_mu μ GREEK SMALL LETTER MU */
1749   { 0x07ed, 0x03bd }, /*                    Greek_nu ν GREEK SMALL LETTER NU */
1750   { 0x07ee, 0x03be }, /*                    Greek_xi ξ GREEK SMALL LETTER XI */
1751   { 0x07ef, 0x03bf }, /*               Greek_omicron ο GREEK SMALL LETTER OMICRON */
1752   { 0x07f0, 0x03c0 }, /*                    Greek_pi π GREEK SMALL LETTER PI */
1753   { 0x07f1, 0x03c1 }, /*                   Greek_rho ρ GREEK SMALL LETTER RHO */
1754   { 0x07f3, 0x03c2 }, /*       Greek_finalsmallsigma ς GREEK SMALL LETTER FINAL SIGMA */
1755   { 0x07f2, 0x03c3 }, /*                 Greek_sigma σ GREEK SMALL LETTER SIGMA */
1756   { 0x07f4, 0x03c4 }, /*                   Greek_tau τ GREEK SMALL LETTER TAU */
1757   { 0x07f5, 0x03c5 }, /*               Greek_upsilon υ GREEK SMALL LETTER UPSILON */
1758   { 0x07f6, 0x03c6 }, /*                   Greek_phi φ GREEK SMALL LETTER PHI */
1759   { 0x07f7, 0x03c7 }, /*                   Greek_chi χ GREEK SMALL LETTER CHI */
1760   { 0x07f8, 0x03c8 }, /*                   Greek_psi ψ GREEK SMALL LETTER PSI */
1761   { 0x07f9, 0x03c9 }, /*                 Greek_omega ω GREEK SMALL LETTER OMEGA */
1762   { 0x07b5, 0x03ca }, /*          Greek_iotadieresis ϊ GREEK SMALL LETTER IOTA WITH DIALYTIKA */
1763   { 0x07b9, 0x03cb }, /*       Greek_upsilondieresis ϋ GREEK SMALL LETTER UPSILON WITH DIALYTIKA */
1764   { 0x07b7, 0x03cc }, /*         Greek_omicronaccent ό GREEK SMALL LETTER OMICRON WITH TONOS */
1765   { 0x07b8, 0x03cd }, /*         Greek_upsilonaccent ύ GREEK SMALL LETTER UPSILON WITH TONOS */
1766   { 0x07bb, 0x03ce }, /*           Greek_omegaaccent ώ GREEK SMALL LETTER OMEGA WITH TONOS */
1767   { 0x06b3, 0x0401 }, /*                 Cyrillic_IO Ё CYRILLIC CAPITAL LETTER IO */
1768   { 0x06b1, 0x0402 }, /*                 Serbian_DJE Ђ CYRILLIC CAPITAL LETTER DJE */
1769   { 0x06b2, 0x0403 }, /*               Macedonia_GJE Ѓ CYRILLIC CAPITAL LETTER GJE */
1770   { 0x06b4, 0x0404 }, /*                Ukrainian_IE Є CYRILLIC CAPITAL LETTER UKRAINIAN IE */
1771   { 0x06b5, 0x0405 }, /*               Macedonia_DSE Ѕ CYRILLIC CAPITAL LETTER DZE */
1772   { 0x06b6, 0x0406 }, /*                 Ukrainian_I І CYRILLIC CAPITAL LETTER BYELORUSSIAN-UKRAINIAN I */
1773   { 0x06b7, 0x0407 }, /*                Ukrainian_YI Ї CYRILLIC CAPITAL LETTER YI */
1774   { 0x06b8, 0x0408 }, /*                 Cyrillic_JE Ј CYRILLIC CAPITAL LETTER JE */
1775   { 0x06b9, 0x0409 }, /*                Cyrillic_LJE Љ CYRILLIC CAPITAL LETTER LJE */
1776   { 0x06ba, 0x040a }, /*                Cyrillic_NJE Њ CYRILLIC CAPITAL LETTER NJE */
1777   { 0x06bb, 0x040b }, /*                Serbian_TSHE Ћ CYRILLIC CAPITAL LETTER TSHE */
1778   { 0x06bc, 0x040c }, /*               Macedonia_KJE Ќ CYRILLIC CAPITAL LETTER KJE */
1779   { 0x06be, 0x040e }, /*         Byelorussian_SHORTU Ў CYRILLIC CAPITAL LETTER SHORT U */
1780   { 0x06bf, 0x040f }, /*               Cyrillic_DZHE Џ CYRILLIC CAPITAL LETTER DZHE */
1781   { 0x06e1, 0x0410 }, /*                  Cyrillic_A А CYRILLIC CAPITAL LETTER A */
1782   { 0x06e2, 0x0411 }, /*                 Cyrillic_BE Б CYRILLIC CAPITAL LETTER BE */
1783   { 0x06f7, 0x0412 }, /*                 Cyrillic_VE В CYRILLIC CAPITAL LETTER VE */
1784   { 0x06e7, 0x0413 }, /*                Cyrillic_GHE Г CYRILLIC CAPITAL LETTER GHE */
1785   { 0x06e4, 0x0414 }, /*                 Cyrillic_DE Д CYRILLIC CAPITAL LETTER DE */
1786   { 0x06e5, 0x0415 }, /*                 Cyrillic_IE Е CYRILLIC CAPITAL LETTER IE */
1787   { 0x06f6, 0x0416 }, /*                Cyrillic_ZHE Ж CYRILLIC CAPITAL LETTER ZHE */
1788   { 0x06fa, 0x0417 }, /*                 Cyrillic_ZE З CYRILLIC CAPITAL LETTER ZE */
1789   { 0x06e9, 0x0418 }, /*                  Cyrillic_I И CYRILLIC CAPITAL LETTER I */
1790   { 0x06ea, 0x0419 }, /*             Cyrillic_SHORTI Й CYRILLIC CAPITAL LETTER SHORT I */
1791   { 0x06eb, 0x041a }, /*                 Cyrillic_KA К CYRILLIC CAPITAL LETTER KA */
1792   { 0x06ec, 0x041b }, /*                 Cyrillic_EL Л CYRILLIC CAPITAL LETTER EL */
1793   { 0x06ed, 0x041c }, /*                 Cyrillic_EM М CYRILLIC CAPITAL LETTER EM */
1794   { 0x06ee, 0x041d }, /*                 Cyrillic_EN Н CYRILLIC CAPITAL LETTER EN */
1795   { 0x06ef, 0x041e }, /*                  Cyrillic_O О CYRILLIC CAPITAL LETTER O */
1796   { 0x06f0, 0x041f }, /*                 Cyrillic_PE П CYRILLIC CAPITAL LETTER PE */
1797   { 0x06f2, 0x0420 }, /*                 Cyrillic_ER Р CYRILLIC CAPITAL LETTER ER */
1798   { 0x06f3, 0x0421 }, /*                 Cyrillic_ES С CYRILLIC CAPITAL LETTER ES */
1799   { 0x06f4, 0x0422 }, /*                 Cyrillic_TE Т CYRILLIC CAPITAL LETTER TE */
1800   { 0x06f5, 0x0423 }, /*                  Cyrillic_U У CYRILLIC CAPITAL LETTER U */
1801   { 0x06e6, 0x0424 }, /*                 Cyrillic_EF Ф CYRILLIC CAPITAL LETTER EF */
1802   { 0x06e8, 0x0425 }, /*                 Cyrillic_HA Х CYRILLIC CAPITAL LETTER HA */
1803   { 0x06e3, 0x0426 }, /*                Cyrillic_TSE Ц CYRILLIC CAPITAL LETTER TSE */
1804   { 0x06fe, 0x0427 }, /*                Cyrillic_CHE Ч CYRILLIC CAPITAL LETTER CHE */
1805   { 0x06fb, 0x0428 }, /*                Cyrillic_SHA Ш CYRILLIC CAPITAL LETTER SHA */
1806   { 0x06fd, 0x0429 }, /*              Cyrillic_SHCHA Щ CYRILLIC CAPITAL LETTER SHCHA */
1807   { 0x06ff, 0x042a }, /*           Cyrillic_HARDSIGN Ъ CYRILLIC CAPITAL LETTER HARD SIGN */
1808   { 0x06f9, 0x042b }, /*               Cyrillic_YERU Ы CYRILLIC CAPITAL LETTER YERU */
1809   { 0x06f8, 0x042c }, /*           Cyrillic_SOFTSIGN Ь CYRILLIC CAPITAL LETTER SOFT SIGN */
1810   { 0x06fc, 0x042d }, /*                  Cyrillic_E Э CYRILLIC CAPITAL LETTER E */
1811   { 0x06e0, 0x042e }, /*                 Cyrillic_YU Ю CYRILLIC CAPITAL LETTER YU */
1812   { 0x06f1, 0x042f }, /*                 Cyrillic_YA Я CYRILLIC CAPITAL LETTER YA */
1813   { 0x06c1, 0x0430 }, /*                  Cyrillic_a а CYRILLIC SMALL LETTER A */
1814   { 0x06c2, 0x0431 }, /*                 Cyrillic_be б CYRILLIC SMALL LETTER BE */
1815   { 0x06d7, 0x0432 }, /*                 Cyrillic_ve в CYRILLIC SMALL LETTER VE */
1816   { 0x06c7, 0x0433 }, /*                Cyrillic_ghe г CYRILLIC SMALL LETTER GHE */
1817   { 0x06c4, 0x0434 }, /*                 Cyrillic_de д CYRILLIC SMALL LETTER DE */
1818   { 0x06c5, 0x0435 }, /*                 Cyrillic_ie е CYRILLIC SMALL LETTER IE */
1819   { 0x06d6, 0x0436 }, /*                Cyrillic_zhe ж CYRILLIC SMALL LETTER ZHE */
1820   { 0x06da, 0x0437 }, /*                 Cyrillic_ze з CYRILLIC SMALL LETTER ZE */
1821   { 0x06c9, 0x0438 }, /*                  Cyrillic_i и CYRILLIC SMALL LETTER I */
1822   { 0x06ca, 0x0439 }, /*             Cyrillic_shorti й CYRILLIC SMALL LETTER SHORT I */
1823   { 0x06cb, 0x043a }, /*                 Cyrillic_ka к CYRILLIC SMALL LETTER KA */
1824   { 0x06cc, 0x043b }, /*                 Cyrillic_el л CYRILLIC SMALL LETTER EL */
1825   { 0x06cd, 0x043c }, /*                 Cyrillic_em м CYRILLIC SMALL LETTER EM */
1826   { 0x06ce, 0x043d }, /*                 Cyrillic_en н CYRILLIC SMALL LETTER EN */
1827   { 0x06cf, 0x043e }, /*                  Cyrillic_o о CYRILLIC SMALL LETTER O */
1828   { 0x06d0, 0x043f }, /*                 Cyrillic_pe п CYRILLIC SMALL LETTER PE */
1829   { 0x06d2, 0x0440 }, /*                 Cyrillic_er р CYRILLIC SMALL LETTER ER */
1830   { 0x06d3, 0x0441 }, /*                 Cyrillic_es с CYRILLIC SMALL LETTER ES */
1831   { 0x06d4, 0x0442 }, /*                 Cyrillic_te т CYRILLIC SMALL LETTER TE */
1832   { 0x06d5, 0x0443 }, /*                  Cyrillic_u у CYRILLIC SMALL LETTER U */
1833   { 0x06c6, 0x0444 }, /*                 Cyrillic_ef ф CYRILLIC SMALL LETTER EF */
1834   { 0x06c8, 0x0445 }, /*                 Cyrillic_ha х CYRILLIC SMALL LETTER HA */
1835   { 0x06c3, 0x0446 }, /*                Cyrillic_tse ц CYRILLIC SMALL LETTER TSE */
1836   { 0x06de, 0x0447 }, /*                Cyrillic_che ч CYRILLIC SMALL LETTER CHE */
1837   { 0x06db, 0x0448 }, /*                Cyrillic_sha ш CYRILLIC SMALL LETTER SHA */
1838   { 0x06dd, 0x0449 }, /*              Cyrillic_shcha щ CYRILLIC SMALL LETTER SHCHA */
1839   { 0x06df, 0x044a }, /*           Cyrillic_hardsign ъ CYRILLIC SMALL LETTER HARD SIGN */
1840   { 0x06d9, 0x044b }, /*               Cyrillic_yeru ы CYRILLIC SMALL LETTER YERU */
1841   { 0x06d8, 0x044c }, /*           Cyrillic_softsign ь CYRILLIC SMALL LETTER SOFT SIGN */
1842   { 0x06dc, 0x044d }, /*                  Cyrillic_e э CYRILLIC SMALL LETTER E */
1843   { 0x06c0, 0x044e }, /*                 Cyrillic_yu ю CYRILLIC SMALL LETTER YU */
1844   { 0x06d1, 0x044f }, /*                 Cyrillic_ya я CYRILLIC SMALL LETTER YA */
1845   { 0x06a3, 0x0451 }, /*                 Cyrillic_io ё CYRILLIC SMALL LETTER IO */
1846   { 0x06a1, 0x0452 }, /*                 Serbian_dje ђ CYRILLIC SMALL LETTER DJE */
1847   { 0x06a2, 0x0453 }, /*               Macedonia_gje ѓ CYRILLIC SMALL LETTER GJE */
1848   { 0x06a4, 0x0454 }, /*                Ukrainian_ie є CYRILLIC SMALL LETTER UKRAINIAN IE */
1849   { 0x06a5, 0x0455 }, /*               Macedonia_dse ѕ CYRILLIC SMALL LETTER DZE */
1850   { 0x06a6, 0x0456 }, /*                 Ukrainian_i і CYRILLIC SMALL LETTER BYELORUSSIAN-UKRAINIAN I */
1851   { 0x06a7, 0x0457 }, /*                Ukrainian_yi ї CYRILLIC SMALL LETTER YI */
1852   { 0x06a8, 0x0458 }, /*                 Cyrillic_je ј CYRILLIC SMALL LETTER JE */
1853   { 0x06a9, 0x0459 }, /*                Cyrillic_lje љ CYRILLIC SMALL LETTER LJE */
1854   { 0x06aa, 0x045a }, /*                Cyrillic_nje њ CYRILLIC SMALL LETTER NJE */
1855   { 0x06ab, 0x045b }, /*                Serbian_tshe ћ CYRILLIC SMALL LETTER TSHE */
1856   { 0x06ac, 0x045c }, /*               Macedonia_kje ќ CYRILLIC SMALL LETTER KJE */
1857   { 0x06ae, 0x045e }, /*         Byelorussian_shortu ў CYRILLIC SMALL LETTER SHORT U */
1858   { 0x06af, 0x045f }, /*               Cyrillic_dzhe џ CYRILLIC SMALL LETTER DZHE */
1859   { 0x0ce0, 0x05d0 }, /*                hebrew_aleph א HEBREW LETTER ALEF */
1860   { 0x0ce1, 0x05d1 }, /*                  hebrew_bet ב HEBREW LETTER BET */
1861   { 0x0ce2, 0x05d2 }, /*                hebrew_gimel ג HEBREW LETTER GIMEL */
1862   { 0x0ce3, 0x05d3 }, /*                hebrew_dalet ד HEBREW LETTER DALET */
1863   { 0x0ce4, 0x05d4 }, /*                   hebrew_he ה HEBREW LETTER HE */
1864   { 0x0ce5, 0x05d5 }, /*                  hebrew_waw ו HEBREW LETTER VAV */
1865   { 0x0ce6, 0x05d6 }, /*                 hebrew_zain ז HEBREW LETTER ZAYIN */
1866   { 0x0ce7, 0x05d7 }, /*                 hebrew_chet ח HEBREW LETTER HET */
1867   { 0x0ce8, 0x05d8 }, /*                  hebrew_tet ט HEBREW LETTER TET */
1868   { 0x0ce9, 0x05d9 }, /*                  hebrew_yod י HEBREW LETTER YOD */
1869   { 0x0cea, 0x05da }, /*            hebrew_finalkaph ך HEBREW LETTER FINAL KAF */
1870   { 0x0ceb, 0x05db }, /*                 hebrew_kaph כ HEBREW LETTER KAF */
1871   { 0x0cec, 0x05dc }, /*                hebrew_lamed ל HEBREW LETTER LAMED */
1872   { 0x0ced, 0x05dd }, /*             hebrew_finalmem ם HEBREW LETTER FINAL MEM */
1873   { 0x0cee, 0x05de }, /*                  hebrew_mem מ HEBREW LETTER MEM */
1874   { 0x0cef, 0x05df }, /*             hebrew_finalnun ן HEBREW LETTER FINAL NUN */
1875   { 0x0cf0, 0x05e0 }, /*                  hebrew_nun נ HEBREW LETTER NUN */
1876   { 0x0cf1, 0x05e1 }, /*               hebrew_samech ס HEBREW LETTER SAMEKH */
1877   { 0x0cf2, 0x05e2 }, /*                 hebrew_ayin ע HEBREW LETTER AYIN */
1878   { 0x0cf3, 0x05e3 }, /*              hebrew_finalpe ף HEBREW LETTER FINAL PE */
1879   { 0x0cf4, 0x05e4 }, /*                   hebrew_pe פ HEBREW LETTER PE */
1880   { 0x0cf5, 0x05e5 }, /*            hebrew_finalzade ץ HEBREW LETTER FINAL TSADI */
1881   { 0x0cf6, 0x05e6 }, /*                 hebrew_zade צ HEBREW LETTER TSADI */
1882   { 0x0cf7, 0x05e7 }, /*                 hebrew_qoph ק HEBREW LETTER QOF */
1883   { 0x0cf8, 0x05e8 }, /*                 hebrew_resh ר HEBREW LETTER RESH */
1884   { 0x0cf9, 0x05e9 }, /*                 hebrew_shin ש HEBREW LETTER SHIN */
1885   { 0x0cfa, 0x05ea }, /*                  hebrew_taw ת HEBREW LETTER TAV */
1886   { 0x05ac, 0x060c }, /*                Arabic_comma ، ARABIC COMMA */
1887   { 0x05bb, 0x061b }, /*            Arabic_semicolon ؛ ARABIC SEMICOLON */
1888   { 0x05bf, 0x061f }, /*        Arabic_question_mark ؟ ARABIC QUESTION MARK */
1889   { 0x05c1, 0x0621 }, /*                Arabic_hamza ء ARABIC LETTER HAMZA */
1890   { 0x05c2, 0x0622 }, /*          Arabic_maddaonalef آ ARABIC LETTER ALEF WITH MADDA ABOVE */
1891   { 0x05c3, 0x0623 }, /*          Arabic_hamzaonalef أ ARABIC LETTER ALEF WITH HAMZA ABOVE */
1892   { 0x05c4, 0x0624 }, /*           Arabic_hamzaonwaw ؤ ARABIC LETTER WAW WITH HAMZA ABOVE */
1893   { 0x05c5, 0x0625 }, /*       Arabic_hamzaunderalef إ ARABIC LETTER ALEF WITH HAMZA BELOW */
1894   { 0x05c6, 0x0626 }, /*           Arabic_hamzaonyeh ئ ARABIC LETTER YEH WITH HAMZA ABOVE */
1895   { 0x05c7, 0x0627 }, /*                 Arabic_alef ا ARABIC LETTER ALEF */
1896   { 0x05c8, 0x0628 }, /*                  Arabic_beh ب ARABIC LETTER BEH */
1897   { 0x05c9, 0x0629 }, /*           Arabic_tehmarbuta ة ARABIC LETTER TEH MARBUTA */
1898   { 0x05ca, 0x062a }, /*                  Arabic_teh ت ARABIC LETTER TEH */
1899   { 0x05cb, 0x062b }, /*                 Arabic_theh ث ARABIC LETTER THEH */
1900   { 0x05cc, 0x062c }, /*                 Arabic_jeem ج ARABIC LETTER JEEM */
1901   { 0x05cd, 0x062d }, /*                  Arabic_hah ح ARABIC LETTER HAH */
1902   { 0x05ce, 0x062e }, /*                 Arabic_khah خ ARABIC LETTER KHAH */
1903   { 0x05cf, 0x062f }, /*                  Arabic_dal د ARABIC LETTER DAL */
1904   { 0x05d0, 0x0630 }, /*                 Arabic_thal ذ ARABIC LETTER THAL */
1905   { 0x05d1, 0x0631 }, /*                   Arabic_ra ر ARABIC LETTER REH */
1906   { 0x05d2, 0x0632 }, /*                 Arabic_zain ز ARABIC LETTER ZAIN */
1907   { 0x05d3, 0x0633 }, /*                 Arabic_seen س ARABIC LETTER SEEN */
1908   { 0x05d4, 0x0634 }, /*                Arabic_sheen ش ARABIC LETTER SHEEN */
1909   { 0x05d5, 0x0635 }, /*                  Arabic_sad ص ARABIC LETTER SAD */
1910   { 0x05d6, 0x0636 }, /*                  Arabic_dad ض ARABIC LETTER DAD */
1911   { 0x05d7, 0x0637 }, /*                  Arabic_tah ط ARABIC LETTER TAH */
1912   { 0x05d8, 0x0638 }, /*                  Arabic_zah ظ ARABIC LETTER ZAH */
1913   { 0x05d9, 0x0639 }, /*                  Arabic_ain ع ARABIC LETTER AIN */
1914   { 0x05da, 0x063a }, /*                Arabic_ghain غ ARABIC LETTER GHAIN */
1915   { 0x05e0, 0x0640 }, /*              Arabic_tatweel ـ ARABIC TATWEEL */
1916   { 0x05e1, 0x0641 }, /*                  Arabic_feh ف ARABIC LETTER FEH */
1917   { 0x05e2, 0x0642 }, /*                  Arabic_qaf ق ARABIC LETTER QAF */
1918   { 0x05e3, 0x0643 }, /*                  Arabic_kaf ك ARABIC LETTER KAF */
1919   { 0x05e4, 0x0644 }, /*                  Arabic_lam ل ARABIC LETTER LAM */
1920   { 0x05e5, 0x0645 }, /*                 Arabic_meem م ARABIC LETTER MEEM */
1921   { 0x05e6, 0x0646 }, /*                 Arabic_noon ن ARABIC LETTER NOON */
1922   { 0x05e7, 0x0647 }, /*                   Arabic_ha ه ARABIC LETTER HEH */
1923   { 0x05e8, 0x0648 }, /*                  Arabic_waw و ARABIC LETTER WAW */
1924   { 0x05e9, 0x0649 }, /*          Arabic_alefmaksura ى ARABIC LETTER ALEF MAKSURA */
1925   { 0x05ea, 0x064a }, /*                  Arabic_yeh ي ARABIC LETTER YEH */
1926   { 0x05eb, 0x064b }, /*             Arabic_fathatan ً ARABIC FATHATAN */
1927   { 0x05ec, 0x064c }, /*             Arabic_dammatan ٌ ARABIC DAMMATAN */
1928   { 0x05ed, 0x064d }, /*             Arabic_kasratan ٍ ARABIC KASRATAN */
1929   { 0x05ee, 0x064e }, /*                Arabic_fatha َ ARABIC FATHA */
1930   { 0x05ef, 0x064f }, /*                Arabic_damma ُ ARABIC DAMMA */
1931   { 0x05f0, 0x0650 }, /*                Arabic_kasra ِ ARABIC KASRA */
1932   { 0x05f1, 0x0651 }, /*               Arabic_shadda ّ ARABIC SHADDA */
1933   { 0x05f2, 0x0652 }, /*                Arabic_sukun ْ ARABIC SUKUN */
1934   { 0x0da1, 0x0e01 }, /*                  Thai_kokai ก THAI CHARACTER KO KAI */
1935   { 0x0da2, 0x0e02 }, /*                Thai_khokhai ข THAI CHARACTER KHO KHAI */
1936   { 0x0da3, 0x0e03 }, /*               Thai_khokhuat ฃ THAI CHARACTER KHO KHUAT */
1937   { 0x0da4, 0x0e04 }, /*               Thai_khokhwai ค THAI CHARACTER KHO KHWAI */
1938   { 0x0da5, 0x0e05 }, /*                Thai_khokhon ฅ THAI CHARACTER KHO KHON */
1939   { 0x0da6, 0x0e06 }, /*             Thai_khorakhang ฆ THAI CHARACTER KHO RAKHANG */
1940   { 0x0da7, 0x0e07 }, /*                 Thai_ngongu ง THAI CHARACTER NGO NGU */
1941   { 0x0da8, 0x0e08 }, /*                Thai_chochan จ THAI CHARACTER CHO CHAN */
1942   { 0x0da9, 0x0e09 }, /*               Thai_choching ฉ THAI CHARACTER CHO CHING */
1943   { 0x0daa, 0x0e0a }, /*               Thai_chochang ช THAI CHARACTER CHO CHANG */
1944   { 0x0dab, 0x0e0b }, /*                   Thai_soso ซ THAI CHARACTER SO SO */
1945   { 0x0dac, 0x0e0c }, /*                Thai_chochoe ฌ THAI CHARACTER CHO CHOE */
1946   { 0x0dad, 0x0e0d }, /*                 Thai_yoying ญ THAI CHARACTER YO YING */
1947   { 0x0dae, 0x0e0e }, /*                Thai_dochada ฎ THAI CHARACTER DO CHADA */
1948   { 0x0daf, 0x0e0f }, /*                Thai_topatak ฏ THAI CHARACTER TO PATAK */
1949   { 0x0db0, 0x0e10 }, /*                Thai_thothan ฐ THAI CHARACTER THO THAN */
1950   { 0x0db1, 0x0e11 }, /*          Thai_thonangmontho ฑ THAI CHARACTER THO NANGMONTHO */
1951   { 0x0db2, 0x0e12 }, /*             Thai_thophuthao ฒ THAI CHARACTER THO PHUTHAO */
1952   { 0x0db3, 0x0e13 }, /*                  Thai_nonen ณ THAI CHARACTER NO NEN */
1953   { 0x0db4, 0x0e14 }, /*                  Thai_dodek ด THAI CHARACTER DO DEK */
1954   { 0x0db5, 0x0e15 }, /*                  Thai_totao ต THAI CHARACTER TO TAO */
1955   { 0x0db6, 0x0e16 }, /*               Thai_thothung ถ THAI CHARACTER THO THUNG */
1956   { 0x0db7, 0x0e17 }, /*              Thai_thothahan ท THAI CHARACTER THO THAHAN */
1957   { 0x0db8, 0x0e18 }, /*               Thai_thothong ธ THAI CHARACTER THO THONG */
1958   { 0x0db9, 0x0e19 }, /*                   Thai_nonu น THAI CHARACTER NO NU */
1959   { 0x0dba, 0x0e1a }, /*               Thai_bobaimai บ THAI CHARACTER BO BAIMAI */
1960   { 0x0dbb, 0x0e1b }, /*                  Thai_popla ป THAI CHARACTER PO PLA */
1961   { 0x0dbc, 0x0e1c }, /*               Thai_phophung ผ THAI CHARACTER PHO PHUNG */
1962   { 0x0dbd, 0x0e1d }, /*                   Thai_fofa ฝ THAI CHARACTER FO FA */
1963   { 0x0dbe, 0x0e1e }, /*                Thai_phophan พ THAI CHARACTER PHO PHAN */
1964   { 0x0dbf, 0x0e1f }, /*                  Thai_fofan ฟ THAI CHARACTER FO FAN */
1965   { 0x0dc0, 0x0e20 }, /*             Thai_phosamphao ภ THAI CHARACTER PHO SAMPHAO */
1966   { 0x0dc1, 0x0e21 }, /*                   Thai_moma ม THAI CHARACTER MO MA */
1967   { 0x0dc2, 0x0e22 }, /*                  Thai_yoyak ย THAI CHARACTER YO YAK */
1968   { 0x0dc3, 0x0e23 }, /*                  Thai_rorua ร THAI CHARACTER RO RUA */
1969   { 0x0dc4, 0x0e24 }, /*                     Thai_ru ฤ THAI CHARACTER RU */
1970   { 0x0dc5, 0x0e25 }, /*                 Thai_loling ล THAI CHARACTER LO LING */
1971   { 0x0dc6, 0x0e26 }, /*                     Thai_lu ฦ THAI CHARACTER LU */
1972   { 0x0dc7, 0x0e27 }, /*                 Thai_wowaen ว THAI CHARACTER WO WAEN */
1973   { 0x0dc8, 0x0e28 }, /*                 Thai_sosala ศ THAI CHARACTER SO SALA */
1974   { 0x0dc9, 0x0e29 }, /*                 Thai_sorusi ษ THAI CHARACTER SO RUSI */
1975   { 0x0dca, 0x0e2a }, /*                  Thai_sosua ส THAI CHARACTER SO SUA */
1976   { 0x0dcb, 0x0e2b }, /*                  Thai_hohip ห THAI CHARACTER HO HIP */
1977   { 0x0dcc, 0x0e2c }, /*                Thai_lochula ฬ THAI CHARACTER LO CHULA */
1978   { 0x0dcd, 0x0e2d }, /*                   Thai_oang อ THAI CHARACTER O ANG */
1979   { 0x0dce, 0x0e2e }, /*               Thai_honokhuk ฮ THAI CHARACTER HO NOKHUK */
1980   { 0x0dcf, 0x0e2f }, /*              Thai_paiyannoi ฯ THAI CHARACTER PAIYANNOI */
1981   { 0x0dd0, 0x0e30 }, /*                  Thai_saraa ะ THAI CHARACTER SARA A */
1982   { 0x0dd1, 0x0e31 }, /*             Thai_maihanakat ั THAI CHARACTER MAI HAN-AKAT */
1983   { 0x0dd2, 0x0e32 }, /*                 Thai_saraaa า THAI CHARACTER SARA AA */
1984   { 0x0dd3, 0x0e33 }, /*                 Thai_saraam ำ THAI CHARACTER SARA AM */
1985   { 0x0dd4, 0x0e34 }, /*                  Thai_sarai ิ THAI CHARACTER SARA I */
1986   { 0x0dd5, 0x0e35 }, /*                 Thai_saraii ี THAI CHARACTER SARA II */
1987   { 0x0dd6, 0x0e36 }, /*                 Thai_saraue ึ THAI CHARACTER SARA UE */
1988   { 0x0dd7, 0x0e37 }, /*                Thai_sarauee ื THAI CHARACTER SARA UEE */
1989   { 0x0dd8, 0x0e38 }, /*                  Thai_sarau ุ THAI CHARACTER SARA U */
1990   { 0x0dd9, 0x0e39 }, /*                 Thai_sarauu ู THAI CHARACTER SARA UU */
1991   { 0x0dda, 0x0e3a }, /*                Thai_phinthu ฺ THAI CHARACTER PHINTHU */
1992   { 0x0ddf, 0x0e3f }, /*                   Thai_baht ฿ THAI CURRENCY SYMBOL BAHT */
1993   { 0x0de0, 0x0e40 }, /*                  Thai_sarae เ THAI CHARACTER SARA E */
1994   { 0x0de1, 0x0e41 }, /*                 Thai_saraae แ THAI CHARACTER SARA AE */
1995   { 0x0de2, 0x0e42 }, /*                  Thai_sarao โ THAI CHARACTER SARA O */
1996   { 0x0de3, 0x0e43 }, /*          Thai_saraaimaimuan ใ THAI CHARACTER SARA AI MAIMUAN */
1997   { 0x0de4, 0x0e44 }, /*         Thai_saraaimaimalai ไ THAI CHARACTER SARA AI MAIMALAI */
1998   { 0x0de5, 0x0e45 }, /*            Thai_lakkhangyao ๅ THAI CHARACTER LAKKHANGYAO */
1999   { 0x0de6, 0x0e46 }, /*               Thai_maiyamok ๆ THAI CHARACTER MAIYAMOK */
2000   { 0x0de7, 0x0e47 }, /*              Thai_maitaikhu ็ THAI CHARACTER MAITAIKHU */
2001   { 0x0de8, 0x0e48 }, /*                  Thai_maiek ่ THAI CHARACTER MAI EK */
2002   { 0x0de9, 0x0e49 }, /*                 Thai_maitho ้ THAI CHARACTER MAI THO */
2003   { 0x0dea, 0x0e4a }, /*                 Thai_maitri ๊ THAI CHARACTER MAI TRI */
2004   { 0x0deb, 0x0e4b }, /*            Thai_maichattawa ๋ THAI CHARACTER MAI CHATTAWA */
2005   { 0x0dec, 0x0e4c }, /*            Thai_thanthakhat ์ THAI CHARACTER THANTHAKHAT */
2006   { 0x0ded, 0x0e4d }, /*               Thai_nikhahit ํ THAI CHARACTER NIKHAHIT */
2007   { 0x0df0, 0x0e50 }, /*                 Thai_leksun ๐ THAI DIGIT ZERO */
2008   { 0x0df1, 0x0e51 }, /*                Thai_leknung ๑ THAI DIGIT ONE */
2009   { 0x0df2, 0x0e52 }, /*                Thai_leksong ๒ THAI DIGIT TWO */
2010   { 0x0df3, 0x0e53 }, /*                 Thai_leksam ๓ THAI DIGIT THREE */
2011   { 0x0df4, 0x0e54 }, /*                  Thai_leksi ๔ THAI DIGIT FOUR */
2012   { 0x0df5, 0x0e55 }, /*                  Thai_lekha ๕ THAI DIGIT FIVE */
2013   { 0x0df6, 0x0e56 }, /*                 Thai_lekhok ๖ THAI DIGIT SIX */
2014   { 0x0df7, 0x0e57 }, /*                Thai_lekchet ๗ THAI DIGIT SEVEN */
2015   { 0x0df8, 0x0e58 }, /*                Thai_lekpaet ๘ THAI DIGIT EIGHT */
2016   { 0x0df9, 0x0e59 }, /*                 Thai_lekkao ๙ THAI DIGIT NINE */
2017   { 0x0ed4, 0x11a8 }, /*             Hangul_J_Kiyeog ᆨ HANGUL JONGSEONG KIYEOK */
2018   { 0x0ed5, 0x11a9 }, /*        Hangul_J_SsangKiyeog ᆩ HANGUL JONGSEONG SSANGKIYEOK */
2019   { 0x0ed6, 0x11aa }, /*         Hangul_J_KiyeogSios ᆪ HANGUL JONGSEONG KIYEOK-SIOS */
2020   { 0x0ed7, 0x11ab }, /*              Hangul_J_Nieun ᆫ HANGUL JONGSEONG NIEUN */
2021   { 0x0ed8, 0x11ac }, /*         Hangul_J_NieunJieuj ᆬ HANGUL JONGSEONG NIEUN-CIEUC */
2022   { 0x0ed9, 0x11ad }, /*         Hangul_J_NieunHieuh ᆭ HANGUL JONGSEONG NIEUN-HIEUH */
2023   { 0x0eda, 0x11ae }, /*             Hangul_J_Dikeud ᆮ HANGUL JONGSEONG TIKEUT */
2024   { 0x0edb, 0x11af }, /*              Hangul_J_Rieul ᆯ HANGUL JONGSEONG RIEUL */
2025   { 0x0edc, 0x11b0 }, /*        Hangul_J_RieulKiyeog ᆰ HANGUL JONGSEONG RIEUL-KIYEOK */
2026   { 0x0edd, 0x11b1 }, /*         Hangul_J_RieulMieum ᆱ HANGUL JONGSEONG RIEUL-MIEUM */
2027   { 0x0ede, 0x11b2 }, /*         Hangul_J_RieulPieub ᆲ HANGUL JONGSEONG RIEUL-PIEUP */
2028   { 0x0edf, 0x11b3 }, /*          Hangul_J_RieulSios ᆳ HANGUL JONGSEONG RIEUL-SIOS */
2029   { 0x0ee0, 0x11b4 }, /*         Hangul_J_RieulTieut ᆴ HANGUL JONGSEONG RIEUL-THIEUTH */
2030   { 0x0ee1, 0x11b5 }, /*        Hangul_J_RieulPhieuf ᆵ HANGUL JONGSEONG RIEUL-PHIEUPH */
2031   { 0x0ee2, 0x11b6 }, /*         Hangul_J_RieulHieuh ᆶ HANGUL JONGSEONG RIEUL-HIEUH */
2032   { 0x0ee3, 0x11b7 }, /*              Hangul_J_Mieum ᆷ HANGUL JONGSEONG MIEUM */
2033   { 0x0ee4, 0x11b8 }, /*              Hangul_J_Pieub ᆸ HANGUL JONGSEONG PIEUP */
2034   { 0x0ee5, 0x11b9 }, /*          Hangul_J_PieubSios ᆹ HANGUL JONGSEONG PIEUP-SIOS */
2035   { 0x0ee6, 0x11ba }, /*               Hangul_J_Sios ᆺ HANGUL JONGSEONG SIOS */
2036   { 0x0ee7, 0x11bb }, /*          Hangul_J_SsangSios ᆻ HANGUL JONGSEONG SSANGSIOS */
2037   { 0x0ee8, 0x11bc }, /*              Hangul_J_Ieung ᆼ HANGUL JONGSEONG IEUNG */
2038   { 0x0ee9, 0x11bd }, /*              Hangul_J_Jieuj ᆽ HANGUL JONGSEONG CIEUC */
2039   { 0x0eea, 0x11be }, /*              Hangul_J_Cieuc ᆾ HANGUL JONGSEONG CHIEUCH */
2040   { 0x0eeb, 0x11bf }, /*             Hangul_J_Khieuq ᆿ HANGUL JONGSEONG KHIEUKH */
2041   { 0x0eec, 0x11c0 }, /*              Hangul_J_Tieut ᇀ HANGUL JONGSEONG THIEUTH */
2042   { 0x0eed, 0x11c1 }, /*             Hangul_J_Phieuf ᇁ HANGUL JONGSEONG PHIEUPH */
2043   { 0x0eee, 0x11c2 }, /*              Hangul_J_Hieuh ᇂ HANGUL JONGSEONG HIEUH */
2044   { 0x0ef8, 0x11eb }, /*            Hangul_J_PanSios ᇫ HANGUL JONGSEONG PANSIOS */
2045   { 0x0efa, 0x11f9 }, /*        Hangul_J_YeorinHieuh ᇹ HANGUL JONGSEONG YEORINHIEUH */
2046   { 0x0aa2, 0x2002 }, /*                     enspace   EN SPACE */
2047   { 0x0aa1, 0x2003 }, /*                     emspace   EM SPACE */
2048   { 0x0aa3, 0x2004 }, /*                    em3space   THREE-PER-EM SPACE */
2049   { 0x0aa4, 0x2005 }, /*                    em4space   FOUR-PER-EM SPACE */
2050   { 0x0aa5, 0x2007 }, /*                  digitspace   FIGURE SPACE */
2051   { 0x0aa6, 0x2008 }, /*                  punctspace   PUNCTUATION SPACE */
2052   { 0x0aa7, 0x2009 }, /*                   thinspace   THIN SPACE */
2053   { 0x0aa8, 0x200a }, /*                   hairspace   HAIR SPACE */
2054   { 0x0abb, 0x2012 }, /*                     figdash ‒ FIGURE DASH */
2055   { 0x0aaa, 0x2013 }, /*                      endash – EN DASH */
2056   { 0x0aa9, 0x2014 }, /*                      emdash — EM DASH */
2057   { 0x07af, 0x2015 }, /*              Greek_horizbar ― HORIZONTAL BAR */
2058   { 0x0cdf, 0x2017 }, /*        hebrew_doublelowline ‗ DOUBLE LOW LINE */
2059   { 0x0ad0, 0x2018 }, /*         leftsinglequotemark ‘ LEFT SINGLE QUOTATION MARK */
2060   { 0x0ad1, 0x2019 }, /*        rightsinglequotemark ’ RIGHT SINGLE QUOTATION MARK */
2061   { 0x0afd, 0x201a }, /*          singlelowquotemark ‚ SINGLE LOW-9 QUOTATION MARK */
2062   { 0x0ad2, 0x201c }, /*         leftdoublequotemark “ LEFT DOUBLE QUOTATION MARK */
2063   { 0x0ad3, 0x201d }, /*        rightdoublequotemark ” RIGHT DOUBLE QUOTATION MARK */
2064   { 0x0afe, 0x201e }, /*          doublelowquotemark „ DOUBLE LOW-9 QUOTATION MARK */
2065   { 0x0af1, 0x2020 }, /*                      dagger † DAGGER */
2066   { 0x0af2, 0x2021 }, /*                doubledagger ‡ DOUBLE DAGGER */
2067   { 0x0ae6, 0x2022 }, /*          enfilledcircbullet • BULLET */
2068   { 0x0aae, 0x2026 }, /*                    ellipsis … HORIZONTAL ELLIPSIS */
2069   { 0x0ad6, 0x2032 }, /*                     minutes ′ PRIME */
2070   { 0x0ad7, 0x2033 }, /*                     seconds ″ DOUBLE PRIME */
2071   { 0x0afc, 0x2038 }, /*                       caret ‸ CARET */
2072   { 0x047e, 0x203e }, /*                    overline ‾ OVERLINE */
2073   { 0x20a0, 0x20a0 }, /*                     EcuSign ₠ EURO-CURRENCY SIGN */
2074   { 0x20a1, 0x20a1 }, /*                   ColonSign ₡ COLON SIGN */
2075   { 0x20a2, 0x20a2 }, /*                CruzeiroSign ₢ CRUZEIRO SIGN */
2076   { 0x20a3, 0x20a3 }, /*                  FFrancSign ₣ FRENCH FRANC SIGN */
2077   { 0x20a4, 0x20a4 }, /*                    LiraSign ₤ LIRA SIGN */
2078   { 0x20a5, 0x20a5 }, /*                    MillSign ₥ MILL SIGN */
2079   { 0x20a6, 0x20a6 }, /*                   NairaSign ₦ NAIRA SIGN */
2080   { 0x20a7, 0x20a7 }, /*                  PesetaSign ₧ PESETA SIGN */
2081   { 0x20a8, 0x20a8 }, /*                   RupeeSign ₨ RUPEE SIGN */
2082   { 0x0eff, 0x20a9 }, /*                  Korean_Won ₩ WON SIGN */
2083   { 0x20a9, 0x20a9 }, /*                     WonSign ₩ WON SIGN */
2084   { 0x20aa, 0x20aa }, /*               NewSheqelSign ₪ NEW SHEQEL SIGN */
2085   { 0x20ab, 0x20ab }, /*                    DongSign ₫ DONG SIGN */
2086   { 0x20ac, 0x20ac }, /*                    EuroSign € EURO SIGN */
2087   { 0x0ab8, 0x2105 }, /*                      careof ℅ CARE OF */
2088   { 0x06b0, 0x2116 }, /*                  numerosign № NUMERO SIGN */
2089   { 0x0afb, 0x2117 }, /*         phonographcopyright ℗ SOUND RECORDING COPYRIGHT */
2090   { 0x0ad4, 0x211e }, /*                prescription ℞ PRESCRIPTION TAKE */
2091   { 0x0ac9, 0x2122 }, /*                   trademark ™ TRADE MARK SIGN */
2092   { 0x0ab0, 0x2153 }, /*                    onethird ⅓ VULGAR FRACTION ONE THIRD */
2093   { 0x0ab1, 0x2154 }, /*                   twothirds ⅔ VULGAR FRACTION TWO THIRDS */
2094   { 0x0ab2, 0x2155 }, /*                    onefifth ⅕ VULGAR FRACTION ONE FIFTH */
2095   { 0x0ab3, 0x2156 }, /*                   twofifths ⅖ VULGAR FRACTION TWO FIFTHS */
2096   { 0x0ab4, 0x2157 }, /*                 threefifths ⅗ VULGAR FRACTION THREE FIFTHS */
2097   { 0x0ab5, 0x2158 }, /*                  fourfifths ⅘ VULGAR FRACTION FOUR FIFTHS */
2098   { 0x0ab6, 0x2159 }, /*                    onesixth ⅙ VULGAR FRACTION ONE SIXTH */
2099   { 0x0ab7, 0x215a }, /*                  fivesixths ⅚ VULGAR FRACTION FIVE SIXTHS */
2100   { 0x0ac3, 0x215b }, /*                   oneeighth ⅛ VULGAR FRACTION ONE EIGHTH */
2101   { 0x0ac4, 0x215c }, /*                threeeighths ⅜ VULGAR FRACTION THREE EIGHTHS */
2102   { 0x0ac5, 0x215d }, /*                 fiveeighths ⅝ VULGAR FRACTION FIVE EIGHTHS */
2103   { 0x0ac6, 0x215e }, /*                seveneighths ⅞ VULGAR FRACTION SEVEN EIGHTHS */
2104   { 0x08fb, 0x2190 }, /*                   leftarrow ← LEFTWARDS ARROW */
2105   { 0x08fc, 0x2191 }, /*                     uparrow ↑ UPWARDS ARROW */
2106   { 0x08fd, 0x2192 }, /*                  rightarrow → RIGHTWARDS ARROW */
2107   { 0x08fe, 0x2193 }, /*                   downarrow ↓ DOWNWARDS ARROW */
2108   { 0x08ce, 0x21d2 }, /*                     implies ⇒ RIGHTWARDS DOUBLE ARROW */
2109   { 0x08cd, 0x21d4 }, /*                    ifonlyif ⇔ LEFT RIGHT DOUBLE ARROW */
2110   { 0x08ef, 0x2202 }, /*           partialderivative ∂ PARTIAL DIFFERENTIAL */
2111   { 0x08c5, 0x2207 }, /*                       nabla ∇ NABLA */
2112   { 0x0bca, 0x2218 }, /*                         jot ∘ RING OPERATOR */
2113   { 0x08d6, 0x221a }, /*                     radical √ SQUARE ROOT */
2114   { 0x08c1, 0x221d }, /*                   variation ∝ PROPORTIONAL TO */
2115   { 0x08c2, 0x221e }, /*                    infinity ∞ INFINITY */
2116   { 0x08de, 0x2227 }, /*                  logicaland ∧ LOGICAL AND */
2117   { 0x0ba9, 0x2227 }, /*                     upcaret ∧ LOGICAL AND */
2118   { 0x08df, 0x2228 }, /*                   logicalor ∨ LOGICAL OR */
2119   { 0x0ba8, 0x2228 }, /*                   downcaret ∨ LOGICAL OR */
2120   { 0x08dc, 0x2229 }, /*                intersection ∩ INTERSECTION */
2121   { 0x0bc3, 0x2229 }, /*                      upshoe ∩ INTERSECTION */
2122   { 0x08dd, 0x222a }, /*                       union ∪ UNION */
2123   { 0x0bd6, 0x222a }, /*                    downshoe ∪ UNION */
2124   { 0x08bf, 0x222b }, /*                    integral ∫ INTEGRAL */
2125   { 0x08c0, 0x2234 }, /*                   therefore ∴ THEREFORE */
2126   { 0x08c8, 0x2245 }, /*                 approximate ≅ APPROXIMATELY EQUAL TO */
2127   { 0x08bd, 0x2260 }, /*                    notequal ≠ NOT EQUAL TO */
2128   { 0x08cf, 0x2261 }, /*                   identical ≡ IDENTICAL TO */
2129   { 0x08bc, 0x2264 }, /*               lessthanequal ≤ LESS-THAN OR EQUAL TO */
2130   { 0x08be, 0x2265 }, /*            greaterthanequal ≥ GREATER-THAN OR EQUAL TO */
2131   { 0x08da, 0x2282 }, /*                  includedin ⊂ SUBSET OF */
2132   { 0x0bda, 0x2282 }, /*                    leftshoe ⊂ SUBSET OF */
2133   { 0x08db, 0x2283 }, /*                    includes ⊃ SUPERSET OF */
2134   { 0x0bd8, 0x2283 }, /*                   rightshoe ⊃ SUPERSET OF */
2135   { 0x0bfc, 0x22a2 }, /*                   righttack ⊢ RIGHT TACK */
2136   { 0x0bdc, 0x22a3 }, /*                    lefttack ⊣ LEFT TACK */
2137   { 0x0bc2, 0x22a4 }, /*                    downtack ⊤ DOWN TACK */
2138   { 0x0bce, 0x22a5 }, /*                      uptack ⊥ UP TACK */
2139   { 0x0bd3, 0x2308 }, /*                     upstile ⌈ LEFT CEILING */
2140   { 0x0bc4, 0x230a }, /*                   downstile ⌊ LEFT FLOOR */
2141   { 0x0afa, 0x2315 }, /*           telephonerecorder ⌕ TELEPHONE RECORDER */
2142   { 0x08a4, 0x2320 }, /*                 topintegral ⌠ TOP HALF INTEGRAL */
2143   { 0x08a5, 0x2321 }, /*                 botintegral ⌡ BOTTOM HALF INTEGRAL */
2144   { 0x0abc, 0x2329 }, /*            leftanglebracket 〈 LEFT-POINTING ANGLE BRACKET */
2145   { 0x0abe, 0x232a }, /*           rightanglebracket 〉 RIGHT-POINTING ANGLE BRACKET */
2146   { 0x0bcc, 0x2395 }, /*                        quad ⎕ APL FUNCTIONAL SYMBOL QUAD (Unicode 3.0) */
2147   { 0x09e2, 0x2409 }, /*                          ht ␉ SYMBOL FOR HORIZONTAL TABULATION */
2148   { 0x09e5, 0x240a }, /*                          lf ␊ SYMBOL FOR LINE FEED */
2149   { 0x09e9, 0x240b }, /*                          vt ␋ SYMBOL FOR VERTICAL TABULATION */
2150   { 0x09e3, 0x240c }, /*                          ff ␌ SYMBOL FOR FORM FEED */
2151   { 0x09e4, 0x240d }, /*                          cr ␍ SYMBOL FOR CARRIAGE RETURN */
2152   { 0x09df, 0x2422 }, /*                       blank ␢ BLANK SYMBOL */
2153   { 0x09e8, 0x2424 }, /*                          nl ␤ SYMBOL FOR NEWLINE */
2154   { 0x09f1, 0x2500 }, /*              horizlinescan5 ─ BOX DRAWINGS LIGHT HORIZONTAL */
2155   { 0x08a6, 0x2502 }, /*               vertconnector │ BOX DRAWINGS LIGHT VERTICAL */
2156   { 0x09f8, 0x2502 }, /*                     vertbar │ BOX DRAWINGS LIGHT VERTICAL */
2157   { 0x09ec, 0x250c }, /*                upleftcorner ┌ BOX DRAWINGS LIGHT DOWN AND RIGHT */
2158   { 0x09eb, 0x2510 }, /*               uprightcorner ┐ BOX DRAWINGS LIGHT DOWN AND LEFT */
2159   { 0x09ed, 0x2514 }, /*               lowleftcorner └ BOX DRAWINGS LIGHT UP AND RIGHT */
2160   { 0x09ea, 0x2518 }, /*              lowrightcorner ┘ BOX DRAWINGS LIGHT UP AND LEFT */
2161   { 0x09f4, 0x251c }, /*                       leftt ├ BOX DRAWINGS LIGHT VERTICAL AND RIGHT */
2162   { 0x09f5, 0x2524 }, /*                      rightt ┤ BOX DRAWINGS LIGHT VERTICAL AND LEFT */
2163   { 0x09f7, 0x252c }, /*                        topt ┬ BOX DRAWINGS LIGHT DOWN AND HORIZONTAL */
2164   { 0x09f6, 0x2534 }, /*                        bott ┴ BOX DRAWINGS LIGHT UP AND HORIZONTAL */
2165   { 0x09ee, 0x253c }, /*               crossinglines ┼ BOX DRAWINGS LIGHT VERTICAL AND HORIZONTAL */
2166   { 0x09e1, 0x2592 }, /*                checkerboard ▒ MEDIUM SHADE */
2167   { 0x0adf, 0x25a0 }, /*                emfilledrect ■ BLACK SQUARE */
2168   { 0x0acf, 0x25a1 }, /*             emopenrectangle □ WHITE SQUARE */
2169   { 0x0ae7, 0x25aa }, /*            enfilledsqbullet ▪ BLACK SMALL SQUARE */
2170   { 0x0ae1, 0x25ab }, /*          enopensquarebullet ▫ WHITE SMALL SQUARE */
2171   { 0x0adb, 0x25ac }, /*            filledrectbullet ▬ BLACK RECTANGLE */
2172   { 0x0ae2, 0x25ad }, /*              openrectbullet ▭ WHITE RECTANGLE */
2173   { 0x0ae8, 0x25b2 }, /*           filledtribulletup ▲ BLACK UP-POINTING TRIANGLE */
2174   { 0x0ae3, 0x25b3 }, /*             opentribulletup △ WHITE UP-POINTING TRIANGLE */
2175   { 0x0add, 0x25b6 }, /*        filledrighttribullet ▶ BLACK RIGHT-POINTING TRIANGLE */
2176   { 0x0acd, 0x25b7 }, /*           rightopentriangle ▷ WHITE RIGHT-POINTING TRIANGLE */
2177   { 0x0ae9, 0x25bc }, /*         filledtribulletdown ▼ BLACK DOWN-POINTING TRIANGLE */
2178   { 0x0ae4, 0x25bd }, /*           opentribulletdown ▽ WHITE DOWN-POINTING TRIANGLE */
2179   { 0x0adc, 0x25c0 }, /*         filledlefttribullet ◀ BLACK LEFT-POINTING TRIANGLE */
2180   { 0x0acc, 0x25c1 }, /*            leftopentriangle ◁ WHITE LEFT-POINTING TRIANGLE */
2181   { 0x09e0, 0x25c6 }, /*                soliddiamond ◆ BLACK DIAMOND */
2182   { 0x0ace, 0x25cb }, /*                emopencircle ○ WHITE CIRCLE */
2183   { 0x0bcf, 0x25cb }, /*                      circle ○ WHITE CIRCLE */
2184   { 0x0ade, 0x25cf }, /*              emfilledcircle ● BLACK CIRCLE */
2185   { 0x0ae0, 0x25e6 }, /*            enopencircbullet ◦ WHITE BULLET */
2186   { 0x0ae5, 0x2606 }, /*                    openstar ☆ WHITE STAR */
2187   { 0x0af9, 0x260e }, /*                   telephone ☎ BLACK TELEPHONE */
2188   { 0x0aca, 0x2613 }, /*               signaturemark ☓ SALTIRE */
2189   { 0x0aea, 0x261c }, /*                 leftpointer ☜ WHITE LEFT POINTING INDEX */
2190   { 0x0aeb, 0x261e }, /*                rightpointer ☞ WHITE RIGHT POINTING INDEX */
2191   { 0x0af8, 0x2640 }, /*                femalesymbol ♀ FEMALE SIGN */
2192   { 0x0af7, 0x2642 }, /*                  malesymbol ♂ MALE SIGN */
2193   { 0x0aec, 0x2663 }, /*                        club ♣ BLACK CLUB SUIT */
2194   { 0x0aee, 0x2665 }, /*                       heart ♥ BLACK HEART SUIT */
2195   { 0x0aed, 0x2666 }, /*                     diamond ♦ BLACK DIAMOND SUIT */
2196   { 0x0af6, 0x266d }, /*                 musicalflat ♭ MUSIC FLAT SIGN */
2197   { 0x0af5, 0x266f }, /*                musicalsharp ♯ MUSIC SHARP SIGN */
2198   { 0x0af3, 0x2713 }, /*                   checkmark ✓ CHECK MARK */
2199   { 0x0af4, 0x2717 }, /*                 ballotcross ✗ BALLOT X */
2200   { 0x0ad9, 0x271d }, /*                  latincross ✝ LATIN CROSS */
2201   { 0x0af0, 0x2720 }, /*                maltesecross ✠ MALTESE CROSS */
2202   { 0x04a4, 0x3001 }, /*                  kana_comma 、 IDEOGRAPHIC COMMA */
2203   { 0x04a1, 0x3002 }, /*               kana_fullstop 。 IDEOGRAPHIC FULL STOP */
2204   { 0x04a2, 0x300c }, /*         kana_openingbracket 「 LEFT CORNER BRACKET */
2205   { 0x04a3, 0x300d }, /*         kana_closingbracket 」 RIGHT CORNER BRACKET */
2206   { 0x04de, 0x309b }, /*                 voicedsound ゛ KATAKANA-HIRAGANA VOICED SOUND MARK */
2207   { 0x04df, 0x309c }, /*             semivoicedsound ゜ KATAKANA-HIRAGANA SEMI-VOICED SOUND MARK */
2208   { 0x04a7, 0x30a1 }, /*                      kana_a ァ KATAKANA LETTER SMALL A */
2209   { 0x04b1, 0x30a2 }, /*                      kana_A ア KATAKANA LETTER A */
2210   { 0x04a8, 0x30a3 }, /*                      kana_i ィ KATAKANA LETTER SMALL I */
2211   { 0x04b2, 0x30a4 }, /*                      kana_I イ KATAKANA LETTER I */
2212   { 0x04a9, 0x30a5 }, /*                      kana_u ゥ KATAKANA LETTER SMALL U */
2213   { 0x04b3, 0x30a6 }, /*                      kana_U ウ KATAKANA LETTER U */
2214   { 0x04aa, 0x30a7 }, /*                      kana_e ェ KATAKANA LETTER SMALL E */
2215   { 0x04b4, 0x30a8 }, /*                      kana_E エ KATAKANA LETTER E */
2216   { 0x04ab, 0x30a9 }, /*                      kana_o ォ KATAKANA LETTER SMALL O */
2217   { 0x04b5, 0x30aa }, /*                      kana_O オ KATAKANA LETTER O */
2218   { 0x04b6, 0x30ab }, /*                     kana_KA カ KATAKANA LETTER KA */
2219   { 0x04b7, 0x30ad }, /*                     kana_KI キ KATAKANA LETTER KI */
2220   { 0x04b8, 0x30af }, /*                     kana_KU ク KATAKANA LETTER KU */
2221   { 0x04b9, 0x30b1 }, /*                     kana_KE ケ KATAKANA LETTER KE */
2222   { 0x04ba, 0x30b3 }, /*                     kana_KO コ KATAKANA LETTER KO */
2223   { 0x04bb, 0x30b5 }, /*                     kana_SA サ KATAKANA LETTER SA */
2224   { 0x04bc, 0x30b7 }, /*                    kana_SHI シ KATAKANA LETTER SI */
2225   { 0x04bd, 0x30b9 }, /*                     kana_SU ス KATAKANA LETTER SU */
2226   { 0x04be, 0x30bb }, /*                     kana_SE セ KATAKANA LETTER SE */
2227   { 0x04bf, 0x30bd }, /*                     kana_SO ソ KATAKANA LETTER SO */
2228   { 0x04c0, 0x30bf }, /*                     kana_TA タ KATAKANA LETTER TA */
2229   { 0x04c1, 0x30c1 }, /*                    kana_CHI チ KATAKANA LETTER TI */
2230   { 0x04af, 0x30c3 }, /*                    kana_tsu ッ KATAKANA LETTER SMALL TU */
2231   { 0x04c2, 0x30c4 }, /*                    kana_TSU ツ KATAKANA LETTER TU */
2232   { 0x04c3, 0x30c6 }, /*                     kana_TE テ KATAKANA LETTER TE */
2233   { 0x04c4, 0x30c8 }, /*                     kana_TO ト KATAKANA LETTER TO */
2234   { 0x04c5, 0x30ca }, /*                     kana_NA ナ KATAKANA LETTER NA */
2235   { 0x04c6, 0x30cb }, /*                     kana_NI ニ KATAKANA LETTER NI */
2236   { 0x04c7, 0x30cc }, /*                     kana_NU ヌ KATAKANA LETTER NU */
2237   { 0x04c8, 0x30cd }, /*                     kana_NE ネ KATAKANA LETTER NE */
2238   { 0x04c9, 0x30ce }, /*                     kana_NO ノ KATAKANA LETTER NO */
2239   { 0x04ca, 0x30cf }, /*                     kana_HA ハ KATAKANA LETTER HA */
2240   { 0x04cb, 0x30d2 }, /*                     kana_HI ヒ KATAKANA LETTER HI */
2241   { 0x04cc, 0x30d5 }, /*                     kana_FU フ KATAKANA LETTER HU */
2242   { 0x04cd, 0x30d8 }, /*                     kana_HE ヘ KATAKANA LETTER HE */
2243   { 0x04ce, 0x30db }, /*                     kana_HO ホ KATAKANA LETTER HO */
2244   { 0x04cf, 0x30de }, /*                     kana_MA マ KATAKANA LETTER MA */
2245   { 0x04d0, 0x30df }, /*                     kana_MI ミ KATAKANA LETTER MI */
2246   { 0x04d1, 0x30e0 }, /*                     kana_MU ム KATAKANA LETTER MU */
2247   { 0x04d2, 0x30e1 }, /*                     kana_ME メ KATAKANA LETTER ME */
2248   { 0x04d3, 0x30e2 }, /*                     kana_MO モ KATAKANA LETTER MO */
2249   { 0x04ac, 0x30e3 }, /*                     kana_ya ャ KATAKANA LETTER SMALL YA */
2250   { 0x04d4, 0x30e4 }, /*                     kana_YA ヤ KATAKANA LETTER YA */
2251   { 0x04ad, 0x30e5 }, /*                     kana_yu ュ KATAKANA LETTER SMALL YU */
2252   { 0x04d5, 0x30e6 }, /*                     kana_YU ユ KATAKANA LETTER YU */
2253   { 0x04ae, 0x30e7 }, /*                     kana_yo ョ KATAKANA LETTER SMALL YO */
2254   { 0x04d6, 0x30e8 }, /*                     kana_YO ヨ KATAKANA LETTER YO */
2255   { 0x04d7, 0x30e9 }, /*                     kana_RA ラ KATAKANA LETTER RA */
2256   { 0x04d8, 0x30ea }, /*                     kana_RI リ KATAKANA LETTER RI */
2257   { 0x04d9, 0x30eb }, /*                     kana_RU ル KATAKANA LETTER RU */
2258   { 0x04da, 0x30ec }, /*                     kana_RE レ KATAKANA LETTER RE */
2259   { 0x04db, 0x30ed }, /*                     kana_RO ロ KATAKANA LETTER RO */
2260   { 0x04dc, 0x30ef }, /*                     kana_WA ワ KATAKANA LETTER WA */
2261   { 0x04a6, 0x30f2 }, /*                     kana_WO ヲ KATAKANA LETTER WO */
2262   { 0x04dd, 0x30f3 }, /*                      kana_N ン KATAKANA LETTER N */
2263   { 0x04a5, 0x30fb }, /*            kana_conjunctive ・ KATAKANA MIDDLE DOT */
2264   { 0x04b0, 0x30fc }, /*              prolongedsound ー KATAKANA-HIRAGANA PROLONGED SOUND MARK */
2265   { 0x0ea1, 0x3131 }, /*               Hangul_Kiyeog ㄱ HANGUL LETTER KIYEOK */
2266   { 0x0ea2, 0x3132 }, /*          Hangul_SsangKiyeog ㄲ HANGUL LETTER SSANGKIYEOK */
2267   { 0x0ea3, 0x3133 }, /*           Hangul_KiyeogSios ㄳ HANGUL LETTER KIYEOK-SIOS */
2268   { 0x0ea4, 0x3134 }, /*                Hangul_Nieun ㄴ HANGUL LETTER NIEUN */
2269   { 0x0ea5, 0x3135 }, /*           Hangul_NieunJieuj ㄵ HANGUL LETTER NIEUN-CIEUC */
2270   { 0x0ea6, 0x3136 }, /*           Hangul_NieunHieuh ㄶ HANGUL LETTER NIEUN-HIEUH */
2271   { 0x0ea7, 0x3137 }, /*               Hangul_Dikeud ㄷ HANGUL LETTER TIKEUT */
2272   { 0x0ea8, 0x3138 }, /*          Hangul_SsangDikeud ㄸ HANGUL LETTER SSANGTIKEUT */
2273   { 0x0ea9, 0x3139 }, /*                Hangul_Rieul ㄹ HANGUL LETTER RIEUL */
2274   { 0x0eaa, 0x313a }, /*          Hangul_RieulKiyeog ㄺ HANGUL LETTER RIEUL-KIYEOK */
2275   { 0x0eab, 0x313b }, /*           Hangul_RieulMieum ㄻ HANGUL LETTER RIEUL-MIEUM */
2276   { 0x0eac, 0x313c }, /*           Hangul_RieulPieub ㄼ HANGUL LETTER RIEUL-PIEUP */
2277   { 0x0ead, 0x313d }, /*            Hangul_RieulSios ㄽ HANGUL LETTER RIEUL-SIOS */
2278   { 0x0eae, 0x313e }, /*           Hangul_RieulTieut ㄾ HANGUL LETTER RIEUL-THIEUTH */
2279   { 0x0eaf, 0x313f }, /*          Hangul_RieulPhieuf ㄿ HANGUL LETTER RIEUL-PHIEUPH */
2280   { 0x0eb0, 0x3140 }, /*           Hangul_RieulHieuh ㅀ HANGUL LETTER RIEUL-HIEUH */
2281   { 0x0eb1, 0x3141 }, /*                Hangul_Mieum ㅁ HANGUL LETTER MIEUM */
2282   { 0x0eb2, 0x3142 }, /*                Hangul_Pieub ㅂ HANGUL LETTER PIEUP */
2283   { 0x0eb3, 0x3143 }, /*           Hangul_SsangPieub ㅃ HANGUL LETTER SSANGPIEUP */
2284   { 0x0eb4, 0x3144 }, /*            Hangul_PieubSios ㅄ HANGUL LETTER PIEUP-SIOS */
2285   { 0x0eb5, 0x3145 }, /*                 Hangul_Sios ㅅ HANGUL LETTER SIOS */
2286   { 0x0eb6, 0x3146 }, /*            Hangul_SsangSios ㅆ HANGUL LETTER SSANGSIOS */
2287   { 0x0eb7, 0x3147 }, /*                Hangul_Ieung ㅇ HANGUL LETTER IEUNG */
2288   { 0x0eb8, 0x3148 }, /*                Hangul_Jieuj ㅈ HANGUL LETTER CIEUC */
2289   { 0x0eb9, 0x3149 }, /*           Hangul_SsangJieuj ㅉ HANGUL LETTER SSANGCIEUC */
2290   { 0x0eba, 0x314a }, /*                Hangul_Cieuc ㅊ HANGUL LETTER CHIEUCH */
2291   { 0x0ebb, 0x314b }, /*               Hangul_Khieuq ㅋ HANGUL LETTER KHIEUKH */
2292   { 0x0ebc, 0x314c }, /*                Hangul_Tieut ㅌ HANGUL LETTER THIEUTH */
2293   { 0x0ebd, 0x314d }, /*               Hangul_Phieuf ㅍ HANGUL LETTER PHIEUPH */
2294   { 0x0ebe, 0x314e }, /*                Hangul_Hieuh ㅎ HANGUL LETTER HIEUH */
2295   { 0x0ebf, 0x314f }, /*                    Hangul_A ㅏ HANGUL LETTER A */
2296   { 0x0ec0, 0x3150 }, /*                   Hangul_AE ㅐ HANGUL LETTER AE */
2297   { 0x0ec1, 0x3151 }, /*                   Hangul_YA ㅑ HANGUL LETTER YA */
2298   { 0x0ec2, 0x3152 }, /*                  Hangul_YAE ㅒ HANGUL LETTER YAE */
2299   { 0x0ec3, 0x3153 }, /*                   Hangul_EO ㅓ HANGUL LETTER EO */
2300   { 0x0ec4, 0x3154 }, /*                    Hangul_E ㅔ HANGUL LETTER E */
2301   { 0x0ec5, 0x3155 }, /*                  Hangul_YEO ㅕ HANGUL LETTER YEO */
2302   { 0x0ec6, 0x3156 }, /*                   Hangul_YE ㅖ HANGUL LETTER YE */
2303   { 0x0ec7, 0x3157 }, /*                    Hangul_O ㅗ HANGUL LETTER O */
2304   { 0x0ec8, 0x3158 }, /*                   Hangul_WA ㅘ HANGUL LETTER WA */
2305   { 0x0ec9, 0x3159 }, /*                  Hangul_WAE ㅙ HANGUL LETTER WAE */
2306   { 0x0eca, 0x315a }, /*                   Hangul_OE ㅚ HANGUL LETTER OE */
2307   { 0x0ecb, 0x315b }, /*                   Hangul_YO ㅛ HANGUL LETTER YO */
2308   { 0x0ecc, 0x315c }, /*                    Hangul_U ㅜ HANGUL LETTER U */
2309   { 0x0ecd, 0x315d }, /*                  Hangul_WEO ㅝ HANGUL LETTER WEO */
2310   { 0x0ece, 0x315e }, /*                   Hangul_WE ㅞ HANGUL LETTER WE */
2311   { 0x0ecf, 0x315f }, /*                   Hangul_WI ㅟ HANGUL LETTER WI */
2312   { 0x0ed0, 0x3160 }, /*                   Hangul_YU ㅠ HANGUL LETTER YU */
2313   { 0x0ed1, 0x3161 }, /*                   Hangul_EU ㅡ HANGUL LETTER EU */
2314   { 0x0ed2, 0x3162 }, /*                   Hangul_YI ㅢ HANGUL LETTER YI */
2315   { 0x0ed3, 0x3163 }, /*                    Hangul_I ㅣ HANGUL LETTER I */
2316   { 0x0eef, 0x316d }, /*     Hangul_RieulYeorinHieuh ㅭ HANGUL LETTER RIEUL-YEORINHIEUH */
2317   { 0x0ef0, 0x3171 }, /*    Hangul_SunkyeongeumMieum ㅱ HANGUL LETTER KAPYEOUNMIEUM */
2318   { 0x0ef1, 0x3178 }, /*    Hangul_SunkyeongeumPieub ㅸ HANGUL LETTER KAPYEOUNPIEUP */
2319   { 0x0ef2, 0x317f }, /*              Hangul_PanSios ㅿ HANGUL LETTER PANSIOS */
2320   { 0x0ef4, 0x3184 }, /*   Hangul_SunkyeongeumPhieuf ㆄ HANGUL LETTER KAPYEOUNPHIEUPH */
2321   { 0x0ef5, 0x3186 }, /*          Hangul_YeorinHieuh ㆆ HANGUL LETTER YEORINHIEUH */
2322   { 0x0ef6, 0x318d }, /*                Hangul_AraeA ㆍ HANGUL LETTER ARAEA */
2323   { 0x0ef7, 0x318e }, /*               Hangul_AraeAE ㆎ HANGUL LETTER ARAEAE */
2324 };
2325
2326 static guint
2327 unicode_to_keyval (wchar_t ucs)
2328 {
2329   int min = 0;
2330   int max = sizeof(u2ktab) / sizeof(u2ktab[0]) - 1;
2331   int mid;
2332
2333   /* First check for Latin-1 characters (1:1 mapping) */
2334   if ((ucs >= 0x0020 && ucs <= 0x007e) ||
2335       (ucs >= 0x00a0 && ucs <= 0x00ff))
2336     return ucs;
2337
2338   /* Binary search in table */
2339   while (max >= min) {
2340     mid = (min + max) / 2;
2341     if (u2ktab[mid].ucs < ucs)
2342       min = mid + 1;
2343     else if (u2ktab[mid].ucs > ucs)
2344       max = mid - 1;
2345     else {
2346       /* found it */
2347       return u2ktab[mid].keysym;
2348     }
2349   }
2350   
2351   /*
2352    * No matching keysym value found, return Unicode value plus 0x01000000
2353    * (a convention introduced in the UTF-8 work on xterm).
2354    */
2355   return ucs | 0x01000000;
2356 }
2357
2358 static void
2359 build_key_event_state (GdkEvent *event)
2360 {
2361   event->key.state = 0;
2362   if (GetKeyState (VK_SHIFT) < 0)
2363     event->key.state |= GDK_SHIFT_MASK;
2364   if (GetKeyState (VK_CAPITAL) & 0x1)
2365     event->key.state |= GDK_LOCK_MASK;
2366   if (!is_AltGr_key)
2367     {
2368       if (GetKeyState (VK_CONTROL) < 0)
2369         {
2370           event->key.state |= GDK_CONTROL_MASK;
2371           if (event->key.keyval < ' ')
2372             event->key.keyval += '@';
2373         }
2374       else if (event->key.keyval < ' ')
2375         {
2376           event->key.state |= GDK_CONTROL_MASK;
2377           event->key.keyval += '@';
2378         }
2379       if (GetKeyState (VK_MENU) < 0)
2380         event->key.state |= GDK_MOD1_MASK;
2381     }
2382 }
2383
2384 static gint
2385 build_pointer_event_state (MSG *xevent)
2386 {
2387   gint state;
2388   
2389   state = 0;
2390   if (xevent->wParam & MK_CONTROL)
2391     state |= GDK_CONTROL_MASK;
2392   if (xevent->wParam & MK_LBUTTON)
2393     state |= GDK_BUTTON1_MASK;
2394   if (xevent->wParam & MK_MBUTTON)
2395     state |= GDK_BUTTON2_MASK;
2396   if (xevent->wParam & MK_RBUTTON)
2397     state |= GDK_BUTTON3_MASK;
2398   if (xevent->wParam & MK_SHIFT)
2399     state |= GDK_SHIFT_MASK;
2400   if (GetKeyState (VK_MENU) < 0)
2401     state |= GDK_MOD1_MASK;
2402   if (GetKeyState (VK_CAPITAL) & 0x1)
2403     state |= GDK_LOCK_MASK;
2404
2405   return state;
2406 }
2407
2408
2409 static void
2410 build_keypress_event (GdkWindowWin32Data *windata,
2411                       GdkEvent           *event,
2412                       MSG                *xevent)
2413 {
2414   HIMC hIMC;
2415   gint i, bytesleft, bytecount, ucount, ucleft, len;
2416   guchar buf[100], *bp;
2417   wchar_t wbuf[100], *wcp;
2418
2419   event->key.type = GDK_KEY_PRESS;
2420   event->key.time = xevent->time;
2421   
2422   if (xevent->message == WM_IME_COMPOSITION)
2423     {
2424       hIMC = ImmGetContext (xevent->hwnd);
2425
2426       bytecount = ImmGetCompositionStringW (hIMC, GCS_RESULTSTR,
2427                                             wbuf, sizeof (wbuf));
2428       ucount = bytecount / 2;
2429     }
2430   else
2431     {
2432       if (xevent->message == WM_CHAR)
2433         {
2434           bytecount = MIN ((xevent->lParam & 0xFFFF), sizeof (buf));
2435           for (i = 0; i < bytecount; i++)
2436             buf[i] = xevent->wParam;
2437         }
2438       else /* WM_IME_CHAR */
2439         {
2440           event->key.keyval = GDK_VoidSymbol;
2441           if (xevent->wParam & 0xFF00)
2442             {
2443               /* Contrary to some versions of the documentation,
2444                * the lead byte is the most significant byte.
2445                */
2446               buf[0] = ((xevent->wParam >> 8) & 0xFF);
2447               buf[1] = (xevent->wParam & 0xFF);
2448               bytecount = 2;
2449             }
2450           else
2451             {
2452               buf[0] = (xevent->wParam & 0xFF);
2453               bytecount = 1;
2454             }
2455         }
2456
2457       /* Convert from the window's current code page
2458        * to Unicode. Then convert to UTF-8.
2459        * We don't handle the surrogate stuff. Should we?
2460        */
2461       ucount = MultiByteToWideChar (windata->charset_info.ciACP,
2462                                     0, buf, bytecount,
2463                                     wbuf, sizeof (wbuf) / sizeof (wbuf[0]));
2464       
2465     }
2466   if (ucount == 0)
2467     event->key.keyval = GDK_VoidSymbol;
2468   else if (xevent->message == WM_CHAR)
2469     if (xevent->wParam < ' ')
2470       event->key.keyval = xevent->wParam + '@';
2471     else
2472       event->key.keyval = unicode_to_keyval (wbuf[0]);
2473   else
2474     event->key.keyval = GDK_VoidSymbol;
2475
2476   build_key_event_state (event);
2477
2478   ucleft = ucount;
2479   len = 0;
2480   wcp = wbuf;
2481   while (ucleft-- > 0)
2482     {
2483       wchar_t c = *wcp++;
2484
2485       if (c < 0x80)
2486         len += 1;
2487       else if (c < 0x800)
2488         len += 2;
2489       else
2490         len += 3;
2491     }
2492
2493   event->key.string = g_malloc (len + 1);
2494   event->key.length = len;
2495   
2496   ucleft = ucount;
2497   wcp = wbuf;
2498   bp = event->key.string;
2499   while (ucleft-- > 0)
2500     {
2501       int first;
2502       int i;
2503       wchar_t c = *wcp++;
2504
2505       if (c < 0x80)
2506         {
2507           first = 0;
2508           len = 1;
2509         }
2510       else if (c < 0x800)
2511         {
2512           first = 0xc0;
2513           len = 2;
2514         }
2515       else
2516         {
2517           first = 0xe0;
2518           len = 3;
2519         }
2520
2521 #if 1      
2522       /* Woo-hoo! */
2523       switch (len)
2524         {
2525         case 3: bp[2] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
2526         case 2: bp[1] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
2527         case 1: bp[0] = c | first;
2528         }
2529 #else
2530       for (i = len - 1; i > 0; --i)
2531         {
2532           bp[i] = (c & 0x3f) | 0x80;
2533           c >>= 6;
2534         }
2535       bp[0] = c | first;
2536 #endif
2537
2538       bp += len;
2539     }
2540   *bp = 0;
2541 }
2542
2543 static void
2544 build_keyrelease_event (GdkWindowWin32Data *windata,
2545                         GdkEvent           *event,
2546                         MSG                *xevent)
2547 {
2548   guchar buf;
2549   wchar_t wbuf;
2550
2551   event->key.type = GDK_KEY_RELEASE;
2552   event->key.time = xevent->time;
2553
2554   if (xevent->message == WM_CHAR)
2555     if (xevent->wParam < ' ')
2556       event->key.keyval = xevent->wParam + '@';
2557     else
2558       {
2559         buf = xevent->wParam;
2560         MultiByteToWideChar (windata->charset_info.ciACP,
2561                              0, &buf, 1, &wbuf, 1);
2562
2563         event->key.keyval = unicode_to_keyval (wbuf);
2564       }
2565   else
2566     event->key.keyval = GDK_VoidSymbol;
2567   build_key_event_state (event);
2568   event->key.string = NULL;
2569   event->key.length = 0;
2570 }
2571
2572 static void
2573 print_event_state (gint state)
2574 {
2575   if (state & GDK_SHIFT_MASK)
2576     g_print ("SHIFT ");
2577   if (state & GDK_LOCK_MASK)
2578     g_print ("LOCK ");
2579   if (state & GDK_CONTROL_MASK)
2580     g_print ("CONTROL ");
2581   if (state & GDK_MOD1_MASK)
2582     g_print ("MOD1 ");
2583   if (state & GDK_BUTTON1_MASK)
2584     g_print ("BUTTON1 ");
2585   if (state & GDK_BUTTON2_MASK)
2586     g_print ("BUTTON2 ");
2587   if (state & GDK_BUTTON3_MASK)
2588     g_print ("BUTTON3 ");
2589 }
2590
2591 static void
2592 print_event (GdkEvent *event)
2593 {
2594   gchar *escaped, *kvname;
2595
2596   switch (event->any.type)
2597     {
2598     case GDK_NOTHING: g_print ("GDK_NOTHING "); break;
2599     case GDK_DELETE: g_print ("GDK_DELETE "); break;
2600     case GDK_DESTROY: g_print ("GDK_DESTROY "); break;
2601     case GDK_EXPOSE: g_print ("GDK_EXPOSE "); break;
2602     case GDK_MOTION_NOTIFY: g_print ("GDK_MOTION_NOTIFY "); break;
2603     case GDK_BUTTON_PRESS: g_print ("GDK_BUTTON_PRESS "); break;
2604     case GDK_2BUTTON_PRESS: g_print ("GDK_2BUTTON_PRESS "); break;
2605     case GDK_3BUTTON_PRESS: g_print ("GDK_3BUTTON_PRESS "); break;
2606     case GDK_BUTTON_RELEASE: g_print ("GDK_BUTTON_RELEASE "); break;
2607     case GDK_KEY_PRESS: g_print ("GDK_KEY_PRESS "); break;
2608     case GDK_KEY_RELEASE: g_print ("GDK_KEY_RELEASE "); break;
2609     case GDK_ENTER_NOTIFY: g_print ("GDK_ENTER_NOTIFY "); break;
2610     case GDK_LEAVE_NOTIFY: g_print ("GDK_LEAVE_NOTIFY "); break;
2611     case GDK_FOCUS_CHANGE: g_print ("GDK_FOCUS_CHANGE "); break;
2612     case GDK_CONFIGURE: g_print ("GDK_CONFIGURE "); break;
2613     case GDK_MAP: g_print ("GDK_MAP "); break;
2614     case GDK_UNMAP: g_print ("GDK_UNMAP "); break;
2615     case GDK_PROPERTY_NOTIFY: g_print ("GDK_PROPERTY_NOTIFY "); break;
2616     case GDK_SELECTION_CLEAR: g_print ("GDK_SELECTION_CLEAR "); break;
2617     case GDK_SELECTION_REQUEST: g_print ("GDK_SELECTION_REQUEST "); break;
2618     case GDK_SELECTION_NOTIFY: g_print ("GDK_SELECTION_NOTIFY "); break;
2619     case GDK_PROXIMITY_IN: g_print ("GDK_PROXIMITY_IN "); break;
2620     case GDK_PROXIMITY_OUT: g_print ("GDK_PROXIMITY_OUT "); break;
2621     case GDK_DRAG_ENTER: g_print ("GDK_DRAG_ENTER "); break;
2622     case GDK_DRAG_LEAVE: g_print ("GDK_DRAG_LEAVE "); break;
2623     case GDK_DRAG_MOTION: g_print ("GDK_DRAG_MOTION "); break;
2624     case GDK_DRAG_STATUS: g_print ("GDK_DRAG_STATUS "); break;
2625     case GDK_DROP_START: g_print ("GDK_DROP_START "); break;
2626     case GDK_DROP_FINISHED: g_print ("GDK_DROP_FINISHED "); break;
2627     case GDK_CLIENT_EVENT: g_print ("GDK_CLIENT_EVENT "); break;
2628     case GDK_VISIBILITY_NOTIFY: g_print ("GDK_VISIBILITY_NOTIFY "); break;
2629     case GDK_NO_EXPOSE: g_print ("GDK_NO_EXPOSE "); break;
2630     case GDK_SCROLL: g_print ("GDK_SCROLL "); break;
2631     }
2632   g_print ("%#x ", GDK_DRAWABLE_XID (event->any.window));
2633
2634   switch (event->any.type)
2635     {
2636     case GDK_EXPOSE:
2637       g_print ("%dx%d@+%d+%d %d",
2638                event->expose.area.width,
2639                event->expose.area.height,
2640                event->expose.area.x,
2641                event->expose.area.y,
2642                event->expose.count);
2643       break;
2644     case GDK_MOTION_NOTIFY:
2645       g_print ("(%.4g,%.4g) %s",
2646                event->motion.x, event->motion.y,
2647                event->motion.is_hint ? "HINT " : "");
2648       print_event_state (event->motion.state);
2649       break;
2650     case GDK_BUTTON_PRESS:
2651     case GDK_2BUTTON_PRESS:
2652     case GDK_3BUTTON_PRESS:
2653     case GDK_BUTTON_RELEASE:
2654       g_print ("%d (%.4g,%.4g) ",
2655                event->button.button,
2656                event->button.x, event->button.y);
2657       print_event_state (event->button.state);
2658       break;
2659     case GDK_KEY_PRESS: 
2660     case GDK_KEY_RELEASE:
2661       if (event->key.length == 0)
2662         escaped = g_strdup ("");
2663       else
2664         escaped = g_strescape (event->key.string, NULL);
2665       kvname = gdk_keyval_name (event->key.keyval);
2666       g_print ("%s %d:\"%s\" ",
2667                (kvname ? kvname : "??"),
2668                event->key.length,
2669                escaped);
2670       g_free (escaped);
2671       print_event_state (event->key.state);
2672       break;
2673     case GDK_ENTER_NOTIFY:
2674     case GDK_LEAVE_NOTIFY:
2675       g_print ("%s ",
2676                (event->crossing.detail == GDK_NOTIFY_INFERIOR ? "INFERIOR" :
2677                 (event->crossing.detail == GDK_NOTIFY_ANCESTOR ? "ANCESTOR" :
2678                  (event->crossing.detail == GDK_NOTIFY_NONLINEAR ? "NONLINEAR" :
2679                   "???"))));
2680       break;
2681     case GDK_SCROLL:
2682       g_print ("%s ",
2683                (event->scroll.direction == GDK_SCROLL_UP ? "UP" :
2684                 (event->scroll.direction == GDK_SCROLL_DOWN ? "DOWN" :
2685                  (event->scroll.direction == GDK_SCROLL_LEFT ? "LEFT" :
2686                   (event->scroll.direction == GDK_SCROLL_RIGHT ? "RIGHT" :
2687                    "???")))));
2688       print_event_state (event->scroll.state);
2689       break;
2690     }  
2691   g_print ("\n");
2692 }
2693
2694 static void
2695 synthesize_crossing_events (GdkWindow *window,
2696                             MSG       *xevent)
2697 {
2698   GdkEvent *event;
2699   
2700   /* If we are not using TrackMouseEvent, generate a leave notify
2701    * event if necessary
2702    */
2703   if (p_TrackMouseEvent == NULL
2704       && curWnd
2705       && (GDK_WINDOW_WIN32DATA (curWnd)->event_mask & GDK_LEAVE_NOTIFY_MASK))
2706     {
2707       GDK_NOTE (EVENTS, g_print ("synthesizing LEAVE_NOTIFY event\n"));
2708
2709       event = gdk_event_new ();
2710       event->crossing.type = GDK_LEAVE_NOTIFY;
2711       event->crossing.window = curWnd;
2712       gdk_drawable_ref (event->crossing.window);
2713       event->crossing.subwindow = NULL;
2714       event->crossing.time = xevent->time;
2715       event->crossing.x = curX;
2716       event->crossing.y = curY;
2717       event->crossing.x_root = curXroot;
2718       event->crossing.y_root = curYroot;
2719       event->crossing.mode = GDK_CROSSING_NORMAL;
2720       if (IsChild (GDK_DRAWABLE_XID (curWnd), GDK_DRAWABLE_XID (window)))
2721         event->crossing.detail = GDK_NOTIFY_INFERIOR;
2722       else if (IsChild (GDK_DRAWABLE_XID (window), GDK_DRAWABLE_XID (curWnd)))
2723         event->crossing.detail = GDK_NOTIFY_ANCESTOR;
2724       else
2725         event->crossing.detail = GDK_NOTIFY_NONLINEAR;
2726
2727       event->crossing.focus = TRUE; /* ??? */
2728       event->crossing.state = 0; /* ??? */
2729
2730       gdk_event_queue_append (event);
2731       GDK_NOTE (EVENTS, print_event (event));
2732     }
2733
2734   if (GDK_WINDOW_WIN32DATA (window)->event_mask & GDK_ENTER_NOTIFY_MASK)
2735     {
2736       GDK_NOTE (EVENTS, g_print ("synthesizing ENTER_NOTIFY event\n"));
2737       
2738       event = gdk_event_new ();
2739       event->crossing.type = GDK_ENTER_NOTIFY;
2740       event->crossing.window = window;
2741       gdk_drawable_ref (event->crossing.window);
2742       event->crossing.subwindow = NULL;
2743       event->crossing.time = xevent->time;
2744       event->crossing.x = LOWORD (xevent->lParam);
2745       event->crossing.y = HIWORD (xevent->lParam);
2746       event->crossing.x_root = (gfloat) xevent->pt.x;
2747       event->crossing.y_root = (gfloat) xevent->pt.y;
2748       event->crossing.mode = GDK_CROSSING_NORMAL;
2749       if (curWnd
2750           && IsChild (GDK_DRAWABLE_XID (curWnd), GDK_DRAWABLE_XID (window)))
2751         event->crossing.detail = GDK_NOTIFY_ANCESTOR;
2752       else if (curWnd
2753                && IsChild (GDK_DRAWABLE_XID (window), GDK_DRAWABLE_XID (curWnd)))
2754         event->crossing.detail = GDK_NOTIFY_INFERIOR;
2755       else
2756         event->crossing.detail = GDK_NOTIFY_NONLINEAR;
2757       
2758       event->crossing.focus = TRUE; /* ??? */
2759       event->crossing.state = 0; /* ??? */
2760       
2761       gdk_event_queue_append (event);
2762
2763       GDK_NOTE (EVENTS, print_event (event));
2764
2765       if (((GdkWindowPrivate *) window)->extension_events != 0
2766           && gdk_input_vtable.enter_event)
2767         gdk_input_vtable.enter_event (&event->crossing, window);
2768
2769     }
2770   
2771   if (curWnd)
2772     gdk_drawable_unref (curWnd);
2773   curWnd = window;
2774   gdk_drawable_ref (curWnd);
2775 #ifdef USE_TRACKMOUSEEVENT
2776   if (p_TrackMouseEvent != NULL)
2777     {
2778       TRACKMOUSEEVENT tme;
2779
2780       tme.cbSize = sizeof (TRACKMOUSEEVENT);
2781       tme.dwFlags = TME_LEAVE;
2782       tme.hwndTrack = GDK_DRAWABLE_XID (curWnd);
2783       tme.dwHoverTime = HOVER_DEFAULT;
2784       
2785       (*p_TrackMouseEvent) (&tme);
2786     }
2787 #endif
2788 }
2789
2790 static void
2791 translate_mouse_coords (GdkWindow *window1,
2792                         GdkWindow *window2,
2793                         MSG       *xevent)
2794 {
2795   POINT pt;
2796
2797   pt.x = LOWORD (xevent->lParam);
2798   pt.y = HIWORD (xevent->lParam);
2799   ClientToScreen (GDK_DRAWABLE_XID (window1), &pt);
2800   ScreenToClient (GDK_DRAWABLE_XID (window2), &pt);
2801   xevent->lParam = MAKELPARAM (pt.x, pt.y);
2802   GDK_NOTE (EVENTS, g_print ("...new coords are (%d,%d)\n", pt.x, pt.y));
2803 }
2804
2805 static gboolean
2806 propagate (GdkWindow  **window,
2807            MSG         *xevent,
2808            GdkWindow   *grab_window,
2809            gboolean     grab_owner_events,
2810            gint         grab_mask,
2811            gboolean   (*doesnt_want_it) (gint mask,
2812                                          MSG *xevent))
2813 {
2814   if (grab_window != NULL && !grab_owner_events)
2815     {
2816       /* Event source is grabbed with owner_events FALSE */
2817       GDK_NOTE (EVENTS, g_print ("...grabbed, owner_events FALSE, "));
2818       if ((*doesnt_want_it) (grab_mask, xevent))
2819         {
2820           GDK_NOTE (EVENTS, g_print ("...grabber doesn't want it\n"));
2821           return FALSE;
2822         }
2823       else
2824         {
2825           GDK_NOTE (EVENTS, g_print ("...sending to grabber %#x\n",
2826                                      GDK_DRAWABLE_XID (grab_window)));
2827           gdk_drawable_unref (*window);
2828           *window = grab_window;
2829           gdk_drawable_ref (*window);
2830           return TRUE;
2831         }
2832     }
2833   while (TRUE)
2834     {
2835      if ((*doesnt_want_it) (GDK_WINDOW_WIN32DATA (*window)->event_mask, xevent))
2836         {
2837           /* Owner doesn't want it, propagate to parent. */
2838           if (((GdkWindowPrivate *) *window)->parent == gdk_parent_root)
2839             {
2840               /* No parent; check if grabbed */
2841               if (grab_window != NULL)
2842                 {
2843                   /* Event source is grabbed with owner_events TRUE */
2844                   GDK_NOTE (EVENTS, g_print ("...undelivered, but grabbed\n"));
2845                   if ((*doesnt_want_it) (grab_mask, xevent))
2846                     {
2847                       /* Grabber doesn't want it either */
2848                       GDK_NOTE (EVENTS, g_print ("...grabber doesn't want it\n"));
2849                       return FALSE;
2850                     }
2851                   else
2852                     {
2853                       /* Grabbed! */
2854                       GDK_NOTE (EVENTS, g_print ("...sending to grabber %#x\n",
2855                                                  GDK_DRAWABLE_XID (grab_window)));
2856                       gdk_drawable_unref (*window);
2857                       *window = grab_window;
2858                       gdk_drawable_ref (*window);
2859                       return TRUE;
2860                     }
2861                 }
2862               else
2863                 {
2864                   GDK_NOTE (EVENTS, g_print ("...undelivered\n"));
2865                   return FALSE;
2866                 }
2867             }
2868           else
2869             {
2870               gdk_drawable_unref (*window);
2871               *window = ((GdkWindowPrivate *) *window)->parent;
2872               gdk_drawable_ref (*window);
2873               GDK_NOTE (EVENTS, g_print ("...propagating to %#x\n",
2874                                          GDK_DRAWABLE_XID (*window)));
2875               /* The only branch where we actually continue the loop */
2876             }
2877         }
2878       else
2879         return TRUE;
2880     }
2881 }
2882
2883 static gboolean
2884 doesnt_want_key (gint mask,
2885                  MSG *xevent)
2886 {
2887   return (((xevent->message == WM_KEYUP
2888             || xevent->message == WM_SYSKEYUP)
2889            && !(mask & GDK_KEY_RELEASE_MASK))
2890           ||
2891           ((xevent->message == WM_KEYDOWN
2892             || xevent->message == WM_SYSKEYDOWN)
2893            && !(mask & GDK_KEY_PRESS_MASK)));
2894 }
2895
2896 static gboolean
2897 doesnt_want_char (gint mask,
2898                   MSG *xevent)
2899 {
2900   return !(mask & (GDK_KEY_PRESS_MASK | GDK_KEY_RELEASE_MASK));
2901 }
2902
2903 static gboolean
2904 doesnt_want_button_press (gint mask,
2905                           MSG *xevent)
2906 {
2907   return !(mask & GDK_BUTTON_PRESS_MASK);
2908 }
2909
2910 static gboolean
2911 doesnt_want_button_release (gint mask,
2912                             MSG *xevent)
2913 {
2914   return !(mask & GDK_BUTTON_RELEASE_MASK);
2915 }
2916
2917 static gboolean
2918 doesnt_want_button_motion (gint mask,
2919                            MSG *xevent)
2920 {
2921   return !((mask & GDK_POINTER_MOTION_MASK)
2922            || ((xevent->wParam & (MK_LBUTTON|MK_MBUTTON|MK_RBUTTON))
2923                && (mask & GDK_BUTTON_MOTION_MASK))
2924            || ((xevent->wParam & MK_LBUTTON)
2925                && (mask & GDK_BUTTON1_MOTION_MASK))
2926            || ((xevent->wParam & MK_MBUTTON)
2927                && (mask & GDK_BUTTON2_MOTION_MASK))
2928            || ((xevent->wParam & MK_RBUTTON)
2929                && (mask & GDK_BUTTON3_MOTION_MASK)));
2930 }
2931
2932 static gboolean
2933 doesnt_want_scroll (gint mask,
2934                     MSG *xevent)
2935 {
2936 #if 0
2937   return !(mask & GDK_SCROLL_MASK);
2938 #else
2939   return !(mask & GDK_BUTTON_PRESS_MASK);
2940 #endif
2941 }
2942
2943 static gboolean
2944 gdk_event_translate (GdkEvent *event,
2945                      MSG      *xevent,
2946                      gboolean *ret_val_flagp,
2947                      gint     *ret_valp)
2948 {
2949   DWORD pidActWin;
2950   DWORD pidThis;
2951   DWORD dwStyle;
2952   PAINTSTRUCT paintstruct;
2953   HDC hdc;
2954   HDC bgdc;
2955   HGDIOBJ oldbitmap;
2956   HBRUSH hbr;
2957   COLORREF bg;
2958   RECT rect;
2959   POINT pt;
2960   MINMAXINFO *lpmmi;
2961   HWND hwnd;
2962   GdkWindow *window, *orig_window, *newwindow;
2963   GdkColormapPrivateWin32 *colormap_private;
2964   GdkEventMask mask;
2965   GdkPixmap *pixmap;
2966   GdkDrawablePrivate *pixmap_private;
2967   int button;
2968   int i, j, n, k;
2969   gchar buf[256];
2970   gchar *msgname;
2971   gboolean return_val;
2972   gboolean flag;
2973   
2974   return_val = FALSE;
2975   
2976   if (ret_val_flagp)
2977     *ret_val_flagp = FALSE;
2978
2979   window = gdk_window_lookup (xevent->hwnd);
2980   orig_window = window;
2981   
2982   event->any.window = window;
2983   event->any.send_event = FALSE;
2984
2985   if (window != NULL)
2986     gdk_drawable_ref (window);
2987   else
2988     {
2989       /* Handle WM_QUIT here ? */
2990       if (xevent->message == WM_QUIT)
2991         {
2992           GDK_NOTE (EVENTS, g_print ("WM_QUIT: %d\n", xevent->wParam));
2993           exit (xevent->wParam);
2994         }
2995       else if (xevent->message == WM_MOVE
2996                || xevent->message == WM_SIZE)
2997         {
2998           /* It's quite normal to get these messages before we have
2999            * had time to register the window in our lookup table, or
3000            * when the window is being destroyed and we already have
3001            * removed it. Repost the same message to our queue so that
3002            * we will get it later when we are prepared.
3003            */
3004           GDK_NOTE(MISC, g_print("gdk_event_translate: %#x %s posted.\n",
3005                                  xevent->hwnd, 
3006                                  xevent->message == WM_MOVE ?
3007                                  "WM_MOVE" : "WM_SIZE"));
3008         
3009           PostMessage (xevent->hwnd, xevent->message,
3010                        xevent->wParam, xevent->lParam);
3011         }
3012       return FALSE;
3013     }
3014   
3015   if (!GDK_DRAWABLE_DESTROYED (window))
3016     {
3017       /* Check for filters for this window */
3018       GdkFilterReturn result;
3019
3020       result = gdk_event_apply_filters
3021         (xevent, event, ((GdkWindowPrivate *) window)->filters);
3022       
3023       if (result != GDK_FILTER_CONTINUE)
3024         {
3025           return_val =  (result == GDK_FILTER_TRANSLATE) ? TRUE : FALSE;
3026           goto done;
3027         }
3028     }
3029
3030   if (xevent->message == gdk_selection_notify_msg)
3031     {
3032       GDK_NOTE (EVENTS, g_print ("gdk_selection_notify_msg: %#x\n",
3033                                  xevent->hwnd));
3034
3035       event->selection.type = GDK_SELECTION_NOTIFY;
3036       event->selection.window = window;
3037       event->selection.selection = xevent->wParam;
3038       event->selection.target = xevent->lParam;
3039       event->selection.property = gdk_selection_property;
3040       event->selection.time = xevent->time;
3041
3042       return_val = !GDK_DRAWABLE_DESTROYED (window);
3043
3044       /* Will pass through switch below without match */
3045     }
3046   else if (xevent->message == gdk_selection_request_msg)
3047     {
3048       GDK_NOTE (EVENTS, g_print ("gdk_selection_request_msg: %#x\n",
3049                                  xevent->hwnd));
3050
3051       event->selection.type = GDK_SELECTION_REQUEST;
3052       event->selection.window = window;
3053       event->selection.selection = gdk_clipboard_atom;
3054       event->selection.target = GDK_TARGET_STRING;
3055       event->selection.property = gdk_selection_property;
3056       event->selection.requestor = (guint32) xevent->hwnd;
3057       event->selection.time = xevent->time;
3058
3059       return_val = !GDK_DRAWABLE_DESTROYED (window);
3060
3061       /* Again, will pass through switch below without match */
3062     }
3063   else if (xevent->message == gdk_selection_clear_msg)
3064     {
3065       GDK_NOTE (EVENTS, g_print ("gdk_selection_clear_msg: %#x\n",
3066                                  xevent->hwnd));
3067
3068       event->selection.type = GDK_SELECTION_CLEAR;
3069       event->selection.window = window;
3070       event->selection.selection = xevent->wParam;
3071       event->selection.time = xevent->time;
3072
3073       return_val = !GDK_DRAWABLE_DESTROYED (window);
3074
3075       /* Once again, we will pass through switch below without match */
3076     }
3077   else
3078     {
3079       GList *tmp_list;
3080       GdkFilterReturn result = GDK_FILTER_CONTINUE;
3081
3082       tmp_list = client_filters;
3083       while (tmp_list)
3084         {
3085           GdkClientFilter *filter = tmp_list->data;
3086           if (filter->type == xevent->message)
3087             {
3088               GDK_NOTE (EVENTS, g_print ("client filter matched\n"));
3089               event->any.window = window;
3090               result = (*filter->function) (xevent, event, filter->data);
3091               switch (result)
3092                 {
3093                 case GDK_FILTER_REMOVE:
3094                   return_val = FALSE;
3095                   break;
3096
3097                 case GDK_FILTER_TRANSLATE:
3098                   return_val = TRUE;
3099                   break;
3100
3101                 case GDK_FILTER_CONTINUE:
3102                   return_val = TRUE;
3103                   event->client.type = GDK_CLIENT_EVENT;
3104                   event->client.window = window;
3105                   event->client.message_type = xevent->message;
3106                   event->client.data_format = 0;
3107                   event->client.data.l[0] = xevent->wParam;
3108                   event->client.data.l[1] = xevent->lParam;
3109                   break;
3110                 }
3111               goto done;
3112             }
3113           tmp_list = tmp_list->next;
3114         }
3115     }
3116
3117   switch (xevent->message)
3118     {
3119     case WM_INPUTLANGCHANGE:
3120       GDK_NOTE (EVENTS,
3121                 g_print ("WM_INPUTLANGCHANGE: %#x charset %d locale %x\n",
3122                          xevent->hwnd, xevent->wParam, xevent->lParam));
3123       GDK_WINDOW_WIN32DATA (window)->input_locale = (HKL) xevent->lParam;
3124       TranslateCharsetInfo ((DWORD FAR *) xevent->wParam,
3125                             &GDK_WINDOW_WIN32DATA (window)->charset_info,
3126                             TCI_SRCCHARSET);
3127       break;
3128
3129     case WM_SYSKEYUP:
3130     case WM_SYSKEYDOWN:
3131       GDK_NOTE (EVENTS,
3132                 g_print ("WM_SYSKEY%s: %#x  key: %s  %#x %#.08x\n",
3133                          (xevent->message == WM_SYSKEYUP ? "UP" : "DOWN"),
3134                          xevent->hwnd,
3135                          (GetKeyNameText (xevent->lParam, buf,
3136                                           sizeof (buf)) > 0 ?
3137                           buf : ""),
3138                          xevent->wParam,
3139                          xevent->lParam));
3140
3141       /* Let the system handle Alt-Tab and Alt-Enter */
3142       if (xevent->wParam == VK_TAB
3143           || xevent->wParam == VK_RETURN
3144           || xevent->wParam == VK_F4)
3145         break;
3146       /* If posted without us having keyboard focus, ignore */
3147       if (!(xevent->lParam & 0x20000000))
3148         break;
3149 #if 0
3150       /* don't generate events for just the Alt key */
3151       if (xevent->wParam == VK_MENU)
3152         break;
3153 #endif
3154       /* Jump to code in common with WM_KEYUP and WM_KEYDOWN */
3155       goto keyup_or_down;
3156
3157     case WM_KEYUP:
3158     case WM_KEYDOWN:
3159       GDK_NOTE (EVENTS, 
3160                 g_print ("WM_KEY%s: %#x  key: %s  %#x %#.08x\n",
3161                          (xevent->message == WM_KEYUP ? "UP" : "DOWN"),
3162                          xevent->hwnd,
3163                          (GetKeyNameText (xevent->lParam, buf,
3164                                           sizeof (buf)) > 0 ?
3165                           buf : ""),
3166                          xevent->wParam,
3167                          xevent->lParam));
3168
3169       ignore_WM_CHAR = TRUE;
3170
3171     keyup_or_down:
3172       if (!propagate (&window, xevent,
3173                       k_grab_window, k_grab_owner_events, GDK_ALL_EVENTS_MASK,
3174                       doesnt_want_key))
3175           break;
3176       event->key.window = window;
3177       switch (xevent->wParam)
3178         {
3179         case VK_LBUTTON:
3180           event->key.keyval = GDK_Pointer_Button1; break;
3181         case VK_RBUTTON:
3182           event->key.keyval = GDK_Pointer_Button3; break;
3183         case VK_MBUTTON:
3184           event->key.keyval = GDK_Pointer_Button2; break;
3185         case VK_CANCEL:
3186           event->key.keyval = GDK_Cancel; break;
3187         case VK_BACK:
3188           event->key.keyval = GDK_BackSpace; break;
3189         case VK_TAB:
3190           event->key.keyval = (GetKeyState(VK_SHIFT) < 0 ? 
3191             GDK_ISO_Left_Tab : GDK_Tab);
3192           break;
3193         case VK_CLEAR:
3194           event->key.keyval = GDK_Clear; break;
3195         case VK_RETURN:
3196           event->key.keyval = GDK_Return; break;
3197         case VK_SHIFT:
3198           /* Don't let Shift auto-repeat */
3199           if (xevent->message == WM_KEYDOWN
3200               && (xevent->lParam & 0x40000000))
3201             ignore_WM_CHAR = FALSE;
3202           else
3203             event->key.keyval = GDK_Shift_L;
3204           break;
3205         case VK_CONTROL:
3206           /* And not Control either */
3207           if (xevent->message == WM_KEYDOWN
3208               && (xevent->lParam & 0x40000000))
3209             ignore_WM_CHAR = FALSE;
3210           else if (xevent->lParam & 0x01000000)
3211             event->key.keyval = GDK_Control_R;
3212           else
3213             event->key.keyval = GDK_Control_L;
3214           break;
3215         case VK_MENU:
3216           /* And not Alt */
3217           if (xevent->message == WM_KEYDOWN
3218               && (xevent->lParam & 0x40000000))
3219             ignore_WM_CHAR = FALSE;
3220           else if (xevent->lParam & 0x01000000)
3221             {
3222               /* AltGr key comes in as Control+Right Alt */
3223               if (GetKeyState (VK_CONTROL) < 0)
3224                 {
3225                   ignore_WM_CHAR = FALSE;
3226                   is_AltGr_key = TRUE;
3227                 }
3228               event->key.keyval = GDK_Alt_R;
3229             }
3230           else
3231             event->key.keyval = GDK_Alt_L;
3232           break;
3233         case VK_PAUSE:
3234           event->key.keyval = GDK_Pause; break;
3235         case VK_CAPITAL:
3236           event->key.keyval = GDK_Caps_Lock; break;
3237         case VK_ESCAPE:
3238           event->key.keyval = GDK_Escape; break;
3239         case VK_PRIOR:
3240           event->key.keyval = GDK_Prior; break;
3241         case VK_NEXT:
3242           event->key.keyval = GDK_Next; break;
3243         case VK_END:
3244           event->key.keyval = GDK_End; break;
3245         case VK_HOME:
3246           event->key.keyval = GDK_Home; break;
3247         case VK_LEFT:
3248           event->key.keyval = GDK_Left; break;
3249         case VK_UP:
3250           event->key.keyval = GDK_Up; break;
3251         case VK_RIGHT:
3252           event->key.keyval = GDK_Right; break;
3253         case VK_DOWN:
3254           event->key.keyval = GDK_Down; break;
3255         case VK_SELECT:
3256           event->key.keyval = GDK_Select; break;
3257         case VK_PRINT:
3258           event->key.keyval = GDK_Print; break;
3259         case VK_EXECUTE:
3260           event->key.keyval = GDK_Execute; break;
3261         case VK_INSERT:
3262           event->key.keyval = GDK_Insert; break;
3263         case VK_DELETE:
3264           event->key.keyval = GDK_Delete; break;
3265         case VK_HELP:
3266           event->key.keyval = GDK_Help; break;
3267         case VK_NUMPAD0:
3268         case VK_NUMPAD1:
3269         case VK_NUMPAD2:
3270         case VK_NUMPAD3:
3271         case VK_NUMPAD4:
3272         case VK_NUMPAD5:
3273         case VK_NUMPAD6:
3274         case VK_NUMPAD7:
3275         case VK_NUMPAD8:
3276         case VK_NUMPAD9:
3277           /* Apparently applications work better if we just pass numpad digits
3278            * on as real digits? So wait for the WM_CHAR instead.
3279            */
3280           ignore_WM_CHAR = FALSE;
3281           break;
3282         case VK_MULTIPLY:
3283           event->key.keyval = GDK_KP_Multiply; break;
3284         case VK_ADD:
3285           /* Pass it on as an ASCII plus in WM_CHAR. */
3286           ignore_WM_CHAR = FALSE;
3287           break;
3288         case VK_SEPARATOR:
3289           event->key.keyval = GDK_KP_Separator; break;
3290         case VK_SUBTRACT:
3291           /* Pass it on as an ASCII minus in WM_CHAR. */
3292           ignore_WM_CHAR = FALSE;
3293           break;
3294         case VK_DECIMAL:
3295           /* The keypad decimal key should also be passed on as the decimal
3296            * sign ('.' or ',' depending on the Windows locale settings,
3297            * apparently). So wait for the WM_CHAR here, also.
3298            */
3299           ignore_WM_CHAR = FALSE;
3300           break;
3301         case VK_DIVIDE:
3302           event->key.keyval = GDK_KP_Divide; break;
3303         case VK_F1:
3304           event->key.keyval = GDK_F1; break;
3305         case VK_F2:
3306           event->key.keyval = GDK_F2; break;
3307         case VK_F3:
3308           event->key.keyval = GDK_F3; break;
3309         case VK_F4:
3310           event->key.keyval = GDK_F4; break;
3311         case VK_F5:
3312           event->key.keyval = GDK_F5; break;
3313         case VK_F6:
3314           event->key.keyval = GDK_F6; break;
3315         case VK_F7:
3316           event->key.keyval = GDK_F7; break;
3317         case VK_F8:
3318           event->key.keyval = GDK_F8; break;
3319         case VK_F9:
3320           event->key.keyval = GDK_F9; break;
3321         case VK_F10:
3322           event->key.keyval = GDK_F10; break;
3323         case VK_F11:
3324           event->key.keyval = GDK_F11; break;
3325         case VK_F12:
3326           event->key.keyval = GDK_F12; break;
3327         case VK_F13:
3328           event->key.keyval = GDK_F13; break;
3329         case VK_F14:
3330           event->key.keyval = GDK_F14; break;
3331         case VK_F15:
3332           event->key.keyval = GDK_F15; break;
3333         case VK_F16:
3334           event->key.keyval = GDK_F16; break;
3335         case '0':
3336         case '1':
3337         case '2':
3338         case '3':
3339         case '4':
3340         case '5':
3341         case '6':
3342         case '7':
3343         case '8':
3344         case '9':
3345           if (!is_AltGr_key && (GetKeyState (VK_CONTROL) < 0
3346                                 || GetKeyState (VK_MENU) < 0))
3347             /* Control- or Alt-digits won't come in as a WM_CHAR,
3348              * but beware of AltGr-digits, which are used for instance
3349              * on Finnish keyboards.
3350              */
3351             event->key.keyval = GDK_0 + (xevent->wParam - '0');
3352           else
3353             ignore_WM_CHAR = FALSE;
3354           break;
3355         case VK_OEM_PLUS:       /* On my Win98, the '+' key comes in
3356                                  * as VK_OEM_PLUS
3357                                  */
3358           if (!is_AltGr_key && (GetKeyState (VK_CONTROL) < 0
3359                                 || GetKeyState (VK_MENU) < 0))
3360             /* Control- or Alt-plus won't come in as WM_CHAR,
3361              * but beware of AltGr-plus which is backslash on
3362              * Finnish keyboards
3363              */
3364             event->key.keyval = '+';
3365           else
3366             ignore_WM_CHAR = FALSE;
3367           break;
3368         default:
3369           if (xevent->message == WM_SYSKEYDOWN || xevent->message == WM_SYSKEYUP)
3370             event->key.keyval = xevent->wParam;
3371           else
3372             ignore_WM_CHAR = FALSE;
3373           break;
3374         }
3375
3376       if (!ignore_WM_CHAR)
3377         break;
3378
3379       is_AltGr_key = FALSE;
3380       event->key.type = ((xevent->message == WM_KEYDOWN
3381                           || xevent->message == WM_SYSKEYDOWN) ?
3382                          GDK_KEY_PRESS : GDK_KEY_RELEASE);
3383       event->key.time = xevent->time;
3384       event->key.state = 0;
3385       if (GetKeyState (VK_SHIFT) < 0)
3386         event->key.state |= GDK_SHIFT_MASK;
3387       if (GetKeyState (VK_CAPITAL) & 0x1)
3388         event->key.state |= GDK_LOCK_MASK;
3389       if (GetKeyState (VK_CONTROL) < 0)
3390         event->key.state |= GDK_CONTROL_MASK;
3391       if (xevent->wParam != VK_MENU && GetKeyState (VK_MENU) < 0)
3392         event->key.state |= GDK_MOD1_MASK;
3393       event->key.string = NULL;
3394       event->key.length = 0;
3395       return_val = !GDK_DRAWABLE_DESTROYED (window);
3396       break;
3397
3398     case WM_IME_COMPOSITION:
3399       if (!use_IME_COMPOSITION)
3400         break;
3401       GDK_NOTE (EVENTS, g_print ("WM_IME_COMPOSITION: %#x  %#x\n",
3402                                  xevent->hwnd, xevent->lParam));
3403       if (xevent->lParam & GCS_RESULTSTR)
3404         goto wm_char;
3405       break;
3406
3407     case WM_IME_CHAR:
3408       GDK_NOTE (EVENTS,
3409                 g_print ("WM_IME_CHAR: %#x  bytes: %#.04x\n",
3410                          xevent->hwnd, xevent->wParam));
3411       goto wm_char;
3412       
3413     case WM_CHAR:
3414       GDK_NOTE (EVENTS, 
3415                 g_print ("WM_CHAR: %#x  char: %#x %#.08x  %s\n",
3416                          xevent->hwnd, xevent->wParam, xevent->lParam,
3417                          (ignore_WM_CHAR ? "ignored" : "")));
3418
3419       if (ignore_WM_CHAR)
3420         {
3421           ignore_WM_CHAR = FALSE;
3422           break;
3423         }
3424
3425     wm_char:
3426       if (!propagate (&window, xevent,
3427                       k_grab_window, k_grab_owner_events, GDK_ALL_EVENTS_MASK,
3428                       doesnt_want_char))
3429           break;
3430       event->key.window = window;
3431       return_val = !GDK_DRAWABLE_DESTROYED (window);
3432       if (return_val && (event->key.window == k_grab_window
3433                          || (GDK_WINDOW_WIN32DATA (window)->event_mask & GDK_KEY_RELEASE_MASK)))
3434         {
3435           if (window == k_grab_window
3436               || (GDK_WINDOW_WIN32DATA (window)->event_mask & GDK_KEY_PRESS_MASK))
3437             {
3438               /* Append a GDK_KEY_PRESS event to the pushback list
3439                * (from which it will be fetched before the release
3440                * event).
3441                */
3442               GdkEvent *event2 = gdk_event_new ();
3443               build_keypress_event (GDK_WINDOW_WIN32DATA (window), event2, xevent);
3444               event2->key.window = window;
3445               gdk_drawable_ref (window);
3446               gdk_event_queue_append (event2);
3447               GDK_NOTE (EVENTS, print_event (event2));
3448             }
3449           /* Return the key release event.  */
3450           build_keyrelease_event (GDK_WINDOW_WIN32DATA (window), event, xevent);
3451         }
3452       else if (return_val
3453                && (GDK_WINDOW_WIN32DATA (window)->event_mask & GDK_KEY_PRESS_MASK))
3454         {
3455           /* Return just the key press event. */
3456           build_keypress_event (GDK_WINDOW_WIN32DATA (window), event, xevent);
3457         }
3458       else
3459         return_val = FALSE;
3460
3461 #if 0 /* Don't reset is_AltGr_key here. Othewise we can't type several
3462        * AltGr-accessed chars while keeping the AltGr pressed down
3463        * all the time.
3464        */
3465       is_AltGr_key = FALSE;
3466 #endif
3467       break;
3468
3469     case WM_LBUTTONDOWN:
3470       button = 1;
3471       goto buttondown0;
3472     case WM_MBUTTONDOWN:
3473       button = 2;
3474       goto buttondown0;
3475     case WM_RBUTTONDOWN:
3476       button = 3;
3477
3478     buttondown0:
3479       GDK_NOTE (EVENTS, 
3480                 g_print ("WM_%cBUTTONDOWN: %#x  (%d,%d)\n",
3481                          " LMR"[button],
3482                          xevent->hwnd,
3483                          LOWORD (xevent->lParam), HIWORD (xevent->lParam)));
3484
3485       if (((GdkWindowPrivate *) window)->extension_events != 0
3486           && gdk_input_ignore_core)
3487         {
3488           GDK_NOTE (EVENTS, g_print ("...ignored\n"));
3489           break;
3490         }
3491
3492       if (window != curWnd)
3493         synthesize_crossing_events (window, xevent);
3494
3495       event->button.type = GDK_BUTTON_PRESS;
3496       if (!propagate (&window, xevent,
3497                       p_grab_window, p_grab_owner_events, p_grab_mask,
3498                       doesnt_want_button_press))
3499           break;
3500       event->button.window = window;
3501       /* Emulate X11's automatic active grab */
3502       if (!p_grab_window)
3503         {
3504           /* No explicit active grab, let's start one automatically */
3505           gint owner_events =
3506             GDK_WINDOW_WIN32DATA (window)->event_mask
3507             & (GDK_BUTTON_PRESS_MASK|GDK_BUTTON_RELEASE_MASK);
3508           
3509           GDK_NOTE (EVENTS, g_print ("...automatic grab started\n"));
3510           gdk_pointer_grab (window,
3511                             owner_events,
3512                             GDK_WINDOW_WIN32DATA (window)->event_mask,
3513                             NULL, NULL, 0);
3514           p_grab_automatic = TRUE;
3515         }
3516
3517       event->button.time = xevent->time;
3518       if (window != orig_window)
3519         translate_mouse_coords (orig_window, window, xevent);
3520       event->button.x = curX = (gint16) LOWORD (xevent->lParam);
3521       event->button.y = curY = (gint16) HIWORD (xevent->lParam);
3522       event->button.x_root = xevent->pt.x;
3523       event->button.y_root = xevent->pt.y;
3524       event->button.pressure = 0.5;
3525       event->button.xtilt = 0;
3526       event->button.ytilt = 0;
3527       event->button.state = build_pointer_event_state (xevent);
3528       event->button.button = button;
3529       event->button.source = GDK_SOURCE_MOUSE;
3530       event->button.deviceid = GDK_CORE_POINTER;
3531
3532       gdk_event_button_generate (event);
3533       
3534       return_val = !GDK_DRAWABLE_DESTROYED (window);
3535       break;
3536
3537     case WM_LBUTTONUP:
3538       button = 1;
3539       goto buttonup0;
3540     case WM_MBUTTONUP:
3541       button = 2;
3542       goto buttonup0;
3543     case WM_RBUTTONUP:
3544       button = 3;
3545
3546     buttonup0:
3547       GDK_NOTE (EVENTS, 
3548                 g_print ("WM_%cBUTTONUP: %#x  (%d,%d)\n",
3549                          " LMR"[button],
3550                          xevent->hwnd,
3551                          LOWORD (xevent->lParam), HIWORD (xevent->lParam)));
3552
3553       if (((GdkWindowPrivate *) window)->extension_events != 0
3554           && gdk_input_ignore_core)
3555         {
3556           GDK_NOTE (EVENTS, g_print ("...ignored\n"));
3557           break;
3558         }
3559
3560       if (window != curWnd)
3561         synthesize_crossing_events (window, xevent);
3562
3563       event->button.type = GDK_BUTTON_RELEASE;
3564       if (!propagate (&window, xevent,
3565                       p_grab_window, p_grab_owner_events, p_grab_mask,
3566                       doesnt_want_button_release))
3567           goto maybe_ungrab;
3568       event->button.window = window;
3569       event->button.time = xevent->time;
3570       if (window != orig_window)
3571         translate_mouse_coords (orig_window, window, xevent);
3572       event->button.x = (gint16) LOWORD (xevent->lParam);
3573       event->button.y = (gint16) HIWORD (xevent->lParam);
3574       event->button.x_root = xevent->pt.x;
3575       event->button.y_root = xevent->pt.y;
3576       event->button.pressure = 0.5;
3577       event->button.xtilt = 0;
3578       event->button.ytilt = 0;
3579       event->button.state = build_pointer_event_state (xevent);
3580       event->button.button = button;
3581       event->button.source = GDK_SOURCE_MOUSE;
3582       event->button.deviceid = GDK_CORE_POINTER;
3583
3584       return_val = !GDK_DRAWABLE_DESTROYED (window);
3585
3586     maybe_ungrab:
3587       if (p_grab_window != NULL
3588           && p_grab_automatic
3589           && (event->button.state & (GDK_BUTTON1_MASK | GDK_BUTTON2_MASK | GDK_BUTTON3_MASK)) == 0)
3590         gdk_pointer_ungrab (0);
3591       break;
3592
3593     case WM_MOUSEMOVE:
3594       GDK_NOTE (EVENTS,
3595                 g_print ("WM_MOUSEMOVE: %#x  %#x (%d,%d)\n",
3596                          xevent->hwnd, xevent->wParam,
3597                          LOWORD (xevent->lParam), HIWORD (xevent->lParam)));
3598
3599       /* If we haven't moved, don't create any event.
3600        * Windows sends WM_MOUSEMOVE messages after button presses
3601        * even if the mouse doesn't move. This disturbs gtk.
3602        */
3603       if (window == curWnd
3604           && LOWORD (xevent->lParam) == curX
3605           && HIWORD (xevent->lParam) == curY)
3606         break;
3607
3608       /* HB: only process mouse move messages if we own the active window. */
3609       GetWindowThreadProcessId(GetActiveWindow(), &pidActWin);
3610       GetWindowThreadProcessId(xevent->hwnd, &pidThis);
3611       if (pidActWin != pidThis)
3612         break;
3613
3614       if (window != curWnd)
3615         synthesize_crossing_events (window, xevent);
3616
3617       if (((GdkWindowPrivate *) window)->extension_events != 0
3618           && gdk_input_ignore_core)
3619         {
3620           GDK_NOTE (EVENTS, g_print ("...ignored\n"));
3621           break;
3622         }
3623
3624       event->motion.type = GDK_MOTION_NOTIFY;
3625       if (!propagate (&window, xevent,
3626                       p_grab_window, p_grab_owner_events, p_grab_mask,
3627                       doesnt_want_button_motion))
3628           break;
3629       event->motion.window = window;
3630       event->motion.time = xevent->time;
3631       if (window != orig_window)
3632         translate_mouse_coords (orig_window, window, xevent);
3633       event->motion.x = curX = (gint16) LOWORD (xevent->lParam);
3634       event->motion.y = curY = (gint16) HIWORD (xevent->lParam);
3635       event->motion.x_root = xevent->pt.x;
3636       event->motion.y_root = xevent->pt.y;
3637       curXroot = event->motion.x_root;
3638       curYroot = event->motion.y_root;
3639       event->motion.pressure = 0.5;
3640       event->motion.xtilt = 0;
3641       event->motion.ytilt = 0;
3642       event->motion.state = build_pointer_event_state (xevent);
3643       event->motion.is_hint = FALSE;
3644       event->motion.source = GDK_SOURCE_MOUSE;
3645       event->motion.deviceid = GDK_CORE_POINTER;
3646
3647       return_val = !GDK_DRAWABLE_DESTROYED (window);
3648       break;
3649
3650     case WM_NCMOUSEMOVE:
3651       GDK_NOTE (EVENTS,
3652                 g_print ("WM_NCMOUSEMOVE: %#x  x,y: %d %d\n",
3653                          xevent->hwnd,
3654                          LOWORD (xevent->lParam), HIWORD (xevent->lParam)));
3655       if (p_TrackMouseEvent == NULL
3656           && curWnd != NULL
3657           && (GDK_WINDOW_WIN32DATA (curWnd)->event_mask & GDK_LEAVE_NOTIFY_MASK))
3658         {
3659           GDK_NOTE (EVENTS, g_print ("...synthesizing LEAVE_NOTIFY event\n"));
3660
3661           event->crossing.type = GDK_LEAVE_NOTIFY;
3662           event->crossing.window = curWnd;
3663           event->crossing.subwindow = NULL;
3664           event->crossing.time = xevent->time;
3665           event->crossing.x = curX;
3666           event->crossing.y = curY;
3667           event->crossing.x_root = curXroot;
3668           event->crossing.y_root = curYroot;
3669           event->crossing.mode = GDK_CROSSING_NORMAL;
3670           event->crossing.detail = GDK_NOTIFY_NONLINEAR;
3671
3672           event->crossing.focus = TRUE; /* ??? */
3673           event->crossing.state = 0; /* ??? */
3674           return_val = TRUE;
3675         }
3676
3677       if (curWnd)
3678         {
3679           gdk_drawable_unref (curWnd);
3680           curWnd = NULL;
3681         }
3682
3683       break;
3684
3685 #ifndef WM_MOUSEWHEEL
3686 #define WM_MOUSEWHEEL 0x20a
3687 #endif
3688
3689     case WM_MOUSEWHEEL:
3690       GDK_NOTE (EVENTS, g_print ("WM_MOUSEWHEEL: %#x\n", xevent->hwnd));
3691
3692       if (((GdkWindowPrivate *) window)->extension_events != 0
3693           && gdk_input_ignore_core)
3694         {
3695           GDK_NOTE (EVENTS, g_print ("...ignored\n"));
3696           break;
3697         }
3698
3699       event->scroll.type = GDK_SCROLL;
3700
3701       /* WM_MOUSEWHEEL seems to be delivered to top-level windows
3702        * only, for some reason. Work around that. Also, the position
3703        * is in screen coordinates, not client coordinates as with the
3704        * button messages. I love the consistency of Windows.
3705        */
3706       pt.x = LOWORD (xevent->lParam);
3707       pt.y = HIWORD (xevent->lParam);
3708       if ((hwnd = WindowFromPoint (pt)) == NULL)
3709         break;
3710       xevent->hwnd = hwnd;
3711       if ((newwindow = gdk_window_lookup (xevent->hwnd)) == NULL)
3712         break;
3713       if (newwindow != window)
3714         {
3715           gdk_drawable_unref (window);
3716           window = newwindow;
3717           gdk_drawable_ref (window);
3718         }
3719       ScreenToClient (xevent->hwnd, &pt);
3720       if (!propagate (&window, xevent,
3721                       p_grab_window, p_grab_owner_events, p_grab_mask,
3722                       doesnt_want_scroll))
3723         break;
3724       event->button.window = window;
3725       event->scroll.direction = (((short) HIWORD (xevent->wParam)) > 0) ?
3726         GDK_SCROLL_UP : GDK_SCROLL_DOWN;
3727       event->scroll.window = window;
3728       event->scroll.time = xevent->time;
3729       event->scroll.x = (gint16) pt.x;
3730       event->scroll.y = (gint16) pt.y;
3731       event->scroll.x_root = (gint16) LOWORD (xevent->lParam);
3732       event->scroll.y_root = (gint16) LOWORD (xevent->lParam);
3733       event->scroll.pressure = 0.5;
3734       event->scroll.xtilt = 0;
3735       event->scroll.ytilt = 0;
3736       event->scroll.state = build_pointer_event_state (xevent);
3737       event->scroll.source = GDK_SOURCE_MOUSE;
3738       event->scroll.deviceid = GDK_CORE_POINTER;
3739       return_val = !GDK_DRAWABLE_DESTROYED (window);
3740       
3741       break;
3742
3743 #ifdef USE_TRACKMOUSEEVENT
3744     case WM_MOUSELEAVE:
3745       GDK_NOTE (EVENTS, g_print ("WM_MOUSELEAVE: %#x\n", xevent->hwnd));
3746
3747       if (!(GDK_WINDOW_WIN32DATA (window)->event_mask & GDK_LEAVE_NOTIFY_MASK))
3748         break;
3749
3750       event->crossing.type = GDK_LEAVE_NOTIFY;
3751       event->crossing.window = window;
3752       event->crossing.subwindow = NULL;
3753       event->crossing.time = xevent->time;
3754       event->crossing.x = curX;
3755       event->crossing.y = curY;
3756       event->crossing.x_root = curXroot;
3757       event->crossing.y_root = curYroot;
3758       event->crossing.mode = GDK_CROSSING_NORMAL;
3759       if (curWnd
3760           && IsChild (GDK_DRAWABLE_XID (curWnd), GDK_DRAWABLE_XID (window)))
3761         event->crossing.detail = GDK_NOTIFY_INFERIOR;
3762       else if (curWnd
3763                && IsChild (GDK_DRAWABLE_XID (window), GDK_DRAWABLE_XID (curWnd)))
3764         event->crossing.detail = GDK_NOTIFY_ANCESTOR;
3765       else
3766         event->crossing.detail = GDK_NOTIFY_NONLINEAR;
3767
3768       event->crossing.focus = TRUE; /* ??? */
3769       event->crossing.state = 0; /* ??? */
3770
3771       if (curWnd)
3772         {
3773           gdk_drawable_unref (curWnd);
3774           curWnd = NULL;
3775         }
3776
3777       return_val = !GDK_DRAWABLE_DESTROYED (window);
3778       break;
3779 #endif
3780         
3781     case WM_SETFOCUS:
3782     case WM_KILLFOCUS:
3783       GDK_NOTE (EVENTS, g_print ("WM_%sFOCUS: %#x\n",
3784                                  (xevent->message == WM_SETFOCUS ?
3785                                   "SET" : "KILL"),
3786                                  xevent->hwnd));
3787       
3788       if (!(GDK_WINDOW_WIN32DATA (window)->event_mask & GDK_FOCUS_CHANGE_MASK))
3789         break;
3790
3791       event->focus_change.type = GDK_FOCUS_CHANGE;
3792       event->focus_change.window = window;
3793       event->focus_change.in = (xevent->message == WM_SETFOCUS);
3794       return_val = !GDK_DRAWABLE_DESTROYED (window);
3795       break;
3796
3797     case WM_ERASEBKGND:
3798       GDK_NOTE (EVENTS, g_print ("WM_ERASEBKGND: %#x  dc %#x\n",
3799                                  xevent->hwnd, xevent->wParam));
3800       
3801       if (GDK_DRAWABLE_DESTROYED (window))
3802         break;
3803
3804       colormap_private = (GdkColormapPrivateWin32 *) ((GdkWindowPrivate *) window)->drawable.colormap;
3805       hdc = (HDC) xevent->wParam;
3806       if (colormap_private
3807           && colormap_private->xcolormap->rc_palette)
3808         {
3809           int k;
3810
3811           if (SelectPalette (hdc,  colormap_private->xcolormap->palette,
3812                              FALSE) == NULL)
3813             WIN32_GDI_FAILED ("SelectPalette");
3814           if ((k = RealizePalette (hdc)) == GDI_ERROR)
3815             WIN32_GDI_FAILED ("RealizePalette");
3816 #if 0
3817           g_print ("WM_ERASEBKGND: selected %#x, realized %d colors\n",
3818                    colormap_private->xcolormap->palette, k);
3819 #endif
3820         }
3821       *ret_val_flagp = TRUE;
3822       *ret_valp = 1;
3823
3824       if (GDK_WINDOW_WIN32DATA (window)->bg_type == GDK_WIN32_BG_TRANSPARENT)
3825         break;
3826
3827       if (GDK_WINDOW_WIN32DATA (window)->bg_type == GDK_WIN32_BG_PARENT_RELATIVE)
3828         {
3829           /* If this window should have the same background as the
3830            * parent, fetch the parent. (And if the same goes for
3831            * the parent, fetch the grandparent, etc.)
3832            */
3833           while (window
3834                  && GDK_WINDOW_WIN32DATA (window)->bg_type == GDK_WIN32_BG_PARENT_RELATIVE)
3835             {
3836               gdk_drawable_unref (window);
3837               window = ((GdkWindowPrivate *) window)->parent;
3838               gdk_drawable_ref (window);
3839             }
3840         }
3841
3842       if (GDK_WINDOW_WIN32DATA (window)->bg_type == GDK_WIN32_BG_PIXEL)
3843         {
3844           bg = gdk_colormap_color (colormap_private,
3845                                    GDK_WINDOW_WIN32DATA (window)->bg_pixel);
3846
3847           GetClipBox (hdc, &rect);
3848           GDK_NOTE (EVENTS,
3849                     g_print ("...%dx%d@+%d+%d BG_PIXEL %.06x\n",
3850                              rect.right - rect.left,
3851                              rect.bottom - rect.top,
3852                              rect.left, rect.top,
3853                              bg));
3854           hbr = CreateSolidBrush (bg);
3855 #if 0
3856           g_print ("...CreateSolidBrush (%.08x) = %.08x\n", bg, hbr);
3857 #endif
3858           if (!FillRect (hdc, &rect, hbr))
3859             WIN32_GDI_FAILED ("FillRect");
3860           DeleteObject (hbr);
3861         }
3862       else if (GDK_WINDOW_WIN32DATA (window)->bg_type == GDK_WIN32_BG_PIXMAP)
3863         {
3864           pixmap = GDK_WINDOW_WIN32DATA (window)->bg_pixmap;
3865           pixmap_private = (GdkDrawablePrivate*) pixmap;
3866           GetClipBox (hdc, &rect);
3867
3868           if (pixmap_private->width <= 8
3869               && pixmap_private->height <= 8)
3870             {
3871               GDK_NOTE (EVENTS, g_print ("...small pixmap, using brush\n"));
3872               hbr = CreatePatternBrush (GDK_DRAWABLE_XID (pixmap));
3873               if (!FillRect (hdc, &rect, hbr))
3874                 WIN32_GDI_FAILED ("FillRect");
3875               DeleteObject (hbr);
3876             }
3877           else
3878             {
3879               GDK_NOTE (EVENTS,
3880                         g_print ("...blitting pixmap %#x (%dx%d) "
3881                                  "all over the place,\n"
3882                                  "...clip box = %dx%d@+%d+%d\n",
3883                                  GDK_DRAWABLE_XID (pixmap),
3884                                  pixmap_private->width, pixmap_private->height,
3885                                  rect.right - rect.left, rect.bottom - rect.top,
3886                                  rect.left, rect.top));
3887
3888               if (!(bgdc = CreateCompatibleDC (hdc)))
3889                 {
3890                   WIN32_GDI_FAILED ("CreateCompatibleDC");
3891                   break;
3892                 }
3893               if (!(oldbitmap = SelectObject (bgdc, GDK_DRAWABLE_XID (pixmap))))
3894                 {
3895                   WIN32_GDI_FAILED ("SelectObject");
3896                   DeleteDC (bgdc);
3897                   break;
3898                 }
3899               i = 0;
3900               while (i < rect.right)
3901                 {
3902                   j = 0;
3903                   while (j < rect.bottom)
3904                     {
3905                       if (i + pixmap_private->width >= rect.left
3906                           && j + pixmap_private->height >= rect.top)
3907                         {
3908                           if (!BitBlt (hdc, i, j,
3909                                        pixmap_private->width, pixmap_private->height,
3910                                        bgdc, 0, 0, SRCCOPY))
3911                             {
3912                               WIN32_GDI_FAILED ("BitBlt");
3913                               goto loopexit;
3914                             }
3915                         }
3916                       j += pixmap_private->height;
3917                     }
3918                   i += pixmap_private->width;
3919                 }
3920             loopexit:
3921               SelectObject (bgdc, oldbitmap);
3922               DeleteDC (bgdc);
3923             }
3924         }
3925       else
3926         {
3927           GDK_NOTE (EVENTS, g_print ("...BLACK_BRUSH (?)\n"));
3928           hbr = GetStockObject (BLACK_BRUSH);
3929           GetClipBox (hdc, &rect);
3930           if (!FillRect (hdc, &rect, hbr))
3931             WIN32_GDI_FAILED ("FillRect");
3932         }
3933       break;
3934
3935     case WM_PAINT:
3936       if (!GetUpdateRect(xevent->hwnd, NULL, FALSE))
3937         {
3938           GDK_NOTE (EVENTS, g_print ("WM_PAINT: %#x no update rect\n",
3939                                      xevent->hwnd));
3940           break;
3941         }
3942
3943       hdc = BeginPaint (xevent->hwnd, &paintstruct);
3944
3945       GDK_NOTE (EVENTS,
3946                 g_print ("WM_PAINT: %#x  %dx%d@+%d+%d %s dc %#x\n",
3947                          xevent->hwnd,
3948                          paintstruct.rcPaint.right - paintstruct.rcPaint.left,
3949                          paintstruct.rcPaint.bottom - paintstruct.rcPaint.top,
3950                          paintstruct.rcPaint.left, paintstruct.rcPaint.top,
3951                          (paintstruct.fErase ? "erase" : ""),
3952                          hdc));
3953
3954       EndPaint (xevent->hwnd, &paintstruct);
3955
3956       if (!(GDK_WINDOW_WIN32DATA (window)->event_mask & GDK_EXPOSURE_MASK))
3957         break;
3958
3959       if ((paintstruct.rcPaint.right == paintstruct.rcPaint.left)
3960           || (paintstruct.rcPaint.bottom == paintstruct.rcPaint.top))
3961         break;
3962
3963       event->expose.type = GDK_EXPOSE;
3964       event->expose.window = window;
3965       event->expose.area.x = paintstruct.rcPaint.left;
3966       event->expose.area.y = paintstruct.rcPaint.top;
3967       event->expose.area.width = paintstruct.rcPaint.right - paintstruct.rcPaint.left;
3968       event->expose.area.height = paintstruct.rcPaint.bottom - paintstruct.rcPaint.top;
3969       event->expose.count = 0;
3970
3971       return_val = !GDK_DRAWABLE_DESTROYED (window);
3972       if (return_val)
3973         {
3974           GList *list = gdk_queued_events;
3975           while (list != NULL )
3976             {
3977               if ((((GdkEvent *)list->data)->any.type == GDK_EXPOSE) &&
3978                   (((GdkEvent *)list->data)->any.window == window) &&
3979                   !(((GdkEventPrivate *)list->data)->flags & GDK_EVENT_PENDING))
3980                 ((GdkEvent *)list->data)->expose.count++;
3981               
3982               list = list->next;
3983             }
3984         }
3985       break;
3986
3987     case WM_SETCURSOR:
3988       GDK_NOTE (EVENTS, g_print ("WM_SETCURSOR: %#x %#x %#x\n",
3989                                  xevent->hwnd,
3990                                  LOWORD (xevent->lParam), HIWORD (xevent->lParam)));
3991
3992       if (LOWORD (xevent->lParam) != HTCLIENT)
3993         break;
3994       if (p_grab_window != NULL && p_grab_cursor != NULL)
3995         {
3996           GDK_NOTE (EVENTS, g_print ("...SetCursor(%#x)\n", p_grab_cursor));
3997           SetCursor (p_grab_cursor);
3998         }
3999       else if (!GDK_DRAWABLE_DESTROYED (window)
4000                && GDK_WINDOW_WIN32DATA (window)->xcursor)
4001         {
4002           GDK_NOTE (EVENTS, g_print ("...SetCursor(%#x)\n",
4003                                      GDK_WINDOW_WIN32DATA (window)->xcursor));
4004           SetCursor (GDK_WINDOW_WIN32DATA (window)->xcursor);
4005         }
4006
4007       if (window != curWnd)
4008         synthesize_crossing_events (window, xevent);
4009
4010       *ret_val_flagp = TRUE;
4011       *ret_valp = FALSE;
4012       break;
4013
4014     case WM_SHOWWINDOW:
4015       GDK_NOTE (EVENTS, g_print ("WM_SHOWWINDOW: %#x  %d\n",
4016                                  xevent->hwnd,
4017                                  xevent->wParam));
4018
4019       if (!(GDK_WINDOW_WIN32DATA (window)->event_mask & GDK_STRUCTURE_MASK))
4020         break;
4021
4022       event->any.type = (xevent->wParam ? GDK_MAP : GDK_UNMAP);
4023       event->any.window = window;
4024
4025       if (event->any.type == GDK_UNMAP
4026           && p_grab_window == window)
4027         gdk_pointer_ungrab (xevent->time);
4028
4029       if (event->any.type == GDK_UNMAP
4030           && k_grab_window == window)
4031         gdk_keyboard_ungrab (xevent->time);
4032
4033       return_val = !GDK_DRAWABLE_DESTROYED (window);
4034       break;
4035
4036     case WM_SIZE:
4037       GDK_NOTE (EVENTS,
4038                 g_print ("WM_SIZE: %#x  %s %dx%d\n",
4039                          xevent->hwnd,
4040                          (xevent->wParam == SIZE_MAXHIDE ? "MAXHIDE" :
4041                           (xevent->wParam == SIZE_MAXIMIZED ? "MAXIMIZED" :
4042                            (xevent->wParam == SIZE_MAXSHOW ? "MAXSHOW" :
4043                             (xevent->wParam == SIZE_MINIMIZED ? "MINIMIZED" :
4044                              (xevent->wParam == SIZE_RESTORED ? "RESTORED" : "?"))))),
4045                          LOWORD (xevent->lParam), HIWORD (xevent->lParam)));
4046
4047       if (!(GDK_WINDOW_WIN32DATA (window)->event_mask & GDK_STRUCTURE_MASK))
4048         break;
4049
4050       if (xevent->wParam == SIZE_MINIMIZED)
4051         {
4052           event->any.type = GDK_UNMAP;
4053           event->any.window = window;
4054
4055           if (p_grab_window == window)
4056             gdk_pointer_ungrab (xevent->time);
4057
4058           if (k_grab_window == window)
4059             gdk_keyboard_ungrab (xevent->time);
4060
4061           return_val = !GDK_DRAWABLE_DESTROYED (window);
4062         }
4063       else if ((xevent->wParam == SIZE_RESTORED
4064                 || xevent->wParam == SIZE_MAXIMIZED)
4065 #if 1
4066                && GDK_DRAWABLE_TYPE (window) != GDK_WINDOW_CHILD
4067 #endif
4068                                                                  )
4069         {
4070           if (LOWORD (xevent->lParam) == 0)
4071             break;
4072
4073           event->configure.type = GDK_CONFIGURE;
4074           event->configure.window = window;
4075           pt.x = 0;
4076           pt.y = 0;
4077           ClientToScreen (xevent->hwnd, &pt);
4078           event->configure.x = pt.x;
4079           event->configure.y = pt.y;
4080           event->configure.width = LOWORD (xevent->lParam);
4081           event->configure.height = HIWORD (xevent->lParam);
4082           ((GdkWindowPrivate *) window)->x = event->configure.x;
4083           ((GdkWindowPrivate *) window)->y = event->configure.y;
4084           ((GdkWindowPrivate *) window)->drawable.width = event->configure.width;
4085           ((GdkWindowPrivate *) window)->drawable.height = event->configure.height;
4086           if (((GdkWindowPrivate *) window)->resize_count > 1)
4087             ((GdkWindowPrivate *) window)->resize_count -= 1;
4088           
4089           return_val = !GDK_DRAWABLE_DESTROYED (window);
4090           if (return_val
4091               && ((GdkWindowPrivate *) window)->extension_events != 0
4092               && gdk_input_vtable.configure_event)
4093             gdk_input_vtable.configure_event (&event->configure, window);
4094         }
4095       break;
4096
4097     case WM_GETMINMAXINFO:
4098       GDK_NOTE (EVENTS, g_print ("WM_GETMINMAXINFO: %#x\n", xevent->hwnd));
4099
4100       lpmmi = (MINMAXINFO*) xevent->lParam;
4101       if (GDK_WINDOW_WIN32DATA (window)->hint_flags & GDK_HINT_MIN_SIZE)
4102         {
4103           lpmmi->ptMinTrackSize.x = GDK_WINDOW_WIN32DATA (window)->hint_min_width;
4104           lpmmi->ptMinTrackSize.y = GDK_WINDOW_WIN32DATA (window)->hint_min_height;
4105         }
4106       if (GDK_WINDOW_WIN32DATA (window)->hint_flags & GDK_HINT_MAX_SIZE)
4107         {
4108           lpmmi->ptMaxTrackSize.x = GDK_WINDOW_WIN32DATA (window)->hint_max_width;
4109           lpmmi->ptMaxTrackSize.y = GDK_WINDOW_WIN32DATA (window)->hint_max_height;
4110             
4111           lpmmi->ptMaxSize.x = GDK_WINDOW_WIN32DATA (window)->hint_max_width;
4112           lpmmi->ptMaxSize.y = GDK_WINDOW_WIN32DATA (window)->hint_max_height;
4113         }
4114       break;
4115
4116     case WM_MOVE:
4117       GDK_NOTE (EVENTS, g_print ("WM_MOVE: %#x  (%d,%d)\n",
4118                                  xevent->hwnd,
4119                                  LOWORD (xevent->lParam), HIWORD (xevent->lParam)));
4120
4121       if (!(GDK_WINDOW_WIN32DATA (window)->event_mask & GDK_STRUCTURE_MASK))
4122         break;
4123
4124       if (GDK_DRAWABLE_TYPE (window) != GDK_WINDOW_CHILD
4125           && !IsIconic(xevent->hwnd)
4126           && IsWindowVisible(xevent->hwnd))
4127         {
4128           event->configure.type = GDK_CONFIGURE;
4129           event->configure.window = window;
4130           event->configure.x = LOWORD (xevent->lParam);
4131           event->configure.y = HIWORD (xevent->lParam);
4132           GetClientRect (xevent->hwnd, &rect);
4133           event->configure.width = rect.right;
4134           event->configure.height = rect.bottom;
4135           ((GdkWindowPrivate *) window)->x = event->configure.x;
4136           ((GdkWindowPrivate *) window)->y = event->configure.y;
4137           ((GdkWindowPrivate *) window)->drawable.width = event->configure.width;
4138           ((GdkWindowPrivate *) window)->drawable.height = event->configure.height;
4139           
4140           return_val = !GDK_DRAWABLE_DESTROYED (window);
4141         }
4142       break;
4143
4144     case WM_CLOSE:
4145       GDK_NOTE (EVENTS, g_print ("WM_CLOSE: %#x\n", xevent->hwnd));
4146
4147       event->any.type = GDK_DELETE;
4148       event->any.window = window;
4149       
4150       return_val = !GDK_DRAWABLE_DESTROYED (window);
4151       break;
4152
4153 #if 0
4154     /* No, don't use delayed rendering after all. It works only if the
4155      * delayed SetClipboardData is called from the WindowProc, it
4156      * seems. (The #else part below is test code for that. It succeeds
4157      * in setting the clipboard data. But if I call SetClipboardData
4158      * in gdk_property_change (as a consequence of the
4159      * GDK_SELECTION_REQUEST event), it fails.  I deduce that this is
4160      * because delayed rendering requires that SetClipboardData is
4161      * called in the window procedure.)
4162      */
4163     case WM_RENDERFORMAT:
4164     case WM_RENDERALLFORMATS:
4165       flag = FALSE;
4166       GDK_NOTE (EVENTS, flag = TRUE);
4167       if (flag)
4168         g_print ("WM_%s: %#x %#x (%s)\n",
4169                  (xevent->message == WM_RENDERFORMAT ? "RENDERFORMAT" :
4170                   "RENDERALLFORMATS"),
4171                  xevent->hwnd,
4172                  xevent->wParam,
4173                  (xevent->wParam == CF_TEXT ? "CF_TEXT" :
4174                   (xevent->wParam == CF_DIB ? "CF_DIB" :
4175                    (xevent->wParam == CF_UNICODETEXT ? "CF_UNICODETEXT" :
4176                     (GetClipboardFormatName (xevent->wParam, buf, sizeof (buf)), buf)))));
4177
4178 #if 0
4179       event->selection.type = GDK_SELECTION_REQUEST;
4180       event->selection.window = window;
4181       event->selection.selection = gdk_clipboard_atom;
4182       if (xevent->wParam == CF_TEXT)
4183         event->selection.target = GDK_TARGET_STRING;
4184       else
4185         {
4186           GetClipboardFormatName (xevent->wParam, buf, sizeof (buf));
4187           event->selection.target = gdk_atom_intern (buf, FALSE);
4188         }
4189       event->selection.property = gdk_selection_property;
4190       event->selection.requestor = (guint32) xevent->hwnd;
4191       event->selection.time = xevent->time;
4192       return_val = !GDK_DRAWABLE_DESTROYED (window);
4193 #else
4194       /* Test code, to see if SetClipboardData works when called from
4195        * the window procedure.
4196        */
4197       {
4198         HGLOBAL hdata = GlobalAlloc (GMEM_MOVEABLE|GMEM_DDESHARE, 10);
4199         char *ptr = GlobalLock (hdata);
4200         strcpy (ptr, "Huhhaa");
4201         GlobalUnlock (hdata);
4202         if (!SetClipboardData (CF_TEXT, hdata))
4203           WIN32_API_FAILED ("SetClipboardData");
4204       }
4205       *ret_valp = 0;
4206       *ret_val_flagp = TRUE;
4207       return_val = FALSE;
4208 #endif
4209       break;
4210 #endif /* No delayed rendering */
4211
4212     case WM_DESTROY:
4213       GDK_NOTE (EVENTS, g_print ("WM_DESTROY: %#x\n", xevent->hwnd));
4214
4215       event->any.type = GDK_DESTROY;
4216       event->any.window = window;
4217       if (window != NULL && window == curWnd)
4218         {
4219           gdk_drawable_unref (curWnd);
4220           curWnd = NULL;
4221         }
4222
4223       if (p_grab_window == window)
4224         gdk_pointer_ungrab (xevent->time);
4225
4226       if (k_grab_window == window)
4227         gdk_keyboard_ungrab (xevent->time);
4228
4229       return_val = window != NULL && !GDK_DRAWABLE_DESTROYED (window);
4230
4231       if (window != NULL)
4232         gdk_window_destroy_notify (window);
4233
4234       break;
4235
4236 #ifdef HAVE_WINTAB
4237       /* Handle WINTAB events here, as we know that gdkinput.c will
4238        * use the fixed WT_DEFBASE as lcMsgBase, and we thus can use the
4239        * constants as case labels.
4240        */
4241     case WT_PACKET:
4242       GDK_NOTE (EVENTS, g_print ("WT_PACKET: %#x %d %#x\n",
4243                                  xevent->hwnd,
4244                                  xevent->wParam, xevent->lParam));
4245       goto wintab;
4246       
4247     case WT_CSRCHANGE:
4248       GDK_NOTE (EVENTS, g_print ("WT_CSRCHANGE: %#x %d %#x\n",
4249                                  xevent->hwnd,
4250                                  xevent->wParam, xevent->lParam));
4251       goto wintab;
4252       
4253     case WT_PROXIMITY:
4254       GDK_NOTE (EVENTS,
4255                 g_print ("WT_PROXIMITY: %#x %#x %d %d\n",
4256                          xevent->hwnd, xevent->wParam,
4257                          LOWORD (xevent->lParam), HIWORD (xevent->lParam)));
4258       /* Fall through */
4259     wintab:
4260       event->any.window = window;
4261       return_val = gdk_input_vtable.other_event(event, xevent);
4262       break;
4263 #endif
4264
4265     default:
4266       GDK_NOTE (EVENTS, g_print ("%s: %#x %#x %#x\n",
4267                                  gdk_win32_message_name (xevent->message),
4268                                  xevent->hwnd,
4269                                  xevent->wParam, xevent->lParam));
4270     }
4271
4272 done:
4273
4274   if (return_val)
4275     {
4276       if (event->any.window)
4277         gdk_drawable_ref (event->any.window);
4278       if (((event->any.type == GDK_ENTER_NOTIFY) ||
4279            (event->any.type == GDK_LEAVE_NOTIFY)) &&
4280           (event->crossing.subwindow != NULL))
4281         gdk_drawable_ref (event->crossing.subwindow);
4282
4283       GDK_NOTE (EVENTS, print_event (event));
4284     }
4285   else
4286     {
4287       /* Mark this event as having no resources to be freed */
4288       event->any.window = NULL;
4289       event->any.type = GDK_NOTHING;
4290     }
4291
4292   if (window)
4293     gdk_drawable_unref (window);
4294   
4295   return return_val;
4296 }
4297
4298 void
4299 gdk_events_queue (void)
4300 {
4301   GList *node;
4302   GdkEvent *event;
4303   MSG msg;
4304   LRESULT lres;
4305
4306   while (!gdk_event_queue_find_first ()
4307          && PeekMessage (&msg, NULL, 0, 0, PM_REMOVE))
4308     {
4309       GDK_NOTE (EVENTS, g_print ("PeekMessage: %#x %#x\n",
4310                                  msg.hwnd, msg.message));
4311
4312       if (paimmmpo == NULL
4313           || (paimmmpo->lpVtbl->OnTranslateMessage) (paimmmpo, &msg) != S_OK)
4314         TranslateMessage (&msg);
4315
4316       if (msg.message == g_pipe_readable_msg)
4317         {
4318           GDK_NOTE (EVENTS, g_print ("g_pipe_readable_msg: %d %d\n",
4319                                      msg.wParam, msg.lParam));
4320
4321           g_io_channel_win32_pipe_readable (msg.wParam, msg.lParam);
4322
4323           continue;
4324         }
4325       
4326       DispatchMessage (&msg);
4327     }
4328 }
4329
4330 static gboolean  
4331 gdk_event_prepare (gpointer  source_data, 
4332                    GTimeVal *current_time,
4333                    gint     *timeout,
4334                    gpointer  user_data)
4335 {
4336   MSG msg;
4337   gboolean retval;
4338   
4339   GDK_THREADS_ENTER ();
4340
4341   *timeout = -1;
4342
4343   retval = (gdk_event_queue_find_first () != NULL)
4344               || PeekMessage (&msg, NULL, 0, 0, PM_NOREMOVE);
4345
4346   GDK_THREADS_LEAVE ();
4347
4348   return retval;
4349 }
4350
4351 static gboolean  
4352 gdk_event_check (gpointer  source_data,
4353                  GTimeVal *current_time,
4354                  gpointer  user_data)
4355 {
4356   MSG msg;
4357   gboolean retval;
4358   
4359   GDK_THREADS_ENTER ();
4360
4361   if (event_poll_fd.revents & G_IO_IN)
4362     retval = (gdk_event_queue_find_first () != NULL)
4363               || PeekMessage (&msg, NULL, 0, 0, PM_NOREMOVE);
4364   else
4365     retval = FALSE;
4366
4367   GDK_THREADS_LEAVE ();
4368
4369   return retval;
4370 }
4371
4372 static gboolean  
4373 gdk_event_dispatch (gpointer  source_data,
4374                     GTimeVal *current_time,
4375                     gpointer  user_data)
4376 {
4377   GdkEvent *event;
4378  
4379   GDK_THREADS_ENTER ();
4380
4381   gdk_events_queue();
4382   event = gdk_event_unqueue();
4383
4384   if (event)
4385     {
4386       if (gdk_event_func)
4387         (*gdk_event_func) (event, gdk_event_data);
4388       
4389       gdk_event_free (event);
4390     }
4391   
4392   GDK_THREADS_LEAVE ();
4393
4394   return TRUE;
4395 }
4396
4397 /* Sends a ClientMessage to all toplevel client windows */
4398 gboolean
4399 gdk_event_send_client_message (GdkEvent *event, guint32 xid)
4400 {
4401   /* XXX */
4402   return FALSE;
4403 }
4404
4405 void
4406 gdk_event_send_clientmessage_toall (GdkEvent *event)
4407 {
4408   /* XXX */
4409 }
4410
4411 void
4412 gdk_flush (void)
4413 {
4414   GdiFlush ();
4415 }
4416
4417 #ifdef G_ENABLE_DEBUG
4418
4419 gchar *
4420 gdk_win32_message_name (UINT msg)
4421 {
4422   static gchar bfr[100];
4423
4424   switch (msg)
4425     {
4426 #define CASE(x) case x: return #x
4427       CASE (WM_NULL);
4428       CASE (WM_CREATE);
4429       CASE (WM_DESTROY);
4430       CASE (WM_MOVE);
4431       CASE (WM_SIZE);
4432       CASE (WM_ACTIVATE);
4433       CASE (WM_SETFOCUS);
4434       CASE (WM_KILLFOCUS);
4435       CASE (WM_ENABLE);
4436       CASE (WM_SETREDRAW);
4437       CASE (WM_SETTEXT);
4438       CASE (WM_GETTEXT);
4439       CASE (WM_GETTEXTLENGTH);
4440       CASE (WM_PAINT);
4441       CASE (WM_CLOSE);
4442       CASE (WM_QUERYENDSESSION);
4443       CASE (WM_QUERYOPEN);
4444       CASE (WM_ENDSESSION);
4445       CASE (WM_QUIT);
4446       CASE (WM_ERASEBKGND);
4447       CASE (WM_SYSCOLORCHANGE);
4448       CASE (WM_SHOWWINDOW);
4449       CASE (WM_WININICHANGE);
4450       CASE (WM_DEVMODECHANGE);
4451       CASE (WM_ACTIVATEAPP);
4452       CASE (WM_FONTCHANGE);
4453       CASE (WM_TIMECHANGE);
4454       CASE (WM_CANCELMODE);
4455       CASE (WM_SETCURSOR);
4456       CASE (WM_MOUSEACTIVATE);
4457       CASE (WM_CHILDACTIVATE);
4458       CASE (WM_QUEUESYNC);
4459       CASE (WM_GETMINMAXINFO);
4460       CASE (WM_PAINTICON);
4461       CASE (WM_ICONERASEBKGND);
4462       CASE (WM_NEXTDLGCTL);
4463       CASE (WM_SPOOLERSTATUS);
4464       CASE (WM_DRAWITEM);
4465       CASE (WM_MEASUREITEM);
4466       CASE (WM_DELETEITEM);
4467       CASE (WM_VKEYTOITEM);
4468       CASE (WM_CHARTOITEM);
4469       CASE (WM_SETFONT);
4470       CASE (WM_GETFONT);
4471       CASE (WM_SETHOTKEY);
4472       CASE (WM_GETHOTKEY);
4473       CASE (WM_QUERYDRAGICON);
4474       CASE (WM_COMPAREITEM);
4475       CASE (WM_GETOBJECT);
4476       CASE (WM_COMPACTING);
4477       CASE (WM_WINDOWPOSCHANGING);
4478       CASE (WM_WINDOWPOSCHANGED);
4479       CASE (WM_POWER);
4480       CASE (WM_COPYDATA);
4481       CASE (WM_CANCELJOURNAL);
4482       CASE (WM_NOTIFY);
4483       CASE (WM_INPUTLANGCHANGEREQUEST);
4484       CASE (WM_INPUTLANGCHANGE);
4485       CASE (WM_TCARD);
4486       CASE (WM_HELP);
4487       CASE (WM_USERCHANGED);
4488       CASE (WM_NOTIFYFORMAT);
4489       CASE (WM_CONTEXTMENU);
4490       CASE (WM_STYLECHANGING);
4491       CASE (WM_STYLECHANGED);
4492       CASE (WM_DISPLAYCHANGE);
4493       CASE (WM_GETICON);
4494       CASE (WM_SETICON);
4495       CASE (WM_NCCREATE);
4496       CASE (WM_NCDESTROY);
4497       CASE (WM_NCCALCSIZE);
4498       CASE (WM_NCHITTEST);
4499       CASE (WM_NCPAINT);
4500       CASE (WM_NCACTIVATE);
4501       CASE (WM_GETDLGCODE);
4502       CASE (WM_SYNCPAINT);
4503       CASE (WM_NCMOUSEMOVE);
4504       CASE (WM_NCLBUTTONDOWN);
4505       CASE (WM_NCLBUTTONUP);
4506       CASE (WM_NCLBUTTONDBLCLK);
4507       CASE (WM_NCRBUTTONDOWN);
4508       CASE (WM_NCRBUTTONUP);
4509       CASE (WM_NCRBUTTONDBLCLK);
4510       CASE (WM_NCMBUTTONDOWN);
4511       CASE (WM_NCMBUTTONUP);
4512       CASE (WM_NCMBUTTONDBLCLK);
4513       CASE (WM_NCXBUTTONDOWN);
4514       CASE (WM_NCXBUTTONUP);
4515       CASE (WM_NCXBUTTONDBLCLK);
4516       CASE (WM_KEYDOWN);
4517       CASE (WM_KEYUP);
4518       CASE (WM_CHAR);
4519       CASE (WM_DEADCHAR);
4520       CASE (WM_SYSKEYDOWN);
4521       CASE (WM_SYSKEYUP);
4522       CASE (WM_SYSCHAR);
4523       CASE (WM_SYSDEADCHAR);
4524       CASE (WM_KEYLAST);
4525       CASE (WM_IME_STARTCOMPOSITION);
4526       CASE (WM_IME_ENDCOMPOSITION);
4527       CASE (WM_IME_COMPOSITION);
4528       CASE (WM_INITDIALOG);
4529       CASE (WM_COMMAND);
4530       CASE (WM_SYSCOMMAND);
4531       CASE (WM_TIMER);
4532       CASE (WM_HSCROLL);
4533       CASE (WM_VSCROLL);
4534       CASE (WM_INITMENU);
4535       CASE (WM_INITMENUPOPUP);
4536       CASE (WM_MENUSELECT);
4537       CASE (WM_MENUCHAR);
4538       CASE (WM_ENTERIDLE);
4539       CASE (WM_MENURBUTTONUP);
4540       CASE (WM_MENUDRAG);
4541       CASE (WM_MENUGETOBJECT);
4542       CASE (WM_UNINITMENUPOPUP);
4543       CASE (WM_MENUCOMMAND);
4544       CASE (WM_CHANGEUISTATE);
4545       CASE (WM_UPDATEUISTATE);
4546       CASE (WM_QUERYUISTATE);
4547       CASE (WM_CTLCOLORMSGBOX);
4548       CASE (WM_CTLCOLOREDIT);
4549       CASE (WM_CTLCOLORLISTBOX);
4550       CASE (WM_CTLCOLORBTN);
4551       CASE (WM_CTLCOLORDLG);
4552       CASE (WM_CTLCOLORSCROLLBAR);
4553       CASE (WM_CTLCOLORSTATIC);
4554       CASE (WM_MOUSEMOVE);
4555       CASE (WM_LBUTTONDOWN);
4556       CASE (WM_LBUTTONUP);
4557       CASE (WM_LBUTTONDBLCLK);
4558       CASE (WM_RBUTTONDOWN);
4559       CASE (WM_RBUTTONUP);
4560       CASE (WM_RBUTTONDBLCLK);
4561       CASE (WM_MBUTTONDOWN);
4562       CASE (WM_MBUTTONUP);
4563       CASE (WM_MBUTTONDBLCLK);
4564       CASE (WM_MOUSEWHEEL);
4565       CASE (WM_XBUTTONDOWN);
4566       CASE (WM_XBUTTONUP);
4567       CASE (WM_XBUTTONDBLCLK);
4568       CASE (WM_PARENTNOTIFY);
4569       CASE (WM_ENTERMENULOOP);
4570       CASE (WM_EXITMENULOOP);
4571       CASE (WM_NEXTMENU);
4572       CASE (WM_SIZING);
4573       CASE (WM_CAPTURECHANGED);
4574       CASE (WM_MOVING);
4575       CASE (WM_POWERBROADCAST);
4576       CASE (WM_DEVICECHANGE);
4577       CASE (WM_MDICREATE);
4578       CASE (WM_MDIDESTROY);
4579       CASE (WM_MDIACTIVATE);
4580       CASE (WM_MDIRESTORE);
4581       CASE (WM_MDINEXT);
4582       CASE (WM_MDIMAXIMIZE);
4583       CASE (WM_MDITILE);
4584       CASE (WM_MDICASCADE);
4585       CASE (WM_MDIICONARRANGE);
4586       CASE (WM_MDIGETACTIVE);
4587       CASE (WM_MDISETMENU);
4588       CASE (WM_ENTERSIZEMOVE);
4589       CASE (WM_EXITSIZEMOVE);
4590       CASE (WM_DROPFILES);
4591       CASE (WM_MDIREFRESHMENU);
4592       CASE (WM_IME_SETCONTEXT);
4593       CASE (WM_IME_NOTIFY);
4594       CASE (WM_IME_CONTROL);
4595       CASE (WM_IME_COMPOSITIONFULL);
4596       CASE (WM_IME_SELECT);
4597       CASE (WM_IME_CHAR);
4598       CASE (WM_IME_REQUEST);
4599       CASE (WM_IME_KEYDOWN);
4600       CASE (WM_IME_KEYUP);
4601       CASE (WM_MOUSEHOVER);
4602       CASE (WM_MOUSELEAVE);
4603       CASE (WM_NCMOUSEHOVER);
4604       CASE (WM_NCMOUSELEAVE);
4605       CASE (WM_CUT);
4606       CASE (WM_COPY);
4607       CASE (WM_PASTE);
4608       CASE (WM_CLEAR);
4609       CASE (WM_UNDO);
4610       CASE (WM_RENDERFORMAT);
4611       CASE (WM_RENDERALLFORMATS);
4612       CASE (WM_DESTROYCLIPBOARD);
4613       CASE (WM_DRAWCLIPBOARD);
4614       CASE (WM_PAINTCLIPBOARD);
4615       CASE (WM_VSCROLLCLIPBOARD);
4616       CASE (WM_SIZECLIPBOARD);
4617       CASE (WM_ASKCBFORMATNAME);
4618       CASE (WM_CHANGECBCHAIN);
4619       CASE (WM_HSCROLLCLIPBOARD);
4620       CASE (WM_QUERYNEWPALETTE);
4621       CASE (WM_PALETTEISCHANGING);
4622       CASE (WM_PALETTECHANGED);
4623       CASE (WM_HOTKEY);
4624       CASE (WM_PRINT);
4625       CASE (WM_PRINTCLIENT);
4626       CASE (WM_APPCOMMAND);
4627       CASE (WM_HANDHELDFIRST);
4628       CASE (WM_HANDHELDLAST);
4629       CASE (WM_AFXFIRST);
4630       CASE (WM_AFXLAST);
4631       CASE (WM_PENWINFIRST);
4632       CASE (WM_PENWINLAST);
4633       CASE (WM_APP);
4634 #undef CASE
4635     default:
4636       if (msg >= WM_HANDHELDFIRST && msg <= WM_HANDHELDLAST)
4637         sprintf (bfr, "WM_HANDHELDFIRST+%d", msg - WM_HANDHELDFIRST);
4638       else if (msg >= WM_AFXFIRST && msg <= WM_AFXLAST)
4639         sprintf (bfr, "WM_AFXFIRST+%d", msg - WM_AFXFIRST);
4640       else if (msg >= WM_PENWINFIRST && msg <= WM_PENWINLAST)
4641         sprintf (bfr, "WM_PENWINFIRST+%d", msg - WM_PENWINFIRST);
4642       else if (msg >= WM_USER && msg <= 0x7FFF)
4643         sprintf (bfr, "WM_USER+%d", msg - WM_USER);
4644       else if (msg >= 0xC000 && msg <= 0xFFFF)
4645         sprintf (bfr, "reg-%#x", msg);
4646       else
4647         sprintf (bfr, "unk-%#x", msg);
4648       return bfr;
4649     }
4650   g_assert_not_reached ();
4651 }
4652       
4653 #endif /* G_ENABLE_DEBUG */