/[rdesktop]/sourceforge.net/trunk/rdesktop/xwin.c
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Diff of /sourceforge.net/trunk/rdesktop/xwin.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 59 by jsorg71, Sun Jul 14 04:20:25 2002 UTC revision 300 by matthewc, Thu Jan 30 11:20:30 2003 UTC
# Line 1  Line 1 
1  /*  /*
2     rdesktop: A Remote Desktop Protocol client.     rdesktop: A Remote Desktop Protocol client.
3     User interface services - X Window System     User interface services - X Window System
4     Copyright (C) Matthew Chapman 1999-2001     Copyright (C) Matthew Chapman 1999-2002
5    
6     This program is free software; you can redistribute it and/or modify     This program is free software; you can redistribute it and/or modify
7     it under the terms of the GNU General Public License as published by     it under the terms of the GNU General Public License as published by
# Line 20  Line 20 
20    
21  #include <X11/Xlib.h>  #include <X11/Xlib.h>
22  #include <X11/Xutil.h>  #include <X11/Xutil.h>
 #include <X11/XKBlib.h>  
23  #include <time.h>  #include <time.h>
24  #include <errno.h>  #include <errno.h>
25  #include "rdesktop.h"  #include "rdesktop.h"
26    
 extern char keymapname[16];  
 extern int keylayout;  
27  extern int width;  extern int width;
28  extern int height;  extern int height;
29  extern BOOL sendmotion;  extern BOOL sendmotion;
30  extern BOOL fullscreen;  extern BOOL fullscreen;
31    extern BOOL grab_keyboard;
32    extern BOOL hide_decorations;
33    extern char title[];
34    BOOL enable_compose = False;
35    BOOL focused;
36    BOOL mouse_in_wnd;
37    
38  Display *display;  Display *display;
 XkbDescPtr xkb;  
39  static int x_socket;  static int x_socket;
40    static Screen *screen;
41  static Window wnd;  static Window wnd;
42  static GC gc;  static GC gc;
43  static Visual *visual;  static Visual *visual;
44  static int depth;  static int depth;
45  static int bpp;  static int bpp;
46    static XIM IM;
47    static XIC IC;
48    static XModifierKeymap *mod_map;
49    static Cursor current_cursor;
50    static Atom protocol_atom, kill_atom;
51    
52  /* endianness */  /* endianness */
53  static BOOL host_be;  static BOOL host_be;
# Line 49  static BOOL xserver_be; Line 57  static BOOL xserver_be;
57  static BOOL ownbackstore;  static BOOL ownbackstore;
58  static Pixmap backstore;  static Pixmap backstore;
59    
60  /* needed to keep track of the modifiers */  /* MWM decorations */
61  static unsigned int numlock_modifier_mask = 0;  #define MWM_HINTS_DECORATIONS   (1L << 1)
62  static unsigned int key_down_state = 0;  #define PROP_MOTIF_WM_HINTS_ELEMENTS    5
63    typedef struct
64    {
65  #define DShift1Mask   (1<<0)          unsigned long flags;
66  #define DLockMask     (1<<1)          unsigned long functions;
67  #define DControl1Mask (1<<2)          unsigned long decorations;
68  #define DMod1Mask     (1<<3)          long inputMode;
69  #define DMod2Mask     (1<<4)          unsigned long status;
70  #define DMod3Mask     (1<<5)  }
71  #define DMod4Mask     (1<<6)  PropMotifWmHints;
72  #define DMod5Mask     (1<<7)  
 #define DShift2Mask   (1<<8)  
 #define DControl2Mask (1<<9)  
 #define DNumLockMask  (1<<10)  
73    
74  #define FILL_RECTANGLE(x,y,cx,cy)\  #define FILL_RECTANGLE(x,y,cx,cy)\
75  { \  { \
# Line 73  static unsigned int key_down_state = 0; Line 78  static unsigned int key_down_state = 0;
78                  XFillRectangle(display, backstore, gc, x, y, cx, cy); \                  XFillRectangle(display, backstore, gc, x, y, cx, cy); \
79  }  }
80    
81    #define FILL_RECTANGLE_BACKSTORE(x,y,cx,cy)\
82    { \
83            XFillRectangle(display, ownbackstore ? backstore : wnd, gc, x, y, cx, cy); \
84    }
85    
86  /* colour maps */  /* colour maps */
87  static BOOL owncolmap;  BOOL owncolmap = False;
88  static Colormap xcolmap;  static Colormap xcolmap;
 static uint32 white;  
89  static uint32 *colmap;  static uint32 *colmap;
90    
91  #define TRANSLATE(col)          ( owncolmap ? col : translate_colour(colmap[col]) )  #define TRANSLATE(col)          ( owncolmap ? col : translate_colour(colmap[col]) )
# Line 105  static int rop2_map[] = { Line 114  static int rop2_map[] = {
114  #define SET_FUNCTION(rop2)      { if (rop2 != ROP2_COPY) XSetFunction(display, gc, rop2_map[rop2]); }  #define SET_FUNCTION(rop2)      { if (rop2 != ROP2_COPY) XSetFunction(display, gc, rop2_map[rop2]); }
115  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }
116    
117  void xwin_get_numlock_mask();  void
118  void xwin_mod_update(uint32 state, uint32 ev_time );  mwm_hide_decorations(void)
119  void xwin_mod_release(uint32 state, uint32 ev_time, uint32 scancode);  {
120  void xwin_mod_press(uint32 state, uint32 ev_time, uint32 scancode);          PropMotifWmHints motif_hints;
121            Atom hintsatom;
122    
123            /* setup the property */
124            motif_hints.flags = MWM_HINTS_DECORATIONS;
125            motif_hints.decorations = 0;
126    
127            /* get the atom for the property */
128            hintsatom = XInternAtom(display, "_MOTIF_WM_HINTS", False);
129            if (!hintsatom)
130            {
131                    warning("Failed to get atom _MOTIF_WM_HINTS: probably your window manager does not support MWM hints\n");
132                    return;
133            }
134    
135            XChangeProperty(display, wnd, hintsatom, hintsatom, 32, PropModeReplace,
136                            (unsigned char *) &motif_hints, PROP_MOTIF_WM_HINTS_ELEMENTS);
137    }
138    
139  static void  static void
140  translate8(uint8 *data, uint8 *out, uint8 *end)  translate8(uint8 * data, uint8 * out, uint8 * end)
141  {  {
142          while (out < end)          while (out < end)
143                  *(out++) = (uint8)colmap[*(data++)];                  *(out++) = (uint8) colmap[*(data++)];
144  }  }
145    
146  static void  static void
147  translate16(uint8 *data, uint16 *out, uint16 *end)  translate16(uint8 * data, uint16 * out, uint16 * end)
148  {  {
149          while (out < end)          while (out < end)
150                  *(out++) = (uint16)colmap[*(data++)];                  *(out++) = (uint16) colmap[*(data++)];
151  }  }
152    
153  /* little endian - conversion happens when colourmap is built */  /* little endian - conversion happens when colourmap is built */
154  static void  static void
155  translate24(uint8 *data, uint8 *out, uint8 *end)  translate24(uint8 * data, uint8 * out, uint8 * end)
156  {  {
157          uint32 value;          uint32 value;
158    
# Line 140  translate24(uint8 *data, uint8 *out, uin Line 166  translate24(uint8 *data, uint8 *out, uin
166  }  }
167    
168  static void  static void
169  translate32(uint8 *data, uint32 *out, uint32 *end)  translate32(uint8 * data, uint32 * out, uint32 * end)
170  {  {
171          while (out < end)          while (out < end)
172                  *(out++) = colmap[*(data++)];                  *(out++) = colmap[*(data++)];
173  }  }
174    
175  static uint8 *  static uint8 *
176  translate_image(int width, int height, uint8 *data)  translate_image(int width, int height, uint8 * data)
177  {  {
178          int size = width * height * bpp/8;          int size = width * height * bpp / 8;
179          uint8 *out = xmalloc(size);          uint8 *out = xmalloc(size);
180          uint8 *end = out + size;          uint8 *end = out + size;
181    
# Line 160  translate_image(int width, int height, u Line 186  translate_image(int width, int height, u
186                          break;                          break;
187    
188                  case 16:                  case 16:
189                          translate16(data, (uint16 *)out, (uint16 *)end);                          translate16(data, (uint16 *) out, (uint16 *) end);
190                          break;                          break;
191    
192                  case 24:                  case 24:
# Line 168  translate_image(int width, int height, u Line 194  translate_image(int width, int height, u
194                          break;                          break;
195    
196                  case 32:                  case 32:
197                          translate32(data, (uint32 *)out, (uint32 *)end);                          translate32(data, (uint32 *) out, (uint32 *) end);
198                          break;                          break;
199          }          }
200    
# Line 205  translate_colour(uint32 colour) Line 231  translate_colour(uint32 colour)
231  }  }
232    
233  BOOL  BOOL
234  ui_create_window(char *title)  get_key_state(unsigned int state, uint32 keysym)
235  {  {
236          XSetWindowAttributes attribs;          int modifierpos, key, keysymMask = 0;
237          XClassHint *classhints;          int offset;
         XSizeHints *sizehints;  
         unsigned long input_mask;  
         XPixmapFormatValues *pfm;  
         Screen *screen;  
         uint16 test;  
         int i;  
           
         int xkb_minor, xkb_major;  
         int xkb_event, xkb_error, xkb_reason;  
   
         /* compare compiletime libs with runtime libs. */  
         xkb_major = XkbMajorVersion;  
         xkb_minor = XkbMinorVersion;  
         if( XkbLibraryVersion( &xkb_major, &xkb_minor ) == False )  
         {  
                 error("please re-compile rdesktop\ncompile time version of xkb is not compatible with\nyour runtime version of the library\n");  
                 return False;  
         }  
238    
239            KeyCode keycode = XKeysymToKeycode(display, keysym);
240    
241            if (keycode == NoSymbol)
242                    return False;
243    
244          display = XkbOpenDisplay( NULL, &xkb_event, &xkb_error, &xkb_major, &xkb_minor, &xkb_reason );          for (modifierpos = 0; modifierpos < 8; modifierpos++)
         switch(xkb_reason)  
245          {          {
246                  case XkbOD_BadLibraryVersion:                  offset = mod_map->max_keypermod * modifierpos;
247                          error("XkbOD_BadLibraryVersion: XKB extensions in server and the library rdesktop is linked against aren't compatible with each other.\n");  
248                          break;                  for (key = 0; key < mod_map->max_keypermod; key++)
249                  case XkbOD_ConnectionRefused:                  {
250                          error("XkbOD_ConnectionRefused\n");                          if (mod_map->modifiermap[offset + key] == keycode)
251                          break;                                  keysymMask |= 1 << modifierpos;
252                  case XkbOD_BadServerVersion:                  }
                         error("XkbOD_BadServerVersion\n");  
                         break;  
                 case XkbOD_NonXkbServer:  
                         error("XkbOD_NonXkbServer: XKB extension not present in server\nupdate your X server.\n");  
                         break;  
                 case XkbOD_Success:  
                         DEBUG("XkbOD_Success: Connection established with display\n");  
                         break;  
253          }          }
254    
255            return (state & keysymMask) ? True : False;
256    }
257    
258    BOOL
259    ui_init(void)
260    {
261            XPixmapFormatValues *pfm;
262            uint16 test;
263            int i;
264    
265            display = XOpenDisplay(NULL);
266          if (display == NULL)          if (display == NULL)
267          {          {
268                  error("Failed to open display\n");                  error("Failed to open display: %s\n", XDisplayName(NULL));
269                  return False;                  return False;
270          }          }
271    
# Line 259  ui_create_window(char *title) Line 273  ui_create_window(char *title)
273          screen = DefaultScreenOfDisplay(display);          screen = DefaultScreenOfDisplay(display);
274          visual = DefaultVisualOfScreen(screen);          visual = DefaultVisualOfScreen(screen);
275          depth = DefaultDepthOfScreen(screen);          depth = DefaultDepthOfScreen(screen);
276            
277          pfm = XListPixmapFormats(display, &i);          pfm = XListPixmapFormats(display, &i);
278          if (pfm != NULL)          if (pfm != NULL)
279          {          {
# Line 267  ui_create_window(char *title) Line 281  ui_create_window(char *title)
281                     desirable, e.g. 24 bits->32 bits. */                     desirable, e.g. 24 bits->32 bits. */
282                  while (i--)                  while (i--)
283                  {                  {
284                          if ((pfm[i].depth == depth)                          if ((pfm[i].depth == depth) && (pfm[i].bits_per_pixel > bpp))
                             && (pfm[i].bits_per_pixel > bpp))  
285                          {                          {
286                                  bpp = pfm[i].bits_per_pixel;                                  bpp = pfm[i].bits_per_pixel;
287                          }                          }
# Line 283  ui_create_window(char *title) Line 296  ui_create_window(char *title)
296                  return False;                  return False;
297          }          }
298    
299          if (depth <= 8)          if (owncolmap != True)
300                  owncolmap = True;          {
         else  
301                  xcolmap = DefaultColormapOfScreen(screen);                  xcolmap = DefaultColormapOfScreen(screen);
302                    if (depth <= 8)
303                            warning("Screen depth is 8 bits or lower: you may want to use -C for a private colourmap\n");
304            }
305    
306            gc = XCreateGC(display, RootWindowOfScreen(screen), 0, NULL);
307    
308            if (DoesBackingStore(screen) != Always)
309                    ownbackstore = True;
310    
311          test = 1;          test = 1;
312          host_be = !(BOOL)(*(uint8 *)(&test));          host_be = !(BOOL) (*(uint8 *) (&test));
313          xserver_be = (ImageByteOrder(display) == MSBFirst);          xserver_be = (ImageByteOrder(display) == MSBFirst);
314    
315          white = WhitePixelOfScreen(screen);          if ((width == 0) || (height == 0))
316          attribs.background_pixel = BlackPixelOfScreen(screen);          {
317          attribs.backing_store = DoesBackingStore(screen);                  /* Fetch geometry from _NET_WORKAREA */
318                    uint32 x, y, cx, cy;
319    
320          if (attribs.backing_store == NotUseful)                  if (get_current_workarea(&x, &y, &cx, &cy) == 0)
321                  ownbackstore = True;                  {
322                            width = cx;
323                            height = cy;
324                    }
325                    else
326                    {
327                            warning("Failed to get workarea: probably your window manager does not support extended hints\n");
328                            width = 800;
329                            height = 600;
330                    }
331            }
332    
333          if (fullscreen)          if (fullscreen)
334          {          {
                 attribs.override_redirect = True;  
335                  width = WidthOfScreen(screen);                  width = WidthOfScreen(screen);
336                  height = HeightOfScreen(screen);                  height = HeightOfScreen(screen);
337          }          }
338          else  
339            /* make sure width is a multiple of 4 */
340            width = (width + 3) & ~3;
341    
342            if (ownbackstore)
343          {          {
344                  attribs.override_redirect = False;                  backstore =
345                            XCreatePixmap(display, RootWindowOfScreen(screen), width, height, depth);
346    
347                    /* clear to prevent rubbish being exposed at startup */
348                    XSetForeground(display, gc, BlackPixelOfScreen(screen));
349                    XFillRectangle(display, backstore, gc, 0, 0, width, height);
350          }          }
351    
352          width = (width + 3) & ~3; /* make width a multiple of 32 bits */          mod_map = XGetModifierMapping(display);
353    
354            if (enable_compose)
355                    IM = XOpenIM(display, NULL, NULL, NULL);
356    
357            xkeymap_init();
358            return True;
359    }
360    
361          wnd = XCreateWindow(display, RootWindowOfScreen(screen),  void
362                              0, 0, width, height, 0, CopyFromParent,  ui_deinit(void)
363                              InputOutput, CopyFromParent,  {
364                              CWBackingStore | CWBackPixel | CWOverrideRedirect,          if (IM != NULL)
365                              &attribs);                  XCloseIM(IM);
366    
367            XFreeModifiermap(mod_map);
368    
369            if (ownbackstore)
370                    XFreePixmap(display, backstore);
371    
372            XFreeGC(display, gc);
373            XCloseDisplay(display);
374            display = NULL;
375    }
376    
377    BOOL
378    ui_create_window(void)
379    {
380            XSetWindowAttributes attribs;
381            XClassHint *classhints;
382            XSizeHints *sizehints;
383            int wndwidth, wndheight;
384            long input_mask, ic_input_mask;
385            XEvent xevent;
386    
387            wndwidth = fullscreen ? WidthOfScreen(screen) : width;
388            wndheight = fullscreen ? HeightOfScreen(screen) : height;
389    
390            attribs.background_pixel = BlackPixelOfScreen(screen);
391            attribs.backing_store = ownbackstore ? NotUseful : Always;
392            attribs.override_redirect = fullscreen;
393    
394            wnd = XCreateWindow(display, RootWindowOfScreen(screen), 0, 0, wndwidth, wndheight,
395                                0, CopyFromParent, InputOutput, CopyFromParent,
396                                CWBackPixel | CWBackingStore | CWOverrideRedirect, &attribs);
397    
398          XStoreName(display, wnd, title);          XStoreName(display, wnd, title);
399    
400            if (hide_decorations)
401                    mwm_hide_decorations();
402    
403          classhints = XAllocClassHint();          classhints = XAllocClassHint();
404          if (classhints != NULL)          if (classhints != NULL)
405          {          {
# Line 338  ui_create_window(char *title) Line 418  ui_create_window(char *title)
418                  XFree(sizehints);                  XFree(sizehints);
419          }          }
420    
421          xkeymap_init();          input_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask |
422                    VisibilityChangeMask | FocusChangeMask;
423    
         input_mask = KeyPressMask | KeyReleaseMask |  
                          ButtonPressMask | ButtonReleaseMask |  
                          EnterWindowMask | LeaveWindowMask;  
424          if (sendmotion)          if (sendmotion)
425                  input_mask |= PointerMotionMask;                  input_mask |= PointerMotionMask;
   
426          if (ownbackstore)          if (ownbackstore)
427                  input_mask |= ExposureMask;                  input_mask |= ExposureMask;
428            if (fullscreen || grab_keyboard)
429                    input_mask |= EnterWindowMask;
430            if (grab_keyboard)
431                    input_mask |= LeaveWindowMask;
432    
433            if (IM != NULL)
434            {
435                    IC = XCreateIC(IM, XNInputStyle, (XIMPreeditNothing | XIMStatusNothing),
436                                   XNClientWindow, wnd, XNFocusWindow, wnd, NULL);
437    
438                    if ((IC != NULL)
439                        && (XGetICValues(IC, XNFilterEvents, &ic_input_mask, NULL) == NULL))
440                            input_mask |= ic_input_mask;
441            }
442    
443          XSelectInput(display, wnd, input_mask);          XSelectInput(display, wnd, input_mask);
         gc = XCreateGC(display, wnd, 0, NULL);  
   
         if (ownbackstore)  
                 backstore = XCreatePixmap(display, wnd, width, height, depth);  
   
444          XMapWindow(display, wnd);          XMapWindow(display, wnd);
445    
446          /* TODO: error texts... make them friendly. */          /* wait for VisibilityNotify */
447          xkb = XkbGetKeyboard(display, XkbAllComponentsMask, XkbUseCoreKbd);          do
         if ((int)xkb == BadAlloc || xkb == NULL)  
         {  
                         error( "XkbGetKeyboard failed.\n");  
                         exit(0);  
         }  
   
         /* TODO: error texts... make them friendly. */  
         if( XkbSelectEvents(display, xkb->device_spec, XkbAllEventsMask, XkbAllEventsMask) == False )  
448          {          {
449                          error( "XkbSelectEvents failed.\n");                  XMaskEvent(display, VisibilityChangeMask, &xevent);
                         exit(0);  
450          }          }
451                    while (xevent.type != VisibilityNotify);
452          xwin_get_numlock_mask();  
453            focused = False;
454            mouse_in_wnd = False;
455    
456            /* handle the WM_DELETE_WINDOW protocol */
457            protocol_atom = XInternAtom(display, "WM_PROTOCOLS", True);
458            kill_atom = XInternAtom(display, "WM_DELETE_WINDOW", True);
459            XSetWMProtocols(display, wnd, &kill_atom, 1);
460    
461          return True;          return True;
462  }  }
463    
464  void  void
465  xwin_get_numlock_mask()  ui_destroy_window(void)
466  {  {
467          KeyCode numlockcode;          if (IC != NULL)
468          KeyCode* keycode;                  XDestroyIC(IC);
         XModifierKeymap *modmap;  
         int i,j;  
   
         /* Find out if numlock is already defined as a modifier key, and if so where */  
         numlockcode = XKeysymToKeycode(display, 0xFF7F);        /* XF_Num_Lock = 0xFF7F */  
         if (numlockcode) {  
                 modmap = XGetModifierMapping(display);  
                 if (modmap) {  
                         keycode = modmap->modifiermap;  
                         for (i = 0; i < 8; i++)  
                                 for (j = modmap->max_keypermod; j--;) {  
                                         if (*keycode == numlockcode) {  
                                                 numlock_modifier_mask = (1 << i);  
                                                 i = 8;  
                                                 break;  
                                         }  
                                         keycode++;  
                                 }  
                 if (!numlock_modifier_mask) {  
                                 modmap->modifiermap[7 * modmap->max_keypermod] = numlockcode;  
                                 if (XSetModifierMapping(display, modmap) == MappingSuccess)  
                                         numlock_modifier_mask = (1 << 7);  
                                 else  
                                         printf("XSetModifierMapping failed!\n");  
                         }  
                         XFreeModifiermap(modmap);  
                 }  
         }  
469    
470          if (!numlock_modifier_mask)          XDestroyWindow(display, wnd);
                 printf("WARNING: Failed to get a numlock modifier mapping.\n");  
                   
471  }  }
472    
473  void  void
474  ui_destroy_window()  xwin_toggle_fullscreen(void)
475  {  {
476          if( xkb != NULL )          Pixmap contents = 0;
                 XkbFreeKeyboard(xkb, XkbAllControlsMask, True);  
477    
478          if (ownbackstore)          if (!ownbackstore)
479                  XFreePixmap(display, backstore);          {
480                    /* need to save contents of window */
481                    contents = XCreatePixmap(display, wnd, width, height, depth);
482                    XCopyArea(display, wnd, contents, gc, 0, 0, width, height, 0, 0);
483            }
484    
485          XFreeGC(display, gc);          ui_destroy_window();
486          XDestroyWindow(display, wnd);          fullscreen = !fullscreen;
487          XCloseDisplay(display);          ui_create_window();
488          display = NULL;  
489            XDefineCursor(display, wnd, current_cursor);
490    
491            if (!ownbackstore)
492            {
493                    XCopyArea(display, contents, wnd, gc, 0, 0, width, height, 0, 0);
494                    XFreePixmap(display, contents);
495            }
496  }  }
497    
498  static void  /* Process all events in Xlib queue
499  xwin_process_events()     Returns 0 after user quit, 1 otherwise */
500    static int
501    xwin_process_events(void)
502  {  {
503          XEvent xevent;          XEvent xevent;
   
504          KeySym keysym;          KeySym keysym;
         uint8 scancode;  
505          uint16 button, flags;          uint16 button, flags;
506          uint32 ev_time;          uint32 ev_time;
507          uint32 tmpmods;          key_translation tr;
508            char str[256];
509            Status status;
510            unsigned int state;
511            Window wdummy;
512            int dummy;
513    
514          while (XCheckMaskEvent(display, ~0, &xevent))          while (XPending(display) > 0)
515          {          {
516                  ev_time = time(NULL);                  XNextEvent(display, &xevent);
517    
518                    if ((IC != NULL) && (XFilterEvent(&xevent, None) == True))
519                    {
520                            DEBUG_KBD(("Filtering event\n"));
521                            continue;
522                    }
523    
524                  flags = 0;                  flags = 0;
525    
526                  switch (xevent.type)                  switch (xevent.type)
527                  {                  {
528                          case KeyRelease:                          case ClientMessage:
529                                  flags = KBD_FLAG_DOWN | KBD_FLAG_UP;                                  /* the window manager told us to quit */
530                                  /* fall through */                                  if ((xevent.xclient.message_type == protocol_atom)
531                                        && (xevent.xclient.data.l[0] == kill_atom))
532                                            /* Quit */
533                                            return 0;
534                                    break;
535    
536                          case KeyPress:                          case KeyPress:
537                                  if( XkbTranslateKeyCode(xkb, xevent.xkey.keycode, xevent.xkey.state, &tmpmods, &keysym) == False )                                  if (IC != NULL)
538                                            /* Multi_key compatible version */
539                                    {
540                                            XmbLookupString(IC,
541                                                            (XKeyPressedEvent *) &
542                                                            xevent, str, sizeof(str), &keysym, &status);
543                                            if (!((status == XLookupKeySym) || (status == XLookupBoth)))
544                                            {
545                                                    error("XmbLookupString failed with status 0x%x\n",
546                                                          status);
547                                                    break;
548                                            }
549                                    }
550                                    else
551                                    {
552                                            /* Plain old XLookupString */
553                                            DEBUG_KBD(("\nNo input context, using XLookupString\n"));
554                                            XLookupString((XKeyEvent *) & xevent,
555                                                          str, sizeof(str), &keysym, NULL);
556                                    }
557    
558                                    DEBUG_KBD(("KeyPress for (keysym 0x%lx, %s)\n", keysym,
559                                               get_ksname(keysym)));
560    
561                                    ev_time = time(NULL);
562                                    if (handle_special_keys(keysym, xevent.xkey.state, ev_time, True))
563                                          break;                                          break;
                                 scancode = xkeymap_translate_key(keysym, xevent.xkey.keycode, &flags);  
564    
565                                  if (scancode == 0 )                                  tr = xkeymap_translate_key(keysym,
566                                                               xevent.xkey.keycode, xevent.xkey.state);
567    
568                                    if (tr.scancode == 0)
569                                          break;                                          break;
570    
571                                  /* keep track of the modifiers -- needed for stickykeys... */                                  ensure_remote_modifiers(ev_time, tr);
572                                  if( xevent.type == KeyPress )  
573                                          xwin_mod_press( xevent.xkey.state, ev_time, scancode );                                  rdp_send_scancode(ev_time, RDP_KEYPRESS, tr.scancode);
574                                    break;
575    
576                            case KeyRelease:
577                                    XLookupString((XKeyEvent *) & xevent, str,
578                                                  sizeof(str), &keysym, NULL);
579    
580                                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE, flags, scancode, 0);                                  DEBUG_KBD(("\nKeyRelease for (keysym 0x%lx, %s)\n", keysym,
581                                               get_ksname(keysym)));
582    
583                                    ev_time = time(NULL);
584                                    if (handle_special_keys(keysym, xevent.xkey.state, ev_time, False))
585                                            break;
586    
587                                  if( xevent.type == KeyRelease )                                  tr = xkeymap_translate_key(keysym,
588                                          xwin_mod_release( xevent.xkey.state, ev_time, scancode );                                                             xevent.xkey.keycode, xevent.xkey.state);
589    
590                                    if (tr.scancode == 0)
591                                            break;
592    
593                                    rdp_send_scancode(ev_time, RDP_KEYRELEASE, tr.scancode);
594                                  break;                                  break;
595    
596                          case ButtonPress:                          case ButtonPress:
# Line 480  xwin_process_events() Line 602  xwin_process_events()
602                                  if (button == 0)                                  if (button == 0)
603                                          break;                                          break;
604    
605                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
606                                                 flags | button,                                                 flags | button, xevent.xbutton.x, xevent.xbutton.y);
                                                xevent.xbutton.x,  
                                                xevent.xbutton.y);  
607                                  break;                                  break;
608    
609                          case MotionNotify:                          case MotionNotify:
610                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  if (fullscreen && !focused)
611                                                 MOUSE_FLAG_MOVE,                                          XSetInputFocus(display, wnd, RevertToPointerRoot,
612                                                 xevent.xmotion.x,                                                         CurrentTime);
613                                                 xevent.xmotion.y);                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
614                                                   MOUSE_FLAG_MOVE, xevent.xmotion.x, xevent.xmotion.y);
615                                  break;                                  break;
616    
617                          case EnterNotify:                          case FocusIn:
618                                  XGrabKeyboard(display, wnd, True, GrabModeAsync,                                  if (xevent.xfocus.mode == NotifyGrab)
619                                                GrabModeAsync, CurrentTime);                                          break;
620                                    focused = True;
621                                    XQueryPointer(display, wnd, &wdummy, &wdummy, &dummy, &dummy,
622                                                  &dummy, &dummy, &state);
623                                    reset_modifier_keys(state);
624                                    if (grab_keyboard && mouse_in_wnd)
625                                            XGrabKeyboard(display, wnd, True,
626                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
627                                    break;
628    
629                                   xwin_mod_update( xevent.xcrossing.state, ev_time );                          case FocusOut:
630                                    if (xevent.xfocus.mode == NotifyUngrab)
631                                            break;
632                                    focused = False;
633                                    if (xevent.xfocus.mode == NotifyWhileGrabbed)
634                                            XUngrabKeyboard(display, CurrentTime);
635                                    break;
636    
637                            case EnterNotify:
638                                    /* we only register for this event when in fullscreen mode */
639                                    /* or grab_keyboard */
640                                    mouse_in_wnd = True;
641                                    if (fullscreen)
642                                    {
643                                            XSetInputFocus(display, wnd, RevertToPointerRoot,
644                                                           CurrentTime);
645                                            break;
646                                    }
647                                    if (focused)
648                                            XGrabKeyboard(display, wnd, True,
649                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
650                                  break;                                  break;
651    
652                          case LeaveNotify:                          case LeaveNotify:
653                                    /* we only register for this event when grab_keyboard */
654                                    mouse_in_wnd = False;
655                                  XUngrabKeyboard(display, CurrentTime);                                  XUngrabKeyboard(display, CurrentTime);
656                                  break;                                  break;
657    
658                          case Expose:                          case Expose:
659                                  XCopyArea(display, backstore, wnd, gc,                                  XCopyArea(display, backstore, wnd, gc,
660                                            xevent.xexpose.x, xevent.xexpose.y,                                            xevent.xexpose.x, xevent.xexpose.y,
661                                            xevent.xexpose.width, xevent.xexpose.height,                                            xevent.xexpose.width,
662                                              xevent.xexpose.height,
663                                            xevent.xexpose.x, xevent.xexpose.y);                                            xevent.xexpose.x, xevent.xexpose.y);
664                                  break;                                  break;
                 }  
         }  
 }  
   
 void  
 xwin_mod_update(uint32 state, uint32 ev_time )  
 {  
         xwin_mod_press(state, ev_time, 0);  
         xwin_mod_release(state, ev_time, 0);  
 }  
665    
666  void                          case MappingNotify:
667  xwin_mod_release(uint32 state, uint32 ev_time, uint32 scancode)                                  /* Refresh keyboard mapping if it has changed. This is important for
668  {                                     Xvnc, since it allocates keycodes dynamically */
669          switch (scancode) {                                  if (xevent.xmapping.request == MappingKeyboard
670          case 0x2a:                                      || xevent.xmapping.request == MappingModifier)
671                  key_down_state &= ~DShift1Mask;                                          XRefreshKeyboardMapping(&xevent.xmapping);
                 break;  
         case 0x36:  
                 key_down_state &= ~DShift2Mask;  
                 break;  
         case 0x1d:  
                 key_down_state &= ~DControl1Mask;  
                 break;  
         case 0x9d:  
                 key_down_state &= ~DControl2Mask;  
                 break;  
         case 0x38:  
                 key_down_state &= ~DMod1Mask;  
                 break;  
         case 0xb8:  
                 key_down_state &= ~DMod2Mask;  
                 break;  
         }  
   
         if( !(numlock_modifier_mask & state) && (key_down_state & DNumLockMask) )  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, 0, 0x45, 0);  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN | KBD_FLAG_UP, 0x45, 0);  
                 key_down_state &= ~DNumLockMask;  
         }  
   
         if( !(LockMask & state) && (key_down_state & DLockMask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, 0, 0x3a, 0);  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN | KBD_FLAG_UP, 0x3a, 0);  
                 key_down_state &= ~DLockMask;  
   
         }  
