/[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 52 by n-ki, Fri Apr 26 08:22:39 2002 UTC revision 279 by n-ki, Tue Nov 26 10:09:14 2002 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 key_modifier_state = 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  #define DShift1Mask   (1<<0)  {
65  #define DShift2Mask   (1<<1)          unsigned long flags;
66  #define DControl1Mask (1<<2)          unsigned long functions;
67  #define DControl2Mask (1<<3)          unsigned long decorations;
68  #define DMod1Mask     (1<<4)          long inputMode;
69  #define DMod2Mask     (1<<5)          unsigned long status;
70    }
71    PropMotifWmHints;
72    
73    
74  #define FILL_RECTANGLE(x,y,cx,cy)\  #define FILL_RECTANGLE(x,y,cx,cy)\
75  { \  { \
# Line 68  static unsigned int key_down_state = 0; Line 79  static unsigned int key_down_state = 0;
79  }  }
80    
81  /* colour maps */  /* colour maps */
82  static BOOL owncolmap;  BOOL owncolmap = False;
83  static Colormap xcolmap;  static Colormap xcolmap;
 static uint32 white;  
84  static uint32 *colmap;  static uint32 *colmap;
85    
86  #define TRANSLATE(col)          ( owncolmap ? col : translate_colour(colmap[col]) )  #define TRANSLATE(col)          ( owncolmap ? col : translate_colour(colmap[col]) )
# Line 99  static int rop2_map[] = { Line 109  static int rop2_map[] = {
109  #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]); }
110  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }
111    
112  void xwin_release_modifiers(XKeyEvent* ev, uint32 ev_time, uint32 scancode);  void
113  void xwin_press_modifiers(XKeyEvent* ev, uint32 ev_time, uint32 scancode);  mwm_hide_decorations(void)
114    {
115            PropMotifWmHints motif_hints;
116            Atom hintsatom;
117    
118            /* setup the property */
119            motif_hints.flags = MWM_HINTS_DECORATIONS;
120            motif_hints.decorations = 0;
121    
122            /* get the atom for the property */
123            hintsatom = XInternAtom(display, "_MOTIF_WM_HINTS", False);
124            if (!hintsatom)
125            {
126                    error("Failed to get atom _MOTIF_WM_HINTS\n");
127                    return;
128            }
129    
130            XChangeProperty(display, wnd, hintsatom, hintsatom, 32, PropModeReplace,
131                            (unsigned char *) &motif_hints, PROP_MOTIF_WM_HINTS_ELEMENTS);
132    }
133    
134  static void  static void
135  translate8(uint8 *data, uint8 *out, uint8 *end)  translate8(uint8 * data, uint8 * out, uint8 * end)
136  {  {
137          while (out < end)          while (out < end)
138                  *(out++) = (uint8)colmap[*(data++)];                  *(out++) = (uint8) colmap[*(data++)];
139  }  }
140    
141  static void  static void
142  translate16(uint8 *data, uint16 *out, uint16 *end)  translate16(uint8 * data, uint16 * out, uint16 * end)
143  {  {
144          while (out < end)          while (out < end)
145                  *(out++) = (uint16)colmap[*(data++)];                  *(out++) = (uint16) colmap[*(data++)];
146  }  }
147    
148  /* little endian - conversion happens when colourmap is built */  /* little endian - conversion happens when colourmap is built */
149  static void  static void
150  translate24(uint8 *data, uint8 *out, uint8 *end)  translate24(uint8 * data, uint8 * out, uint8 * end)
151  {  {
152          uint32 value;          uint32 value;
153    
# Line 132  translate24(uint8 *data, uint8 *out, uin Line 161  translate24(uint8 *data, uint8 *out, uin
161  }  }
162    
163  static void  static void
164  translate32(uint8 *data, uint32 *out, uint32 *end)  translate32(uint8 * data, uint32 * out, uint32 * end)
165  {  {
166          while (out < end)          while (out < end)
167                  *(out++) = colmap[*(data++)];                  *(out++) = colmap[*(data++)];
168  }  }
169    
170  static uint8 *  static uint8 *
171  translate_image(int width, int height, uint8 *data)  translate_image(int width, int height, uint8 * data)
172  {  {
173          int size = width * height * bpp/8;          int size = width * height * bpp / 8;
174          uint8 *out = xmalloc(size);          uint8 *out = xmalloc(size);
175          uint8 *end = out + size;          uint8 *end = out + size;
176    
# Line 152  translate_image(int width, int height, u Line 181  translate_image(int width, int height, u
181                          break;                          break;
182    
183                  case 16:                  case 16:
184                          translate16(data, (uint16 *)out, (uint16 *)end);                          translate16(data, (uint16 *) out, (uint16 *) end);
185                          break;                          break;
186    
187                  case 24:                  case 24:
# Line 160  translate_image(int width, int height, u Line 189  translate_image(int width, int height, u
189                          break;                          break;
190    
191                  case 32:                  case 32:
192                          translate32(data, (uint32 *)out, (uint32 *)end);                          translate32(data, (uint32 *) out, (uint32 *) end);
193                          break;                          break;
194          }          }
195    
# Line 197  translate_colour(uint32 colour) Line 226  translate_colour(uint32 colour)
226  }  }
227    
228  BOOL  BOOL
229  ui_create_window(char *title)  get_key_state(unsigned int state, uint32 keysym)
230  {  {
231          XSetWindowAttributes attribs;          int modifierpos, key, keysymMask = 0;
232          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;  
         }  
233    
234            KeyCode keycode = XKeysymToKeycode(display, keysym);
235    
236          /* XKB is the 'new' keyboard handler in x.. ( the xkb code in Xfree86 originates from SGI, years 1993 and 1995 from what I could tell. )          if (keycode == NoSymbol)
237           * it makes it possible for people with disabilities to use rdesktop, stickykeys, bouncekeys etc. VERY MUCH useful.                  return False;
238           * XFree86 has had support for it since it's earliest incarnation. I believe it is a reasonable dependency.  
239           */          for (modifierpos = 0; modifierpos < 8; modifierpos++)
         display = XkbOpenDisplay( NULL, &xkb_event, &xkb_error, &xkb_major, &xkb_minor, &xkb_reason );  
         switch(xkb_reason)  
240          {          {
241                  case XkbOD_BadLibraryVersion:                  offset = mod_map->max_keypermod * modifierpos;
242                          error("XkbOD_BadLibraryVersion: XKB extensions in server and the library rdesktop is linked against aren't compatible with each other.\n");  
243                          break;                  for (key = 0; key < mod_map->max_keypermod; key++)
244                  case XkbOD_ConnectionRefused:                  {
245                          error("XkbOD_ConnectionRefused\n");                          if (mod_map->modifiermap[offset + key] == keycode)
246                          break;                                  keysymMask |= 1 << modifierpos;
247                  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;  
248          }          }
249    
250            return (state & keysymMask) ? True : False;
251    }
252    
253    BOOL
254    ui_init(void)
255    {
256            XPixmapFormatValues *pfm;
257            uint16 test;
258            int i;
259    
260            display = XOpenDisplay(NULL);
261          if (display == NULL)          if (display == NULL)
262          {          {
263                  error("Failed to open display\n");                  error("Failed to open display: %s\n", XDisplayName(NULL));
264                  return False;                  return False;
265          }          }
266    
# Line 255  ui_create_window(char *title) Line 268  ui_create_window(char *title)
268          screen = DefaultScreenOfDisplay(display);          screen = DefaultScreenOfDisplay(display);
269          visual = DefaultVisualOfScreen(screen);          visual = DefaultVisualOfScreen(screen);
270          depth = DefaultDepthOfScreen(screen);          depth = DefaultDepthOfScreen(screen);
271            
272          pfm = XListPixmapFormats(display, &i);          pfm = XListPixmapFormats(display, &i);
273          if (pfm != NULL)          if (pfm != NULL)
274          {          {
# Line 263  ui_create_window(char *title) Line 276  ui_create_window(char *title)
276                     desirable, e.g. 24 bits->32 bits. */                     desirable, e.g. 24 bits->32 bits. */
277                  while (i--)                  while (i--)
278                  {                  {
279                          if ((pfm[i].depth == depth)                          if ((pfm[i].depth == depth) && (pfm[i].bits_per_pixel > bpp))
                             && (pfm[i].bits_per_pixel > bpp))  
280                          {                          {
281                                  bpp = pfm[i].bits_per_pixel;                                  bpp = pfm[i].bits_per_pixel;
282                          }                          }
# Line 279  ui_create_window(char *title) Line 291  ui_create_window(char *title)
291                  return False;                  return False;
292          }          }
293    
294          if (depth <= 8)          if (owncolmap != True)
295                  owncolmap = True;          {
         else  
296                  xcolmap = DefaultColormapOfScreen(screen);                  xcolmap = DefaultColormapOfScreen(screen);
297                    if (depth <= 8)
298                    {
299                            printf("You're using a screen depth of 8-bits or lower\n");
300                            printf("If you get scewed colours, try the -C switch\n");
301                    }
302            }
303    
304            gc = XCreateGC(display, RootWindowOfScreen(screen), 0, NULL);
305    
306            if (DoesBackingStore(screen) != Always)
307                    ownbackstore = True;
308    
309          test = 1;          test = 1;
310          host_be = !(BOOL)(*(uint8 *)(&test));          host_be = !(BOOL) (*(uint8 *) (&test));
311          xserver_be = (ImageByteOrder(display) == MSBFirst);          xserver_be = (ImageByteOrder(display) == MSBFirst);
312    
313          white = WhitePixelOfScreen(screen);          if ((width == 0) || (height == 0))
314          attribs.background_pixel = BlackPixelOfScreen(screen);          {
315          attribs.backing_store = DoesBackingStore(screen);                  /* Fetch geometry from _NET_WORKAREA */
316                    uint32 xpos, ypos;
317    
318          if (attribs.backing_store == NotUseful)                  if (get_current_workarea(&xpos, &ypos, &width, &height) < 0)
319                  ownbackstore = True;                  {
320                            error("Failed to get workarea.\n");
321                            error("Perhaps your window manager does not support EWMH?\n");
322                            error("Defaulting to geometry 800x600\n");
323                            width = 800;
324                            height = 600;
325                    }
326            }
327    
328          if (fullscreen)          if (fullscreen)
329          {          {
                 attribs.override_redirect = True;  
330                  width = WidthOfScreen(screen);                  width = WidthOfScreen(screen);
331                  height = HeightOfScreen(screen);                  height = HeightOfScreen(screen);
332          }          }
333          else  
334            /* make sure width is a multiple of 4 */
335            width = (width + 3) & ~3;
336    
337            if (ownbackstore)
338          {          {
339                  attribs.override_redirect = False;                  backstore =
340                            XCreatePixmap(display, RootWindowOfScreen(screen), width, height, depth);
341    
342                    /* clear to prevent rubbish being exposed at startup */
343                    XSetForeground(display, gc, BlackPixelOfScreen(screen));
344                    XFillRectangle(display, backstore, gc, 0, 0, width, height);
345          }          }
346    
347          width = (width + 3) & ~3; /* make width a multiple of 32 bits */          mod_map = XGetModifierMapping(display);
348    
349          wnd = XCreateWindow(display, RootWindowOfScreen(screen),          if (enable_compose)
350                              0, 0, width, height, 0, CopyFromParent,                  IM = XOpenIM(display, NULL, NULL, NULL);
351                              InputOutput, CopyFromParent,  
352                              CWBackingStore | CWBackPixel | CWOverrideRedirect,          xkeymap_init();
353                              &attribs);          return True;
354    }
355    
356    void
357    ui_deinit(void)
358    {
359            if (IM != NULL)
360                    XCloseIM(IM);
361    
362            XFreeModifiermap(mod_map);
363    
364            if (ownbackstore)
365                    XFreePixmap(display, backstore);
366    
367            XFreeGC(display, gc);
368            XCloseDisplay(display);
369            display = NULL;
370    }
371    
372    BOOL
373    ui_create_window(void)
374    {
375            XSetWindowAttributes attribs;
376            XClassHint *classhints;
377            XSizeHints *sizehints;
378            int wndwidth, wndheight;
379            long input_mask, ic_input_mask;
380            XEvent xevent;
381    
382            wndwidth = fullscreen ? WidthOfScreen(screen) : width;
383            wndheight = fullscreen ? HeightOfScreen(screen) : height;
384    
385            attribs.background_pixel = BlackPixelOfScreen(screen);
386            attribs.backing_store = ownbackstore ? NotUseful : Always;
387            attribs.override_redirect = fullscreen;
388    
389            wnd = XCreateWindow(display, RootWindowOfScreen(screen), 0, 0, wndwidth, wndheight,
390                                0, CopyFromParent, InputOutput, CopyFromParent,
391                                CWBackPixel | CWBackingStore | CWOverrideRedirect, &attribs);
392    
393          XStoreName(display, wnd, title);          XStoreName(display, wnd, title);
394    
395            if (hide_decorations)
396                    mwm_hide_decorations();
397    
398          classhints = XAllocClassHint();          classhints = XAllocClassHint();
399          if (classhints != NULL)          if (classhints != NULL)
400          {          {
# Line 334  ui_create_window(char *title) Line 413  ui_create_window(char *title)
413                  XFree(sizehints);                  XFree(sizehints);
414          }          }
415    
416          xkeymap_init();          input_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask |
417                    VisibilityChangeMask | FocusChangeMask;
418    
         input_mask = KeyPressMask | KeyReleaseMask |  
                          ButtonPressMask | ButtonReleaseMask |  
                          EnterWindowMask | LeaveWindowMask | KeymapStateMask;  
419          if (sendmotion)          if (sendmotion)
420                  input_mask |= PointerMotionMask;                  input_mask |= PointerMotionMask;
   
421          if (ownbackstore)          if (ownbackstore)
422                  input_mask |= ExposureMask;                  input_mask |= ExposureMask;
423            if (fullscreen || grab_keyboard)
424                    input_mask |= EnterWindowMask;
425            if (grab_keyboard)
426                    input_mask |= LeaveWindowMask;
427    
428            if (IM != NULL)
429            {
430                    IC = XCreateIC(IM, XNInputStyle, (XIMPreeditNothing | XIMStatusNothing),
431                                   XNClientWindow, wnd, XNFocusWindow, wnd, NULL);
432    
433                    if ((IC != NULL)
434                        && (XGetICValues(IC, XNFilterEvents, &ic_input_mask, NULL) == NULL))
435                            input_mask |= ic_input_mask;
436            }
437    
438          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);  
   
439          XMapWindow(display, wnd);          XMapWindow(display, wnd);
440    
441          /* TODO: error texts... make them friendly. */          /* wait for VisibilityNotify */
442          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 )  
443          {          {
444                          error( "XkbSelectEvents failed.\n");                  XMaskEvent(display, VisibilityChangeMask, &xevent);
                         exit(0);  
445          }          }
446            while (xevent.type != VisibilityNotify);
447    
448            focused = False;
449            mouse_in_wnd = False;
450    
451            /* handle the WM_DELETE_WINDOW protocol */
452            protocol_atom = XInternAtom(display, "WM_PROTOCOLS", True);
453            kill_atom = XInternAtom(display, "WM_DELETE_WINDOW", True);
454            XSetWMProtocols(display, wnd, &kill_atom, 1);
455    
456          return True;          return True;
457  }  }
458    
459  void  void
460  ui_destroy_window()  ui_destroy_window(void)
461  {  {
462          if( xkb != NULL )          if (IC != NULL)
463                  XkbFreeKeyboard(xkb, XkbAllControlsMask, True);                  XDestroyIC(IC);
   
         if (ownbackstore)  
                 XFreePixmap(display, backstore);  
464    
         XFreeGC(display, gc);  
465          XDestroyWindow(display, wnd);          XDestroyWindow(display, wnd);
         XCloseDisplay(display);  
         display = NULL;  
466  }  }
467    
468  static void  void
469  xwin_process_events()  xwin_toggle_fullscreen(void)
470  {  {
471          XkbEvent xkbevent;          Pixmap contents = 0;
472            
473            if (!ownbackstore)
474            {
475                    /* need to save contents of window */
476                    contents = XCreatePixmap(display, wnd, width, height, depth);
477                    XCopyArea(display, wnd, contents, gc, 0, 0, width, height, 0, 0);
478            }
479    
480            ui_destroy_window();
481            fullscreen = !fullscreen;
482            ui_create_window();
483    
484            XDefineCursor(display, wnd, current_cursor);
485    
486            if (!ownbackstore)
487            {
488                    XCopyArea(display, contents, wnd, gc, 0, 0, width, height, 0, 0);
489                    XFreePixmap(display, contents);
490            }
491    }
492    
493    /* Process all events in Xlib queue
494       Returns 0 after user quit, 1 otherwise */
495    static int
496    xwin_process_events(void)
497    {
498            XEvent xevent;
499          KeySym keysym;          KeySym keysym;
         uint8 scancode;  
500          uint16 button, flags;          uint16 button, flags;
501          uint32 ev_time;          uint32 ev_time;
502          uint32 tmpmods;          key_translation tr;
503            char str[256];
504            Status status;
505            unsigned int state;
506            Window wdummy;
507            int dummy;
508    
509          if (display == NULL)          while (XPending(display) > 0)
                 return;  
   
         while (XCheckMaskEvent(display, ~0, &xkbevent.core))  
510          {          {
511                  ev_time = time(NULL);                  XNextEvent(display, &xevent);
512    
513                    if ((IC != NULL) && (XFilterEvent(&xevent, None) == True))
514                    {
515                            DEBUG_KBD(("Filtering event\n"));
516                            continue;
517                    }
518    
519                  flags = 0;                  flags = 0;
520    
521                  switch (xkbevent.type)                  switch (xevent.type)
522                  {                  {
523                          case KeymapNotify:                          case ClientMessage:
524                                  /* TODO:                                  /* the window manager told us to quit */
525                                   * read modifier status at focus in, and update the local masks, and the other end as well..                                  if ((xevent.xclient.message_type == protocol_atom)
526                                   * if not, we may get out of sync.                                      && (xevent.xclient.data.l[0] == kill_atom))
527                                   * xkbevent.core.xkeymap.key_vector                                          /* Quit */
528                                   * char key_vector[32];                                          return 0;
                                  */  
529                                  break;                                  break;
530    
                         case KeyRelease:  
                                 flags = KBD_FLAG_DOWN | KBD_FLAG_UP;  
                                 /* fall through */  
   
531                          case KeyPress:                          case KeyPress:
532                                  if( XkbTranslateKeyCode(xkb, xkbevent.core.xkey.keycode, xkbevent.core.xkey.state, &tmpmods, &keysym) == False )                                  if (IC != NULL)
533                                            /* Multi_key compatible version */
534                                    {
535                                            XmbLookupString(IC,
536                                                            (XKeyPressedEvent *) &
537                                                            xevent, str, sizeof(str), &keysym, &status);
538                                            if (!((status == XLookupKeySym) || (status == XLookupBoth)))
539                                            {
540                                                    error("XmbLookupString failed with status 0x%x\n",
541                                                          status);
542                                                    break;
543                                            }
544                                    }
545                                    else
546                                    {
547                                            /* Plain old XLookupString */
548                                            DEBUG_KBD(("\nNo input context, using XLookupString\n"));
549                                            XLookupString((XKeyEvent *) & xevent,
550                                                          str, sizeof(str), &keysym, NULL);
551                                    }
552    
553                                    DEBUG_KBD(("KeyPress for (keysym 0x%lx, %s)\n", keysym,
554                                               get_ksname(keysym)));
555    
556                                    ev_time = time(NULL);
557                                    if (handle_special_keys(keysym, xevent.xkey.state, ev_time, True))
558                                          break;                                          break;
                                 scancode = xkeymap_translate_key(keysym, xkbevent.core.xkey.keycode, &flags);  
559    
560                                  if (scancode == 0 )                                  tr = xkeymap_translate_key(keysym,
561                                                               xevent.xkey.keycode, xevent.xkey.state);
562    
563                                    if (tr.scancode == 0)
564                                          break;                                          break;
565    
566                                  /* keep track of the modifiers -- needed for stickykeys... */                                  ensure_remote_modifiers(ev_time, tr);
                                 if( xkbevent.type == KeyPress )  
                                         xwin_press_modifiers( &xkbevent.core.xkey, ev_time, scancode );  
567    
568                                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE, flags, scancode, 0);                                  rdp_send_scancode(ev_time, RDP_KEYPRESS, tr.scancode);
569                                    break;
570    
571                            case KeyRelease:
572                                    XLookupString((XKeyEvent *) & xevent, str,
573                                                  sizeof(str), &keysym, NULL);
574    
575                                    DEBUG_KBD(("\nKeyRelease for (keysym 0x%lx, %s)\n", keysym,
576                                               get_ksname(keysym)));
577    
578                                  if( xkbevent.type == KeyRelease )                                  ev_time = time(NULL);
579                                          xwin_release_modifiers( &xkbevent.core.xkey, ev_time, scancode );                                  if (handle_special_keys(keysym, xevent.xkey.state, ev_time, False))
580                                            break;
581    
582                                    tr = xkeymap_translate_key(keysym,
583                                                               xevent.xkey.keycode, xevent.xkey.state);
584    
585                                    if (tr.scancode == 0)
586                                            break;
587    
588                                    rdp_send_scancode(ev_time, RDP_KEYRELEASE, tr.scancode);
589                                  break;                                  break;
590    
591                          case ButtonPress:                          case ButtonPress:
# Line 444  xwin_process_events() Line 593  xwin_process_events()
593                                  /* fall through */                                  /* fall through */
594    
595                          case ButtonRelease:                          case ButtonRelease:
596                                  button = xkeymap_translate_button(xkbevent.core.xbutton.button);                                  button = xkeymap_translate_button(xevent.xbutton.button);
597                                  if (button == 0)                                  if (button == 0)
598                                          break;                                          break;
599    
600                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
601                                                 flags | button,                                                 flags | button, xevent.xbutton.x, xevent.xbutton.y);
                                                xkbevent.core.xbutton.x,  
                                                xkbevent.core.xbutton.y);  
602                                  break;                                  break;
603    
604                          case MotionNotify:                          case MotionNotify:
605                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
606                                                 MOUSE_FLAG_MOVE,                                                 MOUSE_FLAG_MOVE, xevent.xmotion.x, xevent.xmotion.y);
607                                                 xkbevent.core.xmotion.x,                                  break;
608                                                 xkbevent.core.xmotion.y);  
609                            case FocusIn:
610                                    if (xevent.xfocus.mode == NotifyGrab)
611                                            break;
612                                    focused = True;
613                                    XQueryPointer(display, wnd, &wdummy, &wdummy, &dummy, &dummy,
614                                                  &dummy, &dummy, &state);
615                                    reset_modifier_keys(state);
616                                    if (grab_keyboard && mouse_in_wnd)
617                                            XGrabKeyboard(display, wnd, True,
618                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
619                                    break;
620    
621                            case FocusOut:
622                                    if (xevent.xfocus.mode == NotifyUngrab)
623                                            break;
624                                    focused = False;
625                                    if (xevent.xfocus.mode == NotifyWhileGrabbed)
626                                            XUngrabKeyboard(display, CurrentTime);
627                                  break;                                  break;
628    
629                          case EnterNotify:                          case EnterNotify:
630                                  XGrabKeyboard(display, wnd, True, GrabModeAsync,                                  /* we only register for this event when in fullscreen mode */
631                                                GrabModeAsync, CurrentTime);                                  /* or grab_keyboard */
632                                    mouse_in_wnd = True;
633                                    if (fullscreen)
634                                    {
635                                            XSetInputFocus(display, wnd, RevertToPointerRoot,
636                                                           CurrentTime);
637                                            break;
638                                    }
639                                    if (focused)
640                                            XGrabKeyboard(display, wnd, True,
641                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
642                                  break;                                  break;
643    
644                          case LeaveNotify:                          case LeaveNotify:
645                                    /* we only register for this event when grab_keyboard */
646                                    mouse_in_wnd = False;
647                                  XUngrabKeyboard(display, CurrentTime);                                  XUngrabKeyboard(display, CurrentTime);
648                                  break;                                  break;
649    
650                          case Expose:                          case Expose:
651                                  XCopyArea(display, backstore, wnd, gc,                                  XCopyArea(display, backstore, wnd, gc,
652                                            xkbevent.core.xexpose.x, xkbevent.core.xexpose.y,                                            xevent.xexpose.x, xevent.xexpose.y,
653                                            xkbevent.core.xexpose.width, xkbevent.core.xexpose.height,                                            xevent.xexpose.width,
654                                            xkbevent.core.xexpose.x, xkbevent.core.xexpose.y);                                            xevent.xexpose.height,
655                                              xevent.xexpose.x, xevent.xexpose.y);
656                                  break;                                  break;
                 }  
         }  
 }  
   
 void  
 xwin_release_modifiers(XKeyEvent* ev, 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 0x38:  
                 key_down_state &= ~DMod1Mask;  
                 break;  
         case 0xb8:  
                 key_down_state &= ~DMod2Mask;  
                 break;  
         }  
   
         if( !(ShiftMask & ev->state) && (key_down_state & DShift1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0x2a, 0);  
                 key_down_state &= ~DShift1Mask;  
657    
658          }                          case MappingNotify:
659                                    /* Refresh keyboard mapping if it has changed. This is important for
660          if( !(ControlMask & ev->state) && (key_down_state & DControl1Mask))                                     Xvnc, since it allocates keycodes dynamically */
661          {                                  if (xevent.xmapping.request == MappingKeyboard
662                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0x1d, 0);                                      || xevent.xmapping.request == MappingModifier)
663                  key_down_state &= ~DControl1Mask;                                          XRefreshKeyboardMapping(&xevent.xmapping);
   
         }  
           
         if( !(Mod1Mask & ev->state) && (key_down_state & DMod1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0x38, 0);  
                 key_down_state &= ~DMod1Mask;  
   
         }  
           
         if( !(Mod2Mask & ev->state) && (key_down_state & DMod2Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0xb8, 0);  
                 key_down_state &= ~DMod2Mask;  
         }  
 }  
   
   
 void  
 xwin_press_modifiers(XKeyEvent* ev, uint32 ev_time, uint32 scancode)  
 {  
         key_modifier_state = ev->state;  
   
         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 0x38:  
                 key_down_state |= DMod1Mask;  
                 break;  
         case 0xb8:  
                 key_down_state |= DMod2Mask;  
                 break;  
         }  
