/[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 64 by astrand, Thu Jul 18 16:38:31 2002 UTC revision 297 by matthewc, Tue Jan 28 12:27:28 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)
# 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;  
238    
239          int xkb_minor, xkb_major;          KeyCode keycode = XKeysymToKeycode(display, keysym);
         int xkb_event, xkb_error, xkb_reason;  
240    
241          /* compare compiletime libs with runtime libs. */          if (keycode == NoSymbol)
         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");  
242                  return False;                  return False;
         }  
   
243    
244          display =          for (modifierpos = 0; modifierpos < 8; modifierpos++)
                 XkbOpenDisplay(NULL, &xkb_event, &xkb_error, &xkb_major,  
                                &xkb_minor, &xkb_reason);  
         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 269  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 285  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 xpos, ypos;
319    
320          if (attribs.backing_store == NotUseful)                  if (get_current_workarea(&xpos, &ypos, &width, &height) < 0)
321                  ownbackstore = True;                  {
322                            warning("Failed to get workarea: probably your window manager does not support extended hints\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            if (enable_compose)
350                    IM = XOpenIM(display, NULL, NULL, NULL);
351    
352            xkeymap_init();
353            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),          wnd = XCreateWindow(display, RootWindowOfScreen(screen), 0, 0, wndwidth, wndheight,
390                              0, 0, width, height, 0, CopyFromParent,                              0, CopyFromParent, InputOutput, CopyFromParent,
391                              InputOutput, CopyFromParent,                              CWBackPixel | CWBackingStore | CWOverrideRedirect, &attribs);
                             CWBackingStore | CWBackPixel | 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 340  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;  
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)  
443          {          {
444                  error("XkbGetKeyboard failed.\n");                  XMaskEvent(display, VisibilityChangeMask, &xevent);
                 exit(0);  
445          }          }
446            while (xevent.type != VisibilityNotify);
447    
448          /* TODO: error texts... make them friendly. */          focused = False;
449          if (XkbSelectEvents          mouse_in_wnd = False;
             (display, xkb->device_spec, XkbAllEventsMask,  
              XkbAllEventsMask) == False)  
         {  
                 error("XkbSelectEvents failed.\n");  
                 exit(0);  
         }  