672    
673                                    if (xevent.xmapping.request == MappingModifier)
674                                    {
675                                            XFreeModifiermap(mod_map);
676                                            mod_map = XGetModifierMapping(display);
677                                    }
678                                    break;
679    
680          if( !(ShiftMask & state) && (key_down_state & DShift1Mask))                  }
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0x2a, 0);  
                 key_down_state &= ~DShift1Mask;  
   
         }  
   
         if( !(ControlMask & state) && (key_down_state & DControl1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0x1d, 0);  
                 key_down_state &= ~DControl1Mask;  
   
         }  
   
         if( !(Mod1Mask & state) && (key_down_state & DMod1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0x38, 0);  
                 key_down_state &= ~DMod1Mask;  
   
         }  
   
         if( !(Mod2Mask & state) && (key_down_state & DMod2Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0xb8, 0);  
                 key_down_state &= ~DMod2Mask;  
         }  
 }  
   
   
 void  
 xwin_mod_press(uint32 state, uint32 ev_time, uint32 scancode)  
 {  
   
         switch (scancode) {  
         case 0x2a:  
                 key_down_state |= DShift1Mask;  
                 break;  
         case 0x36:  
                 key_down_state |= DShift2Mask;  
                 break;  
         case 0x1d:  
                 key_down_state |= DControl1Mask;  
                 break;  
         case 0x9d:  
                 key_down_state |= DControl2Mask;  
                 break;  
         case 0x3a:  
                 key_down_state ^= DLockMask;  
                 break;  
         case 0x45:  
                 key_down_state ^= DNumLockMask;  
                 break;  
         case 0x38:  
                 key_down_state |= DMod1Mask;  
                 break;  
         case 0xb8:  
                 key_down_state |= DMod2Mask;  
                 break;  
         }  
   
         if( (numlock_modifier_mask && state) && !(key_down_state & DNumLockMask) )  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, 0, 0x45, 0);  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN | KBD_FLAG_UP, 0x45, 0);  
                 key_down_state |= DNumLockMask;  
         }  
   
         if( (LockMask & state) && !(key_down_state & DLockMask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, 0, 0x3a, 0);  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN | KBD_FLAG_UP, 0x3a, 0);  
                 key_down_state |= DLockMask;  
   
         }  
   
   
         if( (ShiftMask & state) && !((key_down_state & DShift1Mask) || (key_down_state & DShift2Mask)))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0x2a, 0);  
                 key_down_state |= DShift1Mask;  
   
         }  
   
         if( (ControlMask & state) && !((key_down_state & DControl1Mask) || (key_down_state & DControl2Mask)))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0x1d, 0);  
                 key_down_state |= DControl1Mask;  
   
         }  
   
         if( (Mod1Mask & state) && !(key_down_state & DMod1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0x38, 0);  
                 key_down_state |= DMod1Mask;  
   
         }  
   
         if( (Mod2Mask & state) && !(key_down_state & DMod2Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0xb8, 0);  
                 key_down_state |= DMod2Mask;  
   
681          }          }
682            /* Keep going */
683            return 1;
684  }  }
685    
686  void  /* Returns 0 after user quit, 1 otherwise */
687    int
688  ui_select(int rdp_socket)  ui_select(int rdp_socket)
689  {  {
690          int n = (rdp_socket > x_socket) ? rdp_socket+1 : x_socket+1;          int n = (rdp_socket > x_socket) ? rdp_socket + 1 : x_socket + 1;
691          fd_set rfds;          fd_set rfds;
692    
693          FD_ZERO(&rfds);          FD_ZERO(&rfds);
694    
695          while (True)          while (True)
696          {          {
697                    /* Process any events already waiting */
698                    if (!xwin_process_events())
699                            /* User quit */
700                            return 0;
701    
702                  FD_ZERO(&rfds);                  FD_ZERO(&rfds);
703                  FD_SET(rdp_socket, &rfds);                  FD_SET(rdp_socket, &rfds);
704                  if (display != NULL)                  FD_SET(x_socket, &rfds);
                 {  
                         FD_SET(x_socket, &rfds);  
                         XFlush(display);  
                 }  
705    
706                  switch (select(n, &rfds, NULL, NULL, NULL))                  switch (select(n, &rfds, NULL, NULL, NULL))
707                  {                  {
# Line 693  ui_select(int rdp_socket) Line 712  ui_select(int rdp_socket)
712                                  continue;                                  continue;
713                  }                  }
714    
                 if (FD_ISSET(x_socket, &rfds))  
                         xwin_process_events();  
   
715                  if (FD_ISSET(rdp_socket, &rfds))                  if (FD_ISSET(rdp_socket, &rfds))
716                          return;                          return 1;
717          }          }
718  }  }
719    
# Line 708  ui_move_pointer(int x, int y) Line 724  ui_move_pointer(int x, int y)
724  }  }
725    
726  HBITMAP  HBITMAP
727  ui_create_bitmap(int width, int height, uint8 *data)  ui_create_bitmap(int width, int height, uint8 * data)
728  {  {
729          XImage *image;          XImage *image;
730          Pixmap bitmap;          Pixmap bitmap;
# Line 716  ui_create_bitmap(int width, int height, Line 732  ui_create_bitmap(int width, int height,
732    
733          tdata = (owncolmap ? data : translate_image(width, height, data));          tdata = (owncolmap ? data : translate_image(width, height, data));
734          bitmap = XCreatePixmap(display, wnd, width, height, depth);          bitmap = XCreatePixmap(display, wnd, width, height, depth);
735          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
736                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
737    
738          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);
739    
# Line 728  ui_create_bitmap(int width, int height, Line 744  ui_create_bitmap(int width, int height,
744  }  }
745    
746  void  void
747  ui_paint_bitmap(int x, int y, int cx, int cy,  ui_paint_bitmap(int x, int y, int cx, int cy, int width, int height, uint8 * data)
                 int width, int height, uint8 *data)  
