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