450    
451          xwin_get_numlock_mask();          /* 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  xwin_get_numlock_mask()  ui_destroy_window(void)
461  {  {
462          KeyCode numlockcode;          if (IC != NULL)
463          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);  
                 }  
         }  
   
         if (!numlock_modifier_mask)  
                 printf("WARNING: Failed to get a numlock modifier mapping.\n");  
464    
465            XDestroyWindow(display, wnd);
466  }  }
467    
468  void  void
469  ui_destroy_window()  xwin_toggle_fullscreen(void)
470  {  {
471          if (xkb != NULL)          Pixmap contents = 0;
                 XkbFreeKeyboard(xkb, XkbAllControlsMask, True);  
472    
473          if (ownbackstore)          if (!ownbackstore)
474                  XFreePixmap(display, backstore);          {
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          XFreeGC(display, gc);          ui_destroy_window();
481          XDestroyWindow(display, wnd);          fullscreen = !fullscreen;
482          XCloseDisplay(display);          ui_create_window();
483          display = NULL;  
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  static void  /* Process all events in Xlib queue
494  xwin_process_events()     Returns 0 after user quit, 1 otherwise */
495    static int
496    xwin_process_events(void)
497  {  {
498          XEvent xevent;          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          while (XCheckMaskEvent(display, ~0, &xevent))          while (XPending(display) > 0)
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 (xevent.type)                  switch (xevent.type)
522                  {                  {
523                          case KeyRelease:                          case ClientMessage:
524                                  flags = KBD_FLAG_DOWN | KBD_FLAG_UP;                                  /* the window manager told us to quit */
525                                  /* fall through */                                  if ((xevent.xclient.message_type == protocol_atom)
526                                        && (xevent.xclient.data.l[0] == kill_atom))
527                                            /* Quit */
528                                            return 0;
529                                    break;
530    
531                          case KeyPress:                          case KeyPress:
532                                  if (XkbTranslateKeyCode                                  if (IC != NULL)
533                                      (xkb, xevent.xkey.keycode,                                          /* Multi_key compatible version */
534                                       xevent.xkey.state, &tmpmods,                                  {
535                                       &keysym) == False)                                          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,  
                                                               xevent.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;
565    
566                                    ensure_remote_modifiers(ev_time, tr);
567    
568                                    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                                    ev_time = time(NULL);
579                                    if (handle_special_keys(keysym, xevent.xkey.state, ev_time, False))
580                                          break;                                          break;
581    
582                                  /* keep track of the modifiers -- needed for stickykeys... */                                  tr = xkeymap_translate_key(keysym,
583                                  if (xevent.type == KeyPress)                                                             xevent.xkey.keycode, xevent.xkey.state);
584                                          xwin_mod_press(xevent.xkey.state,  
585                                                         ev_time, scancode);                                  if (tr.scancode == 0)
586                                            break;
                                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE,  
                                                flags, scancode, 0);  
   
                                 if (xevent.type == KeyRelease)  
                                         xwin_mod_release(xevent.xkey.state,  
                                                          ev_time, scancode);  
587    
588                                    rdp_send_scancode(ev_time, RDP_KEYRELEASE, tr.scancode);
589                                  break;                                  break;
590    
591                          case ButtonPress:                          case ButtonPress:
# Line 499  xwin_process_events() Line 593  xwin_process_events()
593                                  /* fall through */                                  /* fall through */
594    
595                          case ButtonRelease:                          case ButtonRelease:
596                                  button = xkeymap_translate_button(xevent.                                  button = xkeymap_translate_button(xevent.xbutton.button);
                                                                   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);
                                                xevent.xbutton.x,  
                                                xevent.xbutton.y);  
602                                  break;                                  break;
603    
604                          case MotionNotify:                          case MotionNotify:
605                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  if (fullscreen && !focused)
606                                                 MOUSE_FLAG_MOVE,                                          XSetInputFocus(display, wnd, RevertToPointerRoot,
607                                                 xevent.xmotion.x,                                                         CurrentTime);
608                                                 xevent.xmotion.y);                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
609                                                   MOUSE_FLAG_MOVE, xevent.xmotion.x, xevent.xmotion.y);
610                                  break;                                  break;
611    
612                          case EnterNotify:                          case FocusIn:
613                                  XGrabKeyboard(display, wnd, True,                                  if (xevent.xfocus.mode == NotifyGrab)
614                                                GrabModeAsync, GrabModeAsync,                                          break;
615                                                CurrentTime);                                  focused = True;
616                                    XQueryPointer(display, wnd, &wdummy, &wdummy, &dummy, &dummy,
617                                                  &dummy, &dummy, &state);
618                                    reset_modifier_keys(state);
619                                    if (grab_keyboard && mouse_in_wnd)
620                                            XGrabKeyboard(display, wnd, True,
621                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
622                                    break;
623    
624                                  xwin_mod_update(xevent.xcrossing.state,                          case FocusOut:
625                                                  ev_time);                                  if (xevent.xfocus.mode == NotifyUngrab)
626                                            break;
627                                    focused = False;
628                                    if (xevent.xfocus.mode == NotifyWhileGrabbed)
629                                            XUngrabKeyboard(display, CurrentTime);
630                                    break;
631    
632                            case EnterNotify:
633                                    /* we only register for this event when in fullscreen mode */
634                                    /* or grab_keyboard */
635                                    mouse_in_wnd = True;
636                                    if (fullscreen)
637                                    {
638                                            XSetInputFocus(display, wnd, RevertToPointerRoot,
639                                                           CurrentTime);
640                                            break;
641                                    }
642                                    if (focused)
643                                            XGrabKeyboard(display, wnd, True,
644                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
645                                  break;                                  break;
646    
647                          case LeaveNotify:                          case LeaveNotify:
648                                    /* we only register for this event when grab_keyboard */
649                                    mouse_in_wnd = False;
650                                  XUngrabKeyboard(display, CurrentTime);                                  XUngrabKeyboard(display, CurrentTime);
651                                  break;                                  break;
652    
# Line 538  xwin_process_events() Line 657  xwin_process_events()
657                                            xevent.xexpose.height,                                            xevent.xexpose.height,
658                                            xevent.xexpose.x, xevent.xexpose.y);                                            xevent.xexpose.x, xevent.xexpose.y);
659                                  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);  
 }  
   
 void  
 xwin_mod_release(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 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))  
         {  
                 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;  
   
         }  