748  {  {
749          XImage *image;          XImage *image;
750          uint8 *tdata;          uint8 *tdata;
751    
752          tdata = (owncolmap ? data : translate_image(width, height, data));          tdata = (owncolmap ? data : translate_image(width, height, data));
753          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
754                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
755    
756          if (ownbackstore)          if (ownbackstore)
757          {          {
# Line 756  ui_paint_bitmap(int x, int y, int cx, in Line 771  ui_paint_bitmap(int x, int y, int cx, in
771  void  void
772  ui_destroy_bitmap(HBITMAP bmp)  ui_destroy_bitmap(HBITMAP bmp)
773  {  {
774          XFreePixmap(display, (Pixmap)bmp);          XFreePixmap(display, (Pixmap) bmp);
775  }  }
776    
777  HGLYPH  HGLYPH
778  ui_create_glyph(int width, int height, uint8 *data)  ui_create_glyph(int width, int height, uint8 * data)
779  {  {
780          XImage *image;          XImage *image;
781          Pixmap bitmap;          Pixmap bitmap;
# Line 772  ui_create_glyph(int width, int height, u Line 787  ui_create_glyph(int width, int height, u
787          bitmap = XCreatePixmap(display, wnd, width, height, 1);          bitmap = XCreatePixmap(display, wnd, width, height, 1);
788          gc = XCreateGC(display, bitmap, 0, NULL);          gc = XCreateGC(display, bitmap, 0, NULL);
789    
790          image = XCreateImage(display, visual, 1, ZPixmap, 0,          image = XCreateImage(display, visual, 1, ZPixmap, 0, (char *) data,
791                               data, width, height, 8, scanline);                               width, height, 8, scanline);
792          image->byte_order = MSBFirst;          image->byte_order = MSBFirst;
793          image->bitmap_bit_order = MSBFirst;          image->bitmap_bit_order = MSBFirst;
794          XInitImage(image);          XInitImage(image);
# Line 782  ui_create_glyph(int width, int height, u Line 797  ui_create_glyph(int width, int height, u
797    
798          XFree(image);          XFree(image);
799          XFreeGC(display, gc);          XFreeGC(display, gc);
800          return (HGLYPH)bitmap;          return (HGLYPH) bitmap;
801  }  }
802    
803  void  void
804  ui_destroy_glyph(HGLYPH glyph)  ui_destroy_glyph(HGLYPH glyph)
805  {  {
806          XFreePixmap(display, (Pixmap)glyph);          XFreePixmap(display, (Pixmap) glyph);
807  }  }
808    
809  HCURSOR  HCURSOR
810  ui_create_cursor(unsigned int x, unsigned int y, int width,  ui_create_cursor(unsigned int x, unsigned int y, int width, int height,
811                   int height, uint8 *andmask, uint8 *xormask)                   uint8 * andmask, uint8 * xormask)
812  {  {
813          HGLYPH maskglyph, cursorglyph;          HGLYPH maskglyph, cursorglyph;
814          XColor bg, fg;          XColor bg, fg;
# Line 850  ui_create_cursor(unsigned int x, unsigne Line 865  ui_create_cursor(unsigned int x, unsigne
865    
866          cursorglyph = ui_create_glyph(width, height, cursor);          cursorglyph = ui_create_glyph(width, height, cursor);
867          maskglyph = ui_create_glyph(width, height, mask);          maskglyph = ui_create_glyph(width, height, mask);
868            
869          xcursor = XCreatePixmapCursor(display, (Pixmap)cursorglyph,          xcursor =
870                                  (Pixmap)maskglyph, &fg, &bg, x, y);                  XCreatePixmapCursor(display, (Pixmap) cursorglyph,
871                                        (Pixmap) maskglyph, &fg, &bg, x, y);
872    
873          ui_destroy_glyph(maskglyph);          ui_destroy_glyph(maskglyph);
874          ui_destroy_glyph(cursorglyph);          ui_destroy_glyph(cursorglyph);
875          xfree(mask);          xfree(mask);
876          xfree(cursor);          xfree(cursor);
877          return (HCURSOR)xcursor;          return (HCURSOR) xcursor;
878  }  }
879    
880  void  void
881  ui_set_cursor(HCURSOR cursor)  ui_set_cursor(HCURSOR cursor)
882  {  {
883          XDefineCursor(display, wnd, (Cursor)cursor);          current_cursor = (Cursor) cursor;
884            XDefineCursor(display, wnd, current_cursor);
885  }  }
886    
887  void  void
888  ui_destroy_cursor(HCURSOR cursor)  ui_destroy_cursor(HCURSOR cursor)
889  {  {
890          XFreeCursor(display, (Cursor)cursor);          XFreeCursor(display, (Cursor) cursor);
891  }  }
892    
893  #define MAKE_XCOLOR(xc,c) \  #define MAKE_XCOLOR(xc,c) \
# Line 879  ui_destroy_cursor(HCURSOR cursor) Line 896  ui_destroy_cursor(HCURSOR cursor)
896                  (xc)->blue  = ((c)->blue  << 8) | (c)->blue; \                  (xc)->blue  = ((c)->blue  << 8) | (c)->blue; \
897                  (xc)->flags = DoRed | DoGreen | DoBlue;                  (xc)->flags = DoRed | DoGreen | DoBlue;
898    
899    
900  HCOLOURMAP  HCOLOURMAP
901  ui_create_colourmap(COLOURMAP *colours)  ui_create_colourmap(COLOURMAP * colours)
902  {  {
903          COLOURENTRY *entry;          COLOURENTRY *entry;
904          int i, ncolours = colours->ncolours;          int i, ncolours = colours->ncolours;
905            if (!owncolmap)
906            {
907                    uint32 *map = xmalloc(sizeof(*colmap) * ncolours);
908                    XColor xentry;
909                    XColor xc_cache[256];
910                    uint32 colour;
911                    int colLookup = 256;
912                    for (i = 0; i < ncolours; i++)
913                    {
914                            entry = &colours->colours[i];
915                            MAKE_XCOLOR(&xentry, entry);
916    
917                            if (XAllocColor(display, xcolmap, &xentry) == 0)
918                            {
919                                    /* Allocation failed, find closest match. */
920                                    int j = 256;
921                                    int nMinDist = 3 * 256 * 256;
922                                    long nDist = nMinDist;
923    
924                                    /* only get the colors once */
925                                    while (colLookup--)
926                                    {
927                                            xc_cache[colLookup].pixel = colLookup;
928                                            xc_cache[colLookup].red = xc_cache[colLookup].green =
929                                                    xc_cache[colLookup].blue = 0;
930                                            xc_cache[colLookup].flags = 0;
931                                            XQueryColor(display,
932                                                        DefaultColormap(display,
933                                                                        DefaultScreen(display)),
934                                                        &xc_cache[colLookup]);
935                                    }
936                                    colLookup = 0;
937    
938                                    /* approximate the pixel */
939                                    while (j--)
940                                    {
941                                            if (xc_cache[j].flags)
942                                            {
943                                                    nDist = ((long) (xc_cache[j].red >> 8) -
944                                                             (long) (xentry.red >> 8)) *
945                                                            ((long) (xc_cache[j].red >> 8) -
946                                                             (long) (xentry.red >> 8)) +
947                                                            ((long) (xc_cache[j].green >> 8) -
948                                                             (long) (xentry.green >> 8)) *
949                                                            ((long) (xc_cache[j].green >> 8) -
950                                                             (long) (xentry.green >> 8)) +
951                                                            ((long) (xc_cache[j].blue >> 8) -
952                                                             (long) (xentry.blue >> 8)) *
953                                                            ((long) (xc_cache[j].blue >> 8) -
954                                                             (long) (xentry.blue >> 8));
955                                            }
956                                            if (nDist < nMinDist)
957                                            {
958                                                    nMinDist = nDist;
959                                                    xentry.pixel = j;
960                                            }
961                                    }
962                            }
963                            colour = xentry.pixel;
964    
965                            /* update our cache */
966                            if (xentry.pixel < 256)
967                            {
968                                    xc_cache[xentry.pixel].red = xentry.red;
969                                    xc_cache[xentry.pixel].green = xentry.green;
970                                    xc_cache[xentry.pixel].blue = xentry.blue;
971    
972                            }
973    
974          if (owncolmap)  
975                            /* byte swap here to make translate_image faster */
976                            map[i] = translate_colour(colour);
977                    }
978                    return map;
979            }
980            else
981          {          {
982                  XColor *xcolours, *xentry;                  XColor *xcolours, *xentry;
983                  Colormap map;                  Colormap map;
# Line 903  ui_create_colourmap(COLOURMAP *colours) Line 995  ui_create_colourmap(COLOURMAP *colours)
995                  XStoreColors(display, map, xcolours, ncolours);                  XStoreColors(display, map, xcolours, ncolours);
996    
997                  xfree(xcolours);                  xfree(xcolours);
998                  return (HCOLOURMAP)map;                  return (HCOLOURMAP) map;
         }  
         else  
         {  
                 uint32 *map = xmalloc(sizeof(*colmap) * ncolours);  
                 XColor xentry;  
                 uint32 colour;  
   
                 for (i = 0; i < ncolours; i++)  
                 {  
                         entry = &colours->colours[i];  
                         MAKE_XCOLOR(&xentry, entry);  
   
                         if (XAllocColor(display, xcolmap, &xentry) != 0)  
                                 colour = xentry.pixel;  
                         else  
                                 colour = white;  
   
                         /* byte swap here to make translate_image faster */  
                         map[i] = translate_colour(colour);  
                 }  
   
                 return map;  
999          }          }
1000  }  }
1001    
1002  void  void
1003  ui_destroy_colourmap(HCOLOURMAP map)  ui_destroy_colourmap(HCOLOURMAP map)
1004  {  {
1005          if (owncolmap)          if (!owncolmap)
                 XFreeColormap(display, (Colormap)map);  
         else  
1006                  xfree(map);                  xfree(map);
1007            else
1008                    XFreeColormap(display, (Colormap) map);
1009  }  }
1010    
1011  void  void
1012  ui_set_colourmap(HCOLOURMAP map)  ui_set_colourmap(HCOLOURMAP map)
1013  {  {
1014          if (owncolmap)          if (!owncolmap)
                 XSetWindowColormap(display, wnd, (Colormap)map);  
         else  
1015                  colmap = map;                  colmap = map;
1016            else
1017                    XSetWindowColormap(display, wnd, (Colormap) map);
1018  }  }
1019    
1020  void  void
# Line 960  ui_set_clip(int x, int y, int cx, int cy Line 1030  ui_set_clip(int x, int y, int cx, int cy
1030  }  }
1031    
1032  void  void
1033  ui_reset_clip()  ui_reset_clip(void)
1034  {  {
1035          XRectangle rect;          XRectangle rect;
1036    
# Line 972  ui_reset_clip() Line 1042  ui_reset_clip()
1042  }  }
1043    
1044  void  void
1045  ui_bell()  ui_bell(void)
1046  {  {
1047          XBell(display, 0);          XBell(display, 0);
1048  }  }
# Line 989  ui_destblt(uint8 opcode, Line 1059  ui_destblt(uint8 opcode,
1059  void  void
1060  ui_patblt(uint8 opcode,  ui_patblt(uint8 opcode,
1061            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
1062            /* brush */ BRUSH *brush, int bgcolour, int fgcolour)            /* brush */ BRUSH * brush, int bgcolour, int fgcolour)
1063  {  {
1064          Pixmap fill;          Pixmap fill;
1065          uint8 i, ipattern[8];          uint8 i, ipattern[8];
# Line 1017  ui_patblt(uint8 opcode, Line 1087  ui_patblt(uint8 opcode,
1087                          FILL_RECTANGLE(x, y, cx, cy);                          FILL_RECTANGLE(x, y, cx, cy);
1088    
1089                          XSetFillStyle(display, gc, FillSolid);                          XSetFillStyle(display, gc, FillSolid);
1090                          ui_destroy_glyph((HGLYPH)fill);                          XSetTSOrigin(display, gc, 0, 0);
1091                            ui_destroy_glyph((HGLYPH) fill);
1092                          break;                          break;
1093    
1094                  default:                  default:
# Line 1035  ui_screenblt(uint8 opcode, Line 1106  ui_screenblt(uint8 opcode,
1106          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1107          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);
1108          if (ownbackstore)          if (ownbackstore)
1109                  XCopyArea(display, backstore, backstore, gc, srcx, srcy,                  XCopyArea(display, backstore, backstore, gc, srcx, srcy, cx, cy, x, y);
                           cx, cy, x, y);  
1110          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1111  }  }
1112    
# Line 1046  ui_memblt(uint8 opcode, Line 1116  ui_memblt(uint8 opcode,
1116            /* src */ HBITMAP src, int srcx, int srcy)            /* src */ HBITMAP src, int srcx, int srcy)
1117  {  {
1118          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1119          XCopyArea(display, (Pixmap)src, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, (Pixmap) src, wnd, gc, srcx, srcy, cx, cy, x, y);
1120          if (ownbackstore)          if (ownbackstore)
1121                  XCopyArea(display, (Pixmap)src, backstore, gc, srcx, srcy,                  XCopyArea(display, (Pixmap) src, backstore, gc, srcx, srcy, cx, cy, x, y);
                           cx, cy, x, y);  
1122          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1123  }  }
1124    
# Line 1057  void Line 1126  void
1126  ui_triblt(uint8 opcode,  ui_triblt(uint8 opcode,
1127            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
1128            /* src */ HBITMAP src, int srcx, int srcy,            /* src */ HBITMAP src, int srcx, int srcy,
1129            /* brush */ BRUSH *brush, int bgcolour, int fgcolour)            /* brush */ BRUSH * brush, int bgcolour, int fgcolour)
1130  {  {
1131          /* This is potentially difficult to do in general. Until someone          /* This is potentially difficult to do in general. Until someone
1132             comes up with a more efficient way of doing it I am using cases. */             comes up with a more efficient way of doing it I am using cases. */
# Line 1066  ui_triblt(uint8 opcode, Line 1135  ui_triblt(uint8 opcode,
1135          {          {
1136                  case 0x69:      /* PDSxxn */                  case 0x69:      /* PDSxxn */
1137                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);
1138                          ui_patblt(ROP2_NXOR, x, y, cx, cy,                          ui_patblt(ROP2_NXOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1139                          break;                          break;
1140    
1141                  case 0xb8:      /* PSDPxax */                  case 0xb8:      /* PSDPxax */
1142                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1143                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);
1144                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1145                          break;                          break;
1146    
1147                  case 0xc0:      /* PSa */                  case 0xc0:      /* PSa */
1148                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);
1149                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour,                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   fgcolour);  
1150                          break;                          break;
1151    
1152                  default:                  default:
# Line 1093  ui_triblt(uint8 opcode, Line 1158  ui_triblt(uint8 opcode,
1158  void  void
1159  ui_line(uint8 opcode,  ui_line(uint8 opcode,
1160          /* dest */ int startx, int starty, int endx, int endy,          /* dest */ int startx, int starty, int endx, int endy,
1161          /* pen */ PEN *pen)          /* pen */ PEN * pen)
1162  {  {
1163          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1164          SET_FOREGROUND(pen->colour);          SET_FOREGROUND(pen->colour);
# Line 1112  ui_rect( Line 1177  ui_rect(
1177          FILL_RECTANGLE(x, y, cx, cy);          FILL_RECTANGLE(x, y, cx, cy);
1178  }  }
1179    
1180    /* warning, this function only draws on wnd or backstore, not both */
1181  void  void
1182  ui_draw_glyph(int mixmode,  ui_draw_glyph(int mixmode,
1183                /* dest */ int x, int y, int cx, int cy,                /* dest */ int x, int y, int cx, int cy,
1184                /* src */ HGLYPH glyph, int srcx, int srcy, int bgcolour,                /* src */ HGLYPH glyph, int srcx, int srcy,
1185                int fgcolour)                int bgcolour, int fgcolour)
1186  {  {
1187          SET_FOREGROUND(fgcolour);          SET_FOREGROUND(fgcolour);
1188          SET_BACKGROUND(bgcolour);          SET_BACKGROUND(bgcolour);
1189    
1190          XSetFillStyle(display, gc, (mixmode == MIX_TRANSPARENT)          XSetFillStyle(display, gc,
1191                        ? FillStippled : FillOpaqueStippled);                        (mixmode == MIX_TRANSPARENT) ? FillStippled : FillOpaqueStippled);
1192          XSetStipple(display, gc, (Pixmap)glyph);          XSetStipple(display, gc, (Pixmap) glyph);
1193          XSetTSOrigin(display, gc, x, y);          XSetTSOrigin(display, gc, x, y);
1194    
1195          FILL_RECTANGLE(x, y, cx, cy);          FILL_RECTANGLE_BACKSTORE(x, y, cx, cy);
1196    
1197          XSetFillStyle(display, gc, FillSolid);          XSetFillStyle(display, gc, FillSolid);
1198  }  }
# Line 1140  ui_draw_glyph(int mixmode, Line 1206  ui_draw_glyph(int mixmode,
1206        if ((xyoffset & 0x80))\        if ((xyoffset & 0x80))\
1207          {\          {\
1208            if (flags & TEXT2_VERTICAL) \            if (flags & TEXT2_VERTICAL) \
1209              y += ttext[++idx] | (ttext[++idx] << 8);\              y += ttext[idx+1] | (ttext[idx+2] << 8);\
1210            else\            else\
1211              x += ttext[++idx] | (ttext[++idx] << 8);\              x += ttext[idx+1] | (ttext[idx+2] << 8);\
1212              idx += 2;\
1213          }\          }\
1214        else\        else\
1215          {\          {\
# Line 1154  ui_draw_glyph(int mixmode, Line 1221  ui_draw_glyph(int mixmode,
1221      }\      }\
1222    if (glyph != NULL)\    if (glyph != NULL)\
1223      {\      {\
1224        ui_draw_glyph (mixmode, x + (short) glyph->offset,\        ui_draw_glyph (mixmode, x + glyph->offset,\
1225                       y + (short) glyph->baseline,\                       y + glyph->baseline,\
1226                       glyph->width, glyph->height,\                       glyph->width, glyph->height,\
1227                       glyph->pixmap, 0, 0, bgcolour, fgcolour);\                       glyph->pixmap, 0, 0, bgcolour, fgcolour);\
1228        if (flags & TEXT2_IMPLICIT_X)\        if (flags & TEXT2_IMPLICIT_X)\
# Line 1165  ui_draw_glyph(int mixmode, Line 1232  ui_draw_glyph(int mixmode,
1232    
1233  void  void
1234  ui_draw_text(uint8 font, uint8 flags, int mixmode, int x, int y,  ui_draw_text(uint8 font, uint8 flags, int mixmode, int x, int y,
1235               int clipx, int clipy, int clipcx, int clipcy, int boxx,               int clipx, int clipy, int clipcx, int clipcy,
1236               int boxy, int boxcx, int boxcy, int bgcolour,               int boxx, int boxy, int boxcx, int boxcy, int bgcolour,
1237               int fgcolour, uint8 * text, uint8 length)               int fgcolour, uint8 * text, uint8 length)
1238  {  {
1239          FONTGLYPH *glyph;          FONTGLYPH *glyph;
# Line 1177  ui_draw_text(uint8 font, uint8 flags, in Line 1244  ui_draw_text(uint8 font, uint8 flags, in
1244    
1245          if (boxcx > 1)          if (boxcx > 1)
1246          {          {
1247                  FILL_RECTANGLE(boxx, boxy, boxcx, boxcy);                  FILL_RECTANGLE_BACKSTORE(boxx, boxy, boxcx, boxcy);
1248          }          }
1249          else if (mixmode == MIX_OPAQUE)          else if (mixmode == MIX_OPAQUE)
1250          {          {
1251                  FILL_RECTANGLE(clipx, clipy, clipcx, clipcy);                  FILL_RECTANGLE_BACKSTORE(clipx, clipy, clipcx, clipcy);
1252          }          }
1253    
1254          /* Paint text, character by character */          /* Paint text, character by character */
1255          for (i = 0; i < length;) {          for (i = 0; i < length;)
1256                  switch (text[i]) {          {
1257                  case 0xff:                  switch (text[i])
1258                          if (i + 2 < length)                  {
1259                                  cache_put_text(text[i + 1], text, text[i + 2]);                          case 0xff:
1260                          else {                                  if (i + 2 < length)
1261                                  error("this shouldn't be happening\n");                                          cache_put_text(text[i + 1], text, text[i + 2]);
1262                                    else
1263                                    {
1264                                            error("this shouldn't be happening\n");
1265                                            exit(1);
1266                                    }
1267                                    /* this will move pointer from start to first character after FF command */
1268                                    length -= i + 3;
1269                                    text = &(text[i + 3]);
1270                                    i = 0;
1271                                  break;                                  break;
                         }  
                         /* this will move pointer from start to first character after FF command */  
                         length -= i + 3;  
                         text = &(text[i + 3]);  
                         i = 0;  
                         break;  
1272    
1273                  case 0xfe:                          case 0xfe:
1274                          entry = cache_get_text(text[i + 1]);                                  entry = cache_get_text(text[i + 1]);
1275                          if (entry != NULL) {                                  if (entry != NULL)
1276                                  if ((((uint8 *) (entry->data))[1] == 0)                                  {
1277                                      && (!(flags & TEXT2_IMPLICIT_X))) {                                          if ((((uint8 *) (entry->data))[1] ==
1278                                          if (flags & TEXT2_VERTICAL)                                               0) && (!(flags & TEXT2_IMPLICIT_X)))
1279                                                  y += text[i + 2];                                          {
1280                                          else                                                  if (flags & TEXT2_VERTICAL)
1281                                                  x += text[i + 2];                                                          y += text[i + 2];
1282                                                    else
1283                                                            x += text[i + 2];
1284                                            }
1285                                            for (j = 0; j < entry->size; j++)
1286                                                    DO_GLYPH(((uint8 *) (entry->data)), j);
1287                                  }                                  }
1288                                  if (i + 2 < length)                                  if (i + 2 < length)
1289                                          i += 3;                                          i += 3;
1290                                  else                                  else
1291                                          i += 2;                                          i += 2;
1292                                  length -= i;                                  length -= i;
1293                                  /* this will move pointer from start to first character after FE command */                                  /* this will move pointer from start to first character after FE command */
1294                                  text = &(text[i]);                                  text = &(text[i]);
1295                                  i = 0;                                  i = 0;
1296                                  for (j = 0; j < entry->size; j++)                                  break;
                                         DO_GLYPH(((uint8 *) (entry->data)), j);  
                         }  
                         break;  
1297    
1298                  default:                          default:
1299                          DO_GLYPH(text, i);                                  DO_GLYPH(text, i);
1300                          i++;                                  i++;
1301                          break;                                  break;
1302                  }                  }
1303          }          }
1304            if (ownbackstore)
1305            {
1306                    if (boxcx > 1)
1307                            XCopyArea(display, backstore, wnd, gc, boxx,
1308                                      boxy, boxcx, boxcy, boxx, boxy);
1309                    else
1310                            XCopyArea(display, backstore, wnd, gc, clipx,
1311                                      clipy, clipcx, clipcy, clipx, clipy);
1312            }
1313  }  }
1314    
1315  void  void
# Line 1241  ui_desktop_save(uint32 offset, int x, in Line 1320  ui_desktop_save(uint32 offset, int x, in
1320    
1321          if (ownbackstore)          if (ownbackstore)
1322          {          {
1323                  image = XGetImage(display, backstore, x, y, cx, cy, AllPlanes,                  image = XGetImage(display, backstore, x, y, cx, cy, AllPlanes, ZPixmap);
                                   ZPixmap);  
1324          }          }
1325          else          else
1326          {          {
1327                  pix = XCreatePixmap(display, wnd, cx, cy, depth);                  pix = XCreatePixmap(display, wnd, cx, cy, depth);
1328                  XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);                  XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);
1329                  image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes,                  image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes, ZPixmap);
                                   ZPixmap);  
1330                  XFreePixmap(display, pix);                  XFreePixmap(display, pix);
1331          }          }
1332    
1333          offset *= bpp/8;          offset *= bpp / 8;
1334          cache_put_desktop(offset, cx, cy, image->bytes_per_line,          cache_put_desktop(offset, cx, cy, image->bytes_per_line, bpp / 8, (uint8 *) image->data);
                           bpp/8, (uint8 *)image->data);  
1335    
1336          XDestroyImage(image);          XDestroyImage(image);
1337  }  }
# Line 1266  ui_desktop_restore(uint32 offset, int x, Line 1342  ui_desktop_restore(uint32 offset, int x,
1342          XImage *image;          XImage *image;
1343          uint8 *data;          uint8 *data;
1344    
1345          offset *= bpp/8;          offset *= bpp / 8;
1346          data = cache_get_desktop(offset, cx, cy, bpp/8);          data = cache_get_desktop(offset, cx, cy, bpp / 8);
1347          if (data == NULL)          if (data == NULL)
1348                  return;                  return;
1349    
1350          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
1351                               0, data, cx, cy, BitmapPad(display),                               (char *) data, cx, cy, BitmapPad(display), cx * bpp / 8);
                              cx * bpp/8);  
1352    
1353          if (ownbackstore)          if (ownbackstore)
1354          {          {

Legend:
Removed from v.59  
changed lines
  Added in v.300

  ViewVC Help
Powered by ViewVC 1.1.26