664    
665          if( (ShiftMask & ev->state) && !((key_down_state & DShift1Mask) || (key_down_state & DShift2Mask)))                                  if (xevent.xmapping.request == MappingModifier)
666          {                                  {
667                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0x2a, 0);                                          XFreeModifiermap(mod_map);
668                  key_down_state |= DShift1Mask;                                          mod_map = XGetModifierMapping(display);
669                                    }
670          }                                  break;
   
         if( (ControlMask & ev->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 & ev->state) && !(key_down_state & DMod1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0x38, 0);  
                 key_down_state |= DMod1Mask;  
   
         }  
   
         if( (Mod2Mask & ev->state) && !(key_down_state & DMod2Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0xb8, 0);  
                 key_down_state |= DMod2Mask;  
671    
672                    }
673          }          }
674            /* Keep going */
675            return 1;
676  }  }
677    
678  void  /* Returns 0 after user quit, 1 otherwise */
679    int
680  ui_select(int rdp_socket)  ui_select(int rdp_socket)
681  {  {
682          int n = (rdp_socket > x_socket) ? rdp_socket+1 : x_socket+1;          int n = (rdp_socket > x_socket) ? rdp_socket + 1 : x_socket + 1;
683          fd_set rfds;          fd_set rfds;
684    
         XFlush(display);  
   
685          FD_ZERO(&rfds);          FD_ZERO(&rfds);
686    
687          while (True)          while (True)
688          {          {
689                    /* Process any events already waiting */
690                    if (!xwin_process_events())
691                            /* User quit */
692                            return 0;
693    
694                  FD_ZERO(&rfds);                  FD_ZERO(&rfds);
695                  FD_SET(rdp_socket, &rfds);                  FD_SET(rdp_socket, &rfds);
696                  FD_SET(x_socket, &rfds);                  FD_SET(x_socket, &rfds);
# Line 613  ui_select(int rdp_socket) Line 704  ui_select(int rdp_socket)
704                                  continue;                                  continue;
705                  }                  }
706    
                 if (FD_ISSET(x_socket, &rfds))  
                         xwin_process_events();  
   
707                  if (FD_ISSET(rdp_socket, &rfds))                  if (FD_ISSET(rdp_socket, &rfds))
708                          return;                          return 1;
709          }          }
710  }  }
711    
# Line 628  ui_move_pointer(int x, int y) Line 716  ui_move_pointer(int x, int y)
716  }  }
717    
718  HBITMAP  HBITMAP
719  ui_create_bitmap(int width, int height, uint8 *data)  ui_create_bitmap(int width, int height, uint8 * data)
720  {  {
721          XImage *image;          XImage *image;
722          Pixmap bitmap;          Pixmap bitmap;
# Line 636  ui_create_bitmap(int width, int height, Line 724  ui_create_bitmap(int width, int height,
724    
725          tdata = (owncolmap ? data : translate_image(width, height, data));          tdata = (owncolmap ? data : translate_image(width, height, data));
726          bitmap = XCreatePixmap(display, wnd, width, height, depth);          bitmap = XCreatePixmap(display, wnd, width, height, depth);
727          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
728                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
729    
730          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);
731    
# Line 648  ui_create_bitmap(int width, int height, Line 736  ui_create_bitmap(int width, int height,
736  }  }
737    
738  void  void
739  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)  
740  {  {
741          XImage *image;          XImage *image;
742          uint8 *tdata;          uint8 *tdata;
743    
744          tdata = (owncolmap ? data : translate_image(width, height, data));          tdata = (owncolmap ? data : translate_image(width, height, data));
745          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
746                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
747    
748          if (ownbackstore)          if (ownbackstore)
749          {          {
# Line 676  ui_paint_bitmap(int x, int y, int cx, in Line 763  ui_paint_bitmap(int x, int y, int cx, in
763  void  void
764  ui_destroy_bitmap(HBITMAP bmp)  ui_destroy_bitmap(HBITMAP bmp)
765  {  {
766          XFreePixmap(display, (Pixmap)bmp);          XFreePixmap(display, (Pixmap) bmp);
767  }  }
768    
769  HGLYPH  HGLYPH
770  ui_create_glyph(int width, int height, uint8 *data)  ui_create_glyph(int width, int height, uint8 * data)
771  {  {
772          XImage *image;          XImage *image;
773          Pixmap bitmap;          Pixmap bitmap;
# Line 692  ui_create_glyph(int width, int height, u Line 779  ui_create_glyph(int width, int height, u
779          bitmap = XCreatePixmap(display, wnd, width, height, 1);          bitmap = XCreatePixmap(display, wnd, width, height, 1);
780          gc = XCreateGC(display, bitmap, 0, NULL);          gc = XCreateGC(display, bitmap, 0, NULL);
781    
782          image = XCreateImage(display, visual, 1, ZPixmap, 0,          image = XCreateImage(display, visual, 1, ZPixmap, 0, (char *) data,
783                               data, width, height, 8, scanline);                               width, height, 8, scanline);
784          image->byte_order = MSBFirst;          image->byte_order = MSBFirst;
785          image->bitmap_bit_order = MSBFirst;          image->bitmap_bit_order = MSBFirst;
786          XInitImage(image);          XInitImage(image);
# Line 702  ui_create_glyph(int width, int height, u Line 789  ui_create_glyph(int width, int height, u
789    
790          XFree(image);          XFree(image);
791          XFreeGC(display, gc);          XFreeGC(display, gc);
792          return (HGLYPH)bitmap;          return (HGLYPH) bitmap;
793  }  }
794    
795  void  void
796  ui_destroy_glyph(HGLYPH glyph)  ui_destroy_glyph(HGLYPH glyph)
797  {  {
798          XFreePixmap(display, (Pixmap)glyph);          XFreePixmap(display, (Pixmap) glyph);
799  }  }
800    
801  HCURSOR  HCURSOR
802  ui_create_cursor(unsigned int x, unsigned int y, int width,  ui_create_cursor(unsigned int x, unsigned int y, int width, int height,
803                   int height, uint8 *andmask, uint8 *xormask)                   uint8 * andmask, uint8 * xormask)
804  {  {
805          HGLYPH maskglyph, cursorglyph;          HGLYPH maskglyph, cursorglyph;
806          XColor bg, fg;          XColor bg, fg;
# Line 770  ui_create_cursor(unsigned int x, unsigne Line 857  ui_create_cursor(unsigned int x, unsigne
857    
858          cursorglyph = ui_create_glyph(width, height, cursor);          cursorglyph = ui_create_glyph(width, height, cursor);
859          maskglyph = ui_create_glyph(width, height, mask);          maskglyph = ui_create_glyph(width, height, mask);
860            
861          xcursor = XCreatePixmapCursor(display, (Pixmap)cursorglyph,          xcursor =
862                                  (Pixmap)maskglyph, &fg, &bg, x, y);                  XCreatePixmapCursor(display, (Pixmap) cursorglyph,
863                                        (Pixmap) maskglyph, &fg, &bg, x, y);
864    
865          ui_destroy_glyph(maskglyph);          ui_destroy_glyph(maskglyph);
866          ui_destroy_glyph(cursorglyph);          ui_destroy_glyph(cursorglyph);
867          xfree(mask);          xfree(mask);
868          xfree(cursor);          xfree(cursor);
869          return (HCURSOR)xcursor;          return (HCURSOR) xcursor;
870  }  }
871    
872  void  void
873  ui_set_cursor(HCURSOR cursor)  ui_set_cursor(HCURSOR cursor)
874  {  {
875          XDefineCursor(display, wnd, (Cursor)cursor);          current_cursor = (Cursor) cursor;
876            XDefineCursor(display, wnd, current_cursor);
877  }  }
878    
879  void  void
880  ui_destroy_cursor(HCURSOR cursor)  ui_destroy_cursor(HCURSOR cursor)
881  {  {
882          XFreeCursor(display, (Cursor)cursor);          XFreeCursor(display, (Cursor) cursor);
883  }  }
884    
885  #define MAKE_XCOLOR(xc,c) \  #define MAKE_XCOLOR(xc,c) \
# Line 799  ui_destroy_cursor(HCURSOR cursor) Line 888  ui_destroy_cursor(HCURSOR cursor)
888                  (xc)->blue  = ((c)->blue  << 8) | (c)->blue; \                  (xc)->blue  = ((c)->blue  << 8) | (c)->blue; \
889                  (xc)->flags = DoRed | DoGreen | DoBlue;                  (xc)->flags = DoRed | DoGreen | DoBlue;
890    
891    
892  HCOLOURMAP  HCOLOURMAP
893  ui_create_colourmap(COLOURMAP *colours)  ui_create_colourmap(COLOURMAP * colours)
894  {  {
895          COLOURENTRY *entry;          COLOURENTRY *entry;
896          int i, ncolours = colours->ncolours;          int i, ncolours = colours->ncolours;
897            if (!owncolmap)
898            {
899                    uint32 *map = xmalloc(sizeof(*colmap) * ncolours);
900                    XColor xentry;
901                    XColor xc_cache[256];
902                    uint32 colour;
903                    int colLookup = 256;
904                    for (i = 0; i < ncolours; i++)
905                    {
906                            entry = &colours->colours[i];
907                            MAKE_XCOLOR(&xentry, entry);
908    
909                            if (XAllocColor(display, xcolmap, &xentry) == 0)
910                            {
911                                    /* Allocation failed, find closest match. */
912                                    int j = 256;
913                                    int nMinDist = 3 * 256 * 256;
914                                    long nDist = nMinDist;
915    
916          if (owncolmap)                                  /* only get the colors once */
917                                    while (colLookup--)
918                                    {
919                                            xc_cache[colLookup].pixel = colLookup;
920                                            xc_cache[colLookup].red = xc_cache[colLookup].green =
921                                                    xc_cache[colLookup].blue = 0;
922                                            xc_cache[colLookup].flags = 0;
923                                            XQueryColor(display,
924                                                        DefaultColormap(display,
925                                                                        DefaultScreen(display)),
926                                                        &xc_cache[colLookup]);
927                                    }
928                                    colLookup = 0;
929    
930                                    /* approximate the pixel */
931                                    while (j--)
932                                    {
933                                            if (xc_cache[j].flags)
934                                            {
935                                                    nDist = ((long) (xc_cache[j].red >> 8) -
936                                                             (long) (xentry.red >> 8)) *
937                                                            ((long) (xc_cache[j].red >> 8) -
938                                                             (long) (xentry.red >> 8)) +
939                                                            ((long) (xc_cache[j].green >> 8) -
940                                                             (long) (xentry.green >> 8)) *
941                                                            ((long) (xc_cache[j].green >> 8) -
942                                                             (long) (xentry.green >> 8)) +
943                                                            ((long) (xc_cache[j].blue >> 8) -
944                                                             (long) (xentry.blue >> 8)) *
945                                                            ((long) (xc_cache[j].blue >> 8) -
946                                                             (long) (xentry.blue >> 8));
947                                            }
948                                            if (nDist < nMinDist)
949                                            {
950                                                    nMinDist = nDist;
951                                                    xentry.pixel = j;
952                                            }
953                                    }
954                            }
955                            colour = xentry.pixel;
956    
957                            /* update our cache */
958                            if (xentry.pixel < 256)
959                            {
960                                    xc_cache[xentry.pixel].red = xentry.red;
961                                    xc_cache[xentry.pixel].green = xentry.green;
962                                    xc_cache[xentry.pixel].blue = xentry.blue;
963    
964                            }
965    
966    
967                            /* byte swap here to make translate_image faster */
968                            map[i] = translate_colour(colour);
969                    }
970                    return map;
971            }
972            else
973          {          {
974                  XColor *xcolours, *xentry;                  XColor *xcolours, *xentry;
975                  Colormap map;                  Colormap map;
# Line 823  ui_create_colourmap(COLOURMAP *colours) Line 987  ui_create_colourmap(COLOURMAP *colours)
987                  XStoreColors(display, map, xcolours, ncolours);                  XStoreColors(display, map, xcolours, ncolours);
988    
989                  xfree(xcolours);                  xfree(xcolours);
990                  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;  
991          }          }
992  }  }
993    
994  void  void
995  ui_destroy_colourmap(HCOLOURMAP map)  ui_destroy_colourmap(HCOLOURMAP map)
996  {  {
997          if (owncolmap)          if (!owncolmap)
                 XFreeColormap(display, (Colormap)map);  
         else  
998                  xfree(map);                  xfree(map);
999            else
1000                    XFreeColormap(display, (Colormap) map);
1001  }  }
1002    
1003  void  void
1004  ui_set_colourmap(HCOLOURMAP map)  ui_set_colourmap(HCOLOURMAP map)
1005  {  {
1006          if (owncolmap)          if (!owncolmap)
                 XSetWindowColormap(display, wnd, (Colormap)map);  
         else  
1007                  colmap = map;                  colmap = map;
1008            else
1009                    XSetWindowColormap(display, wnd, (Colormap) map);
1010  }  }
1011    
1012  void  void
# Line 880  ui_set_clip(int x, int y, int cx, int cy Line 1022  ui_set_clip(int x, int y, int cx, int cy
1022  }  }
1023    
1024  void  void
1025  ui_reset_clip()  ui_reset_clip(void)
1026  {  {
1027          XRectangle rect;          XRectangle rect;
1028    
# Line 892  ui_reset_clip() Line 1034  ui_reset_clip()
1034  }  }
1035    
1036  void  void
1037  ui_bell()  ui_bell(void)
1038  {  {
1039          XBell(display, 0);          XBell(display, 0);
1040  }  }
# Line 909  ui_destblt(uint8 opcode, Line 1051  ui_destblt(uint8 opcode,
1051  void  void
1052  ui_patblt(uint8 opcode,  ui_patblt(uint8 opcode,
1053            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
1054            /* brush */ BRUSH *brush, int bgcolour, int fgcolour)            /* brush */ BRUSH * brush, int bgcolour, int fgcolour)
1055  {  {
1056          Pixmap fill;          Pixmap fill;
1057            uint8 i, ipattern[8];
1058    
1059          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1060    
# Line 923  ui_patblt(uint8 opcode, Line 1066  ui_patblt(uint8 opcode,
1066                          break;                          break;
1067    
1068                  case 3: /* Pattern */                  case 3: /* Pattern */
1069                          fill = (Pixmap)ui_create_glyph(8, 8, brush->pattern);                          for (i = 0; i != 8; i++)
1070                                    ipattern[7 - i] = brush->pattern[i];
1071                            fill = (Pixmap) ui_create_glyph(8, 8, ipattern);
1072    
1073                          SET_FOREGROUND(bgcolour);                          SET_FOREGROUND(bgcolour);
1074                          SET_BACKGROUND(fgcolour);                          SET_BACKGROUND(fgcolour);
# Line 934  ui_patblt(uint8 opcode, Line 1079  ui_patblt(uint8 opcode,
1079                          FILL_RECTANGLE(x, y, cx, cy);                          FILL_RECTANGLE(x, y, cx, cy);
1080    
1081                          XSetFillStyle(display, gc, FillSolid);                          XSetFillStyle(display, gc, FillSolid);
1082                          ui_destroy_glyph((HGLYPH)fill);                          XSetTSOrigin(display, gc, 0, 0);
1083                            ui_destroy_glyph((HGLYPH) fill);
1084                          break;                          break;
1085    
1086                  default:                  default:
# Line 952  ui_screenblt(uint8 opcode, Line 1098  ui_screenblt(uint8 opcode,
1098          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1099          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);
1100          if (ownbackstore)          if (ownbackstore)
1101                  XCopyArea(display, backstore, backstore, gc, srcx, srcy,                  XCopyArea(display, backstore, backstore, gc, srcx, srcy, cx, cy, x, y);
                           cx, cy, x, y);  
1102          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1103  }  }
1104    
# Line 963  ui_memblt(uint8 opcode, Line 1108  ui_memblt(uint8 opcode,
1108            /* src */ HBITMAP src, int srcx, int srcy)            /* src */ HBITMAP src, int srcx, int srcy)
1109  {  {
1110          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1111          XCopyArea(display, (Pixmap)src, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, (Pixmap) src, wnd, gc, srcx, srcy, cx, cy, x, y);
1112          if (ownbackstore)          if (ownbackstore)
1113                  XCopyArea(display, (Pixmap)src, backstore, gc, srcx, srcy,                  XCopyArea(display, (Pixmap) src, backstore, gc, srcx, srcy, cx, cy, x, y);
                           cx, cy, x, y);  
1114          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1115  }  }
1116    
# Line 974  void Line 1118  void
1118  ui_triblt(uint8 opcode,  ui_triblt(uint8 opcode,
1119            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
1120            /* src */ HBITMAP src, int srcx, int srcy,            /* src */ HBITMAP src, int srcx, int srcy,
1121            /* brush */ BRUSH *brush, int bgcolour, int fgcolour)            /* brush */ BRUSH * brush, int bgcolour, int fgcolour)
1122  {  {
1123          /* This is potentially difficult to do in general. Until someone          /* This is potentially difficult to do in general. Until someone
1124             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 983  ui_triblt(uint8 opcode, Line 1127  ui_triblt(uint8 opcode,
1127          {          {
1128                  case 0x69:      /* PDSxxn */                  case 0x69:      /* PDSxxn */
1129                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);
1130                          ui_patblt(ROP2_NXOR, x, y, cx, cy,                          ui_patblt(ROP2_NXOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1131                          break;                          break;
1132    
1133                  case 0xb8:      /* PSDPxax */                  case 0xb8:      /* PSDPxax */
1134                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1135                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);
1136                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1137                          break;                          break;
1138    
1139                  case 0xc0:      /* PSa */                  case 0xc0:      /* PSa */
1140                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);
1141                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour,                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   fgcolour);  
1142                          break;                          break;
1143    
1144                  default:                  default:
# Line 1010  ui_triblt(uint8 opcode, Line 1150  ui_triblt(uint8 opcode,
1150  void  void
1151  ui_line(uint8 opcode,  ui_line(uint8 opcode,
1152          /* dest */ int startx, int starty, int endx, int endy,          /* dest */ int startx, int starty, int endx, int endy,
1153          /* pen */ PEN *pen)          /* pen */ PEN * pen)
1154  {  {
1155          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1156          SET_FOREGROUND(pen->colour);          SET_FOREGROUND(pen->colour);
# Line 1029  ui_rect( Line 1169  ui_rect(
1169          FILL_RECTANGLE(x, y, cx, cy);          FILL_RECTANGLE(x, y, cx, cy);
1170  }  }
1171    
1172    /* warning, this function only draws on wnd or backstore, not both */
1173  void  void
1174  ui_draw_glyph(int mixmode,  ui_draw_glyph(int mixmode,
1175                /* dest */ int x, int y, int cx, int cy,                /* dest */ int x, int y, int cx, int cy,
1176                /* src */ HGLYPH glyph, int srcx, int srcy, int bgcolour,                /* src */ HGLYPH glyph, int srcx, int srcy,
1177                int fgcolour)                int bgcolour, int fgcolour)
1178  {  {
1179          SET_FOREGROUND(fgcolour);          SET_FOREGROUND(fgcolour);
1180          SET_BACKGROUND(bgcolour);          SET_BACKGROUND(bgcolour);
1181    
1182          XSetFillStyle(display, gc, (mixmode == MIX_TRANSPARENT)          XSetFillStyle(display, gc,
1183                        ? FillStippled : FillOpaqueStippled);                        (mixmode == MIX_TRANSPARENT) ? FillStippled : FillOpaqueStippled);
1184          XSetStipple(display, gc, (Pixmap)glyph);          XSetStipple(display, gc, (Pixmap) glyph);
1185          XSetTSOrigin(display, gc, x, y);          XSetTSOrigin(display, gc, x, y);
1186    
1187          FILL_RECTANGLE(x, y, cx, cy);          if (ownbackstore)
1188                    XFillRectangle(display, backstore, gc, x, y, cx, cy);
1189            else
1190                    XFillRectangle(display, wnd, gc, x, y, cx, cy);
1191    
1192          XSetFillStyle(display, gc, FillSolid);          XSetFillStyle(display, gc, FillSolid);
1193  }  }
# Line 1057  ui_draw_glyph(int mixmode, Line 1201  ui_draw_glyph(int mixmode,
1201        if ((xyoffset & 0x80))\        if ((xyoffset & 0x80))\
1202          {\          {\
1203            if (flags & TEXT2_VERTICAL) \            if (flags & TEXT2_VERTICAL) \
1204              y += ttext[++idx] | (ttext[++idx] << 8);\              y += ttext[idx+1] | (ttext[idx+2] << 8);\
1205            else\            else\
1206              x += ttext[++idx] | (ttext[++idx] << 8);\              x += ttext[idx+1] | (ttext[idx+2] << 8);\
1207              idx += 2;\
1208          }\          }\
1209        else\        else\
1210          {\          {\
# Line 1082  ui_draw_glyph(int mixmode, Line 1227  ui_draw_glyph(int mixmode,
1227    
1228  void  void
1229  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,
1230               int clipx, int clipy, int clipcx, int clipcy, int boxx,               int clipx, int clipy, int clipcx, int clipcy,
1231               int boxy, int boxcx, int boxcy, int bgcolour,               int boxx, int boxy, int boxcx, int boxcy, int bgcolour,
1232               int fgcolour, uint8 * text, uint8 length)               int fgcolour, uint8 * text, uint8 length)
1233  {  {
1234          FONTGLYPH *glyph;          FONTGLYPH *glyph;
# Line 1102  ui_draw_text(uint8 font, uint8 flags, in Line 1247  ui_draw_text(uint8 font, uint8 flags, in
1247          }          }
1248    
1249          /* Paint text, character by character */          /* Paint text, character by character */
1250          for (i = 0; i < length;) {          for (i = 0; i < length;)
1251                  switch (text[i]) {          {
1252                  case 0xff:                  switch (text[i])
1253                          if (i + 2 < length)                  {
1254                                  cache_put_text(text[i + 1], text, text[i + 2]);                          case 0xff:
1255                          else {                                  if (i + 2 < length)
1256                                  error("this shouldn't be happening\n");                                          cache_put_text(text[i + 1], text, text[i + 2]);
1257                                    else
1258                                    {
1259                                            error("this shouldn't be happening\n");
1260                                            exit(1);
1261                                    }
1262                                    /* this will move pointer from start to first character after FF command */
1263                                    length -= i + 3;
1264                                    text = &(text[i + 3]);
1265                                    i = 0;
1266                                  break;                                  break;
                         }  
                         /* this will move pointer from start to first character after FF command */  
                         length -= i + 3;  
                         text = &(text[i + 3]);  
                         i = 0;  
                         break;  
1267    
1268                  case 0xfe:                          case 0xfe:
1269                          entry = cache_get_text(text[i + 1]);                                  entry = cache_get_text(text[i + 1]);
1270                          if (entry != NULL) {                                  if (entry != NULL)
1271                                  if ((((uint8 *) (entry->data))[1] == 0)                                  {
1272                                      && (!(flags & TEXT2_IMPLICIT_X))) {                                          if ((((uint8 *) (entry->data))[1] ==
1273                                          if (flags & TEXT2_VERTICAL)                                               0) && (!(flags & TEXT2_IMPLICIT_X)))
1274                                                  y += text[i + 2];                                          {
1275                                                    if (flags & TEXT2_VERTICAL)
1276                                                            y += text[i + 2];
1277                                                    else
1278                                                            x += text[i + 2];
1279                                            }
1280                                            if (i + 2 < length)
1281                                                    i += 3;
1282                                          else                                          else
1283                                                  x += text[i + 2];                                                  i += 2;
1284                                            length -= i;
1285                                            /* this will move pointer from start to first character after FE command */
1286                                            text = &(text[i]);
1287                                            i = 0;
1288                                            for (j = 0; j < entry->size; j++)
1289                                                    DO_GLYPH(((uint8 *) (entry->data)), j);
1290                                  }                                  }
1291                                  if (i + 2 < length)                                  break;
                                         i += 3;  
                                 else  
                                         i += 2;  
                                 length -= i;  
                                 /* this will move pointer from start to first character after FE command */  
                                 text = &(text[i]);  
                                 i = 0;  
                                 for (j = 0; j < entry->size; j++)  
                                         DO_GLYPH(((uint8 *) (entry->data)), j);  
                         }  
                         break;  
1292    
1293                  default:                          default:
1294                          DO_GLYPH(text, i);                                  DO_GLYPH(text, i);
1295                          i++;                                  i++;
1296                          break;                                  break;
1297                  }                  }
1298          }          }
1299            if (ownbackstore)
1300            {
1301                    if (boxcx > 1)
1302                            XCopyArea(display, backstore, wnd, gc, boxx,
1303                                      boxy, boxcx, boxcy, boxx, boxy);
1304                    else
1305                            XCopyArea(display, backstore, wnd, gc, clipx,
1306                                      clipy, clipcx, clipcy, clipx, clipy);
1307            }
1308  }  }
1309    
1310  void  void
# Line 1158  ui_desktop_save(uint32 offset, int x, in Line 1315  ui_desktop_save(uint32 offset, int x, in
1315    
1316          if (ownbackstore)          if (ownbackstore)
1317          {          {
1318                  image = XGetImage(display, backstore, x, y, cx, cy, AllPlanes,                  image = XGetImage(display, backstore, x, y, cx, cy, AllPlanes, ZPixmap);
                                   ZPixmap);  
1319          }          }
1320          else          else
1321          {          {
1322                  pix = XCreatePixmap(display, wnd, cx, cy, depth);                  pix = XCreatePixmap(display, wnd, cx, cy, depth);
1323                  XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);                  XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);
1324                  image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes,                  image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes, ZPixmap);
                                   ZPixmap);  
1325                  XFreePixmap(display, pix);                  XFreePixmap(display, pix);
1326          }          }
1327    
1328          offset *= bpp/8;          offset *= bpp / 8;
1329          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);  
1330    
1331          XDestroyImage(image);          XDestroyImage(image);
1332  }  }
# Line 1183  ui_desktop_restore(uint32 offset, int x, Line 1337  ui_desktop_restore(uint32 offset, int x,
1337          XImage *image;          XImage *image;
1338          uint8 *data;          uint8 *data;
1339    
1340          offset *= bpp/8;          offset *= bpp / 8;
1341          data = cache_get_desktop(offset, cx, cy, bpp/8);          data = cache_get_desktop(offset, cx, cy, bpp / 8);
1342          if (data == NULL)          if (data == NULL)
1343                  return;                  return;
1344    
1345          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
1346                               0, data, cx, cy, BitmapPad(display),                               (char *) data, cx, cy, BitmapPad(display), cx * bpp / 8);
                              cx * bpp/8);  
1347    
1348          if (ownbackstore)          if (ownbackstore)
1349          {          {

Legend:
Removed from v.52  
changed lines
  Added in v.279

  ViewVC Help
Powered by ViewVC 1.1.26