660    
661          if ((ControlMask & state)                          case MappingNotify:
662              && !((key_down_state & DControl1Mask)                                  /* Refresh keyboard mapping if it has changed. This is important for
663                   || (key_down_state & DControl2Mask)))                                     Xvnc, since it allocates keycodes dynamically */
664          {                                  if (xevent.xmapping.request == MappingKeyboard
665                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN,                                      || xevent.xmapping.request == MappingModifier)
666                                 0x1d, 0);                                          XRefreshKeyboardMapping(&xevent.xmapping);
                 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;  
667    
668          }                                  if (xevent.xmapping.request == MappingModifier)
669                                    {
670          if ((Mod2Mask & state) && !(key_down_state & DMod2Mask))                                          XFreeModifiermap(mod_map);
671          {                                          mod_map = XGetModifierMapping(display);
672                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN,                                  }
673                                 0xb8, 0);                                  break;
                 key_down_state |= DMod2Mask;  
674    
675                    }
676          }          }
677            /* Keep going */
678            return 1;
679  }  }
680    
681  void  /* Returns 0 after user quit, 1 otherwise */
682    int
683  ui_select(int rdp_socket)  ui_select(int rdp_socket)
684  {  {
685          int n = (rdp_socket > x_socket) ? rdp_socket + 1 : x_socket + 1;          int n = (rdp_socket > x_socket) ? rdp_socket + 1 : x_socket + 1;
# Line 724  ui_select(int rdp_socket) Line 689  ui_select(int rdp_socket)
689    
690          while (True)          while (True)
691          {          {
692                    /* Process any events already waiting */
693                    if (!xwin_process_events())
694                            /* User quit */
695                            return 0;
696    
697                  FD_ZERO(&rfds);                  FD_ZERO(&rfds);
698                  FD_SET(rdp_socket, &rfds);                  FD_SET(rdp_socket, &rfds);
699                  if (display != NULL)                  FD_SET(x_socket, &rfds);
                 {  
                         FD_SET(x_socket, &rfds);  
                         XFlush(display);  
                 }  
700    
701                  switch (select(n, &rfds, NULL, NULL, NULL))                  switch (select(n, &rfds, NULL, NULL, NULL))
702                  {                  {
# Line 741  ui_select(int rdp_socket) Line 707  ui_select(int rdp_socket)
707                                  continue;                                  continue;
708                  }                  }
709    
                 if (FD_ISSET(x_socket, &rfds))  
                         xwin_process_events();  
   
710                  if (FD_ISSET(rdp_socket, &rfds))                  if (FD_ISSET(rdp_socket, &rfds))
711                          return;                          return 1;
712          }          }
713  }  }
714    
# Line 764  ui_create_bitmap(int width, int height, Line 727  ui_create_bitmap(int width, int height,
727    
728          tdata = (owncolmap ? data : translate_image(width, height, data));          tdata = (owncolmap ? data : translate_image(width, height, data));
729          bitmap = XCreatePixmap(display, wnd, width, height, depth);          bitmap = XCreatePixmap(display, wnd, width, height, depth);
730          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
731                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
732    
733          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);
734    
# Line 776  ui_create_bitmap(int width, int height, Line 739  ui_create_bitmap(int width, int height,
739  }  }
740    
741  void  void
742  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)  
743  {  {
744          XImage *image;          XImage *image;
745          uint8 *tdata;          uint8 *tdata;
746    
747          tdata = (owncolmap ? data : translate_image(width, height, data));          tdata = (owncolmap ? data : translate_image(width, height, data));
748          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
749                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
750    
751          if (ownbackstore)          if (ownbackstore)
752          {          {
# Line 820  ui_create_glyph(int width, int height, u Line 782  ui_create_glyph(int width, int height, u
782          bitmap = XCreatePixmap(display, wnd, width, height, 1);          bitmap = XCreatePixmap(display, wnd, width, height, 1);
783          gc = XCreateGC(display, bitmap, 0, NULL);          gc = XCreateGC(display, bitmap, 0, NULL);
784    
785          image = XCreateImage(display, visual, 1, ZPixmap, 0,          image = XCreateImage(display, visual, 1, ZPixmap, 0, (char *) data,
786                               data, width, height, 8, scanline);                               width, height, 8, scanline);
787          image->byte_order = MSBFirst;          image->byte_order = MSBFirst;
788          image->bitmap_bit_order = MSBFirst;          image->bitmap_bit_order = MSBFirst;
789          XInitImage(image);          XInitImage(image);
# Line 840  ui_destroy_glyph(HGLYPH glyph) Line 802  ui_destroy_glyph(HGLYPH glyph)
802  }  }
803    
804  HCURSOR  HCURSOR
805  ui_create_cursor(unsigned int x, unsigned int y, int width,  ui_create_cursor(unsigned int x, unsigned int y, int width, int height,
806                   int height, uint8 * andmask, uint8 * xormask)                   uint8 * andmask, uint8 * xormask)
807  {  {
808          HGLYPH maskglyph, cursorglyph;          HGLYPH maskglyph, cursorglyph;
809          XColor bg, fg;          XColor bg, fg;
# Line 899  ui_create_cursor(unsigned int x, unsigne Line 861  ui_create_cursor(unsigned int x, unsigne
861          cursorglyph = ui_create_glyph(width, height, cursor);          cursorglyph = ui_create_glyph(width, height, cursor);
862          maskglyph = ui_create_glyph(width, height, mask);          maskglyph = ui_create_glyph(width, height, mask);
863    
864          xcursor = XCreatePixmapCursor(display, (Pixmap) cursorglyph,          xcursor =
865                                        (Pixmap) maskglyph, &fg, &bg, x, y);                  XCreatePixmapCursor(display, (Pixmap) cursorglyph,
866                                        (Pixmap) maskglyph, &fg, &bg, x, y);
867    
868          ui_destroy_glyph(maskglyph);          ui_destroy_glyph(maskglyph);
869          ui_destroy_glyph(cursorglyph);          ui_destroy_glyph(cursorglyph);
# Line 912  ui_create_cursor(unsigned int x, unsigne Line 875  ui_create_cursor(unsigned int x, unsigne
875  void  void
876  ui_set_cursor(HCURSOR cursor)  ui_set_cursor(HCURSOR cursor)
877  {  {
878          XDefineCursor(display, wnd, (Cursor) cursor);          current_cursor = (Cursor) cursor;
879            XDefineCursor(display, wnd, current_cursor);
880  }  }
881    
882  void  void
# Line 927  ui_destroy_cursor(HCURSOR cursor) Line 891  ui_destroy_cursor(HCURSOR cursor)
891                  (xc)->blue  = ((c)->blue  << 8) | (c)->blue; \                  (xc)->blue  = ((c)->blue  << 8) | (c)->blue; \
892                  (xc)->flags = DoRed | DoGreen | DoBlue;                  (xc)->flags = DoRed | DoGreen | DoBlue;
893    
894    
895  HCOLOURMAP  HCOLOURMAP
896  ui_create_colourmap(COLOURMAP * colours)  ui_create_colourmap(COLOURMAP * colours)
897  {  {
898          COLOURENTRY *entry;          COLOURENTRY *entry;
899          int i, ncolours = colours->ncolours;          int i, ncolours = colours->ncolours;
900            if (!owncolmap)
901            {
902                    uint32 *map = xmalloc(sizeof(*colmap) * ncolours);
903                    XColor xentry;
904                    XColor xc_cache[256];
905                    uint32 colour;
906                    int colLookup = 256;
907                    for (i = 0; i < ncolours; i++)
908                    {
909                            entry = &colours->colours[i];
910                            MAKE_XCOLOR(&xentry, entry);
911    
912                            if (XAllocColor(display, xcolmap, &xentry) == 0)
913                            {
914                                    /* Allocation failed, find closest match. */
915                                    int j = 256;
916                                    int nMinDist = 3 * 256 * 256;
917                                    long nDist = nMinDist;
918    
919                                    /* only get the colors once */
920                                    while (colLookup--)
921                                    {
922                                            xc_cache[colLookup].pixel = colLookup;
923                                            xc_cache[colLookup].red = xc_cache[colLookup].green =
924                                                    xc_cache[colLookup].blue = 0;
925                                            xc_cache[colLookup].flags = 0;
926                                            XQueryColor(display,
927                                                        DefaultColormap(display,
928                                                                        DefaultScreen(display)),
929                                                        &xc_cache[colLookup]);
930                                    }
931                                    colLookup = 0;
932    
933                                    /* approximate the pixel */
934                                    while (j--)
935                                    {
936                                            if (xc_cache[j].flags)
937                                            {
938                                                    nDist = ((long) (xc_cache[j].red >> 8) -
939                                                             (long) (xentry.red >> 8)) *
940                                                            ((long) (xc_cache[j].red >> 8) -
941                                                             (long) (xentry.red >> 8)) +
942                                                            ((long) (xc_cache[j].green >> 8) -
943                                                             (long) (xentry.green >> 8)) *
944                                                            ((long) (xc_cache[j].green >> 8) -
945                                                             (long) (xentry.green >> 8)) +
946                                                            ((long) (xc_cache[j].blue >> 8) -
947                                                             (long) (xentry.blue >> 8)) *
948                                                            ((long) (xc_cache[j].blue >> 8) -
949                                                             (long) (xentry.blue >> 8));
950                                            }
951                                            if (nDist < nMinDist)
952                                            {
953                                                    nMinDist = nDist;
954                                                    xentry.pixel = j;
955                                            }
956                                    }
957                            }
958                            colour = xentry.pixel;
959    
960          if (owncolmap)                          /* update our cache */
961                            if (xentry.pixel < 256)
962                            {
963                                    xc_cache[xentry.pixel].red = xentry.red;
964                                    xc_cache[xentry.pixel].green = xentry.green;
965                                    xc_cache[xentry.pixel].blue = xentry.blue;
966    
967                            }
968    
969    
970                            /* byte swap here to make translate_image faster */
971                            map[i] = translate_colour(colour);
972                    }
973                    return map;
974            }
975            else
976          {          {
977                  XColor *xcolours, *xentry;                  XColor *xcolours, *xentry;
978                  Colormap map;                  Colormap map;
# Line 953  ui_create_colourmap(COLOURMAP * colours) Line 992  ui_create_colourmap(COLOURMAP * colours)
992                  xfree(xcolours);                  xfree(xcolours);
993                  return (HCOLOURMAP) map;                  return (HCOLOURMAP) map;
994          }          }
         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;  
         }  
995  }  }
996    
997  void  void
998  ui_destroy_colourmap(HCOLOURMAP map)  ui_destroy_colourmap(HCOLOURMAP map)
999  {  {
1000          if (owncolmap)          if (!owncolmap)
                 XFreeColormap(display, (Colormap) map);  
         else  
1001                  xfree(map);                  xfree(map);
1002            else
1003                    XFreeColormap(display, (Colormap) map);
1004  }  }
1005    
1006  void  void
1007  ui_set_colourmap(HCOLOURMAP map)  ui_set_colourmap(HCOLOURMAP map)
1008  {  {
1009          if (owncolmap)          if (!owncolmap)
                 XSetWindowColormap(display, wnd, (Colormap) map);  
         else  
1010                  colmap = map;                  colmap = map;
1011            else
1012                    XSetWindowColormap(display, wnd, (Colormap) map);
1013  }  }
1014    
1015  void  void
# Line 1008  ui_set_clip(int x, int y, int cx, int cy Line 1025  ui_set_clip(int x, int y, int cx, int cy
1025  }  }
1026    
1027  void  void
1028  ui_reset_clip()  ui_reset_clip(void)
1029  {  {
1030          XRectangle rect;          XRectangle rect;
1031    
# Line 1020  ui_reset_clip() Line 1037  ui_reset_clip()
1037  }  }
1038    
1039  void  void
1040  ui_bell()  ui_bell(void)
1041  {  {
1042          XBell(display, 0);          XBell(display, 0);
1043  }  }
# Line 1060  ui_patblt(uint8 opcode, Line 1077  ui_patblt(uint8 opcode,
1077                          SET_BACKGROUND(fgcolour);                          SET_BACKGROUND(fgcolour);
1078                          XSetFillStyle(display, gc, FillOpaqueStippled);                          XSetFillStyle(display, gc, FillOpaqueStippled);
1079                          XSetStipple(display, gc, fill);                          XSetStipple(display, gc, fill);
1080                          XSetTSOrigin(display, gc, brush->xorigin,                          XSetTSOrigin(display, gc, brush->xorigin, brush->yorigin);
                                      brush->yorigin);  
1081    
1082                          FILL_RECTANGLE(x, y, cx, cy);                          FILL_RECTANGLE(x, y, cx, cy);
1083    
1084                          XSetFillStyle(display, gc, FillSolid);                          XSetFillStyle(display, gc, FillSolid);
1085                            XSetTSOrigin(display, gc, 0, 0);
1086                          ui_destroy_glyph((HGLYPH) fill);                          ui_destroy_glyph((HGLYPH) fill);
1087                          break;                          break;
1088    
# Line 1084  ui_screenblt(uint8 opcode, Line 1101  ui_screenblt(uint8 opcode,
1101          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1102          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);
1103          if (ownbackstore)          if (ownbackstore)
1104                  XCopyArea(display, backstore, backstore, gc, srcx, srcy,                  XCopyArea(display, backstore, backstore, gc, srcx, srcy, cx, cy, x, y);
                           cx, cy, x, y);  
1105          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1106  }  }
1107    
# Line 1097  ui_memblt(uint8 opcode, Line 1113  ui_memblt(uint8 opcode,
1113          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1114          XCopyArea(display, (Pixmap) src, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, (Pixmap) src, wnd, gc, srcx, srcy, cx, cy, x, y);
1115          if (ownbackstore)          if (ownbackstore)
1116                  XCopyArea(display, (Pixmap) src, backstore, gc, srcx, srcy,                  XCopyArea(display, (Pixmap) src, backstore, gc, srcx, srcy, cx, cy, x, y);
                           cx, cy, x, y);  
1117          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1118  }  }
1119    
# Line 1115  ui_triblt(uint8 opcode, Line 1130  ui_triblt(uint8 opcode,
1130          {          {
1131                  case 0x69:      /* PDSxxn */                  case 0x69:      /* PDSxxn */
1132                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);
1133                          ui_patblt(ROP2_NXOR, x, y, cx, cy,                          ui_patblt(ROP2_NXOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1134                          break;                          break;
1135    
1136                  case 0xb8:      /* PSDPxax */                  case 0xb8:      /* PSDPxax */
1137                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1138                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);
1139                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1140                          break;                          break;
1141    
1142                  case 0xc0:      /* PSa */                  case 0xc0:      /* PSa */
1143                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);
1144                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour,                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   fgcolour);  
1145                          break;                          break;
1146    
1147                  default:                  default:
# Line 1161  ui_rect( Line 1172  ui_rect(
1172          FILL_RECTANGLE(x, y, cx, cy);          FILL_RECTANGLE(x, y, cx, cy);
1173  }  }
1174    
1175    /* warning, this function only draws on wnd or backstore, not both */
1176  void  void
1177  ui_draw_glyph(int mixmode,  ui_draw_glyph(int mixmode,
1178                /* dest */ int x, int y, int cx, int cy,                /* dest */ int x, int y, int cx, int cy,
1179                /* src */ HGLYPH glyph, int srcx, int srcy, int bgcolour,                /* src */ HGLYPH glyph, int srcx, int srcy,
1180                int fgcolour)                int bgcolour, int fgcolour)
1181  {  {
1182          SET_FOREGROUND(fgcolour);          SET_FOREGROUND(fgcolour);
1183          SET_BACKGROUND(bgcolour);          SET_BACKGROUND(bgcolour);
1184    
1185          XSetFillStyle(display, gc, (mixmode == MIX_TRANSPARENT)          XSetFillStyle(display, gc,
1186                        ? FillStippled : FillOpaqueStippled);                        (mixmode == MIX_TRANSPARENT) ? FillStippled : FillOpaqueStippled);
1187          XSetStipple(display, gc, (Pixmap) glyph);          XSetStipple(display, gc, (Pixmap) glyph);
1188          XSetTSOrigin(display, gc, x, y);          XSetTSOrigin(display, gc, x, y);
1189    
1190          FILL_RECTANGLE(x, y, cx, cy);          FILL_RECTANGLE_BACKSTORE(x, y, cx, cy);
1191    
1192          XSetFillStyle(display, gc, FillSolid);          XSetFillStyle(display, gc, FillSolid);
1193  }  }
# Line 1189  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 1214  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 1226  ui_draw_text(uint8 font, uint8 flags, in Line 1239  ui_draw_text(uint8 font, uint8 flags, in
1239    
1240          if (boxcx > 1)          if (boxcx > 1)
1241          {          {
1242                  FILL_RECTANGLE(boxx, boxy, boxcx, boxcy);                  FILL_RECTANGLE_BACKSTORE(boxx, boxy, boxcx, boxcy);
1243          }          }
1244          else if (mixmode == MIX_OPAQUE)          else if (mixmode == MIX_OPAQUE)
1245          {          {
1246                  FILL_RECTANGLE(clipx, clipy, clipcx, clipcy);                  FILL_RECTANGLE_BACKSTORE(clipx, clipy, clipcx, clipcy);
1247          }          }
1248    
1249          /* Paint text, character by character */          /* Paint text, character by character */
# Line 1240  ui_draw_text(uint8 font, uint8 flags, in Line 1253  ui_draw_text(uint8 font, uint8 flags, in
1253                  {                  {
1254                          case 0xff:                          case 0xff:
1255                                  if (i + 2 < length)                                  if (i + 2 < length)
1256                                          cache_put_text(text[i + 1], text,                                          cache_put_text(text[i + 1], text, text[i + 2]);
                                                        text[i + 2]);  
1257                                  else                                  else
1258                                  {                                  {
1259                                          error("this shouldn't be happening\n");                                          error("this shouldn't be happening\n");
1260                                          break;                                          exit(1);
1261                                  }                                  }
1262                                  /* this will move pointer from start to first character after FF command */                                  /* this will move pointer from start to first character after FF command */
1263                                  length -= i + 3;                                  length -= i + 3;
# Line 1258  ui_draw_text(uint8 font, uint8 flags, in Line 1270  ui_draw_text(uint8 font, uint8 flags, in
1270                                  if (entry != NULL)                                  if (entry != NULL)
1271                                  {                                  {
1272                                          if ((((uint8 *) (entry->data))[1] ==                                          if ((((uint8 *) (entry->data))[1] ==
1273                                               0)                                               0) && (!(flags & TEXT2_IMPLICIT_X)))
                                             && (!(flags & TEXT2_IMPLICIT_X)))  
1274                                          {                                          {
1275                                                  if (flags & TEXT2_VERTICAL)                                                  if (flags & TEXT2_VERTICAL)
1276                                                          y += text[i + 2];                                                          y += text[i + 2];
1277                                                  else                                                  else
1278                                                          x += text[i + 2];                                                          x += text[i + 2];
1279                                          }                                          }
                                         if (i + 2 < length)  
                                                 i += 3;  
                                         else  
                                                 i += 2;  
                                         length -= i;  
                                         /* this will move pointer from start to first character after FE command */  
                                         text = &(text[i]);  
                                         i = 0;  
1280                                          for (j = 0; j < entry->size; j++)                                          for (j = 0; j < entry->size; j++)
1281                                                  DO_GLYPH(((uint8 *) (entry->                                                  DO_GLYPH(((uint8 *) (entry->data)), j);
                                                                      data)),  
                                                          j);  
1282                                  }                                  }
1283                                    if (i + 2 < length)
1284                                            i += 3;
1285                                    else
1286                                            i += 2;
1287                                    length -= i;
1288                                    /* this will move pointer from start to first character after FE command */
1289                                    text = &(text[i]);
1290                                    i = 0;
1291                                  break;                                  break;
1292    
1293                          default:                          default:
# Line 1287  ui_draw_text(uint8 font, uint8 flags, in Line 1296  ui_draw_text(uint8 font, uint8 flags, in
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 1299  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 1329  ui_desktop_restore(uint32 offset, int x, Line 1342  ui_desktop_restore(uint32 offset, int x,
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.64  
changed lines
  Added in v.297

  ViewVC Help
Powered by ViewVC 1.1.26