/[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 278 by jsorg71, Mon Nov 25 18:23:44 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 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 74  static unsigned int key_down_state = 0; Line 79  static unsigned int key_down_state = 0;
79  }  }
80    
81  /* colour maps */  /* colour maps */
 static BOOL owncolmap;  
82  static Colormap xcolmap;  static Colormap xcolmap;
 static uint32 white;  
83  static uint32 *colmap;  static uint32 *colmap;
84    
85  #define TRANSLATE(col)          ( owncolmap ? col : translate_colour(colmap[col]) )  #define SET_FOREGROUND(col)     XSetForeground(display, gc, translate_colour(colmap[col]));
86  #define SET_FOREGROUND(col)     XSetForeground(display, gc, TRANSLATE(col));  #define SET_BACKGROUND(col)     XSetBackground(display, gc, translate_colour(colmap[col]));
 #define SET_BACKGROUND(col)     XSetBackground(display, gc, TRANSLATE(col));  
87    
88  static int rop2_map[] = {  static int rop2_map[] = {
89          GXclear,                /* 0 */          GXclear,                /* 0 */
# Line 105  static int rop2_map[] = { Line 107  static int rop2_map[] = {
107  #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]); }
108  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }
109    
110  void xwin_get_numlock_mask();  void
111  void xwin_mod_update(uint32 state, uint32 ev_time);  mwm_hide_decorations(void)
112  void xwin_mod_release(uint32 state, uint32 ev_time, uint32 scancode);  {
113  void xwin_mod_press(uint32 state, uint32 ev_time, uint32 scancode);          PropMotifWmHints motif_hints;
114            Atom hintsatom;
115    
116            /* setup the property */
117            motif_hints.flags = MWM_HINTS_DECORATIONS;
118            motif_hints.decorations = 0;
119    
120            /* get the atom for the property */
121            hintsatom = XInternAtom(display, "_MOTIF_WM_HINTS", False);
122            if (!hintsatom)
123            {
124                    error("Failed to get atom _MOTIF_WM_HINTS\n");
125                    return;
126            }
127    
128            XChangeProperty(display, wnd, hintsatom, hintsatom, 32, PropModeReplace,
129                            (unsigned char *) &motif_hints, PROP_MOTIF_WM_HINTS_ELEMENTS);
130    }
131    
132  static void  static void
133  translate8(uint8 * data, uint8 * out, uint8 * end)  translate8(uint8 * data, uint8 * out, uint8 * end)
# Line 205  translate_colour(uint32 colour) Line 224  translate_colour(uint32 colour)
224  }  }
225    
226  BOOL  BOOL
227  ui_create_window(char *title)  get_key_state(unsigned int state, uint32 keysym)
228  {  {
229          XSetWindowAttributes attribs;          int modifierpos, key, keysymMask = 0;
230          XClassHint *classhints;          int offset;
         XSizeHints *sizehints;  
         unsigned long input_mask;  
         XPixmapFormatValues *pfm;  
         Screen *screen;  
         uint16 test;  
         int i;  
231    
232          int xkb_minor, xkb_major;          KeyCode keycode = XKeysymToKeycode(display, keysym);
         int xkb_event, xkb_error, xkb_reason;  
233    
234          /* 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");  
235                  return False;                  return False;
         }  
   
236    
237          display =          for (modifierpos = 0; modifierpos < 8; modifierpos++)
                 XkbOpenDisplay(NULL, &xkb_event, &xkb_error, &xkb_major,  
                                &xkb_minor, &xkb_reason);  
         switch (xkb_reason)  
238          {          {
239                  case XkbOD_BadLibraryVersion:                  offset = mod_map->max_keypermod * modifierpos;
240                          error("XkbOD_BadLibraryVersion: XKB extensions in server and the library rdesktop is linked against aren't compatible with each other.\n");  
241                          break;                  for (key = 0; key < mod_map->max_keypermod; key++)
242                  case XkbOD_ConnectionRefused:                  {
243                          error("XkbOD_ConnectionRefused\n");                          if (mod_map->modifiermap[offset + key] == keycode)
244                          break;                                  keysymMask |= 1 << modifierpos;
245                  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;  
246          }          }
247    
248            return (state & keysymMask) ? True : False;
249    }
250    
251    BOOL
252    ui_init(void)
253    {
254            XPixmapFormatValues *pfm;
255            uint16 test;
256            int i;
257    
258            display = XOpenDisplay(NULL);
259          if (display == NULL)          if (display == NULL)
260          {          {
261                  error("Failed to open display\n");                  error("Failed to open display: %s\n", XDisplayName(NULL));
262                  return False;                  return False;
263          }          }
264    
# Line 269  ui_create_window(char *title) Line 274  ui_create_window(char *title)
274                     desirable, e.g. 24 bits->32 bits. */                     desirable, e.g. 24 bits->32 bits. */
275                  while (i--)                  while (i--)
276                  {                  {
277                          if ((pfm[i].depth == depth)                          if ((pfm[i].depth == depth) && (pfm[i].bits_per_pixel > bpp))
                             && (pfm[i].bits_per_pixel > bpp))  
278                          {                          {
279                                  bpp = pfm[i].bits_per_pixel;                                  bpp = pfm[i].bits_per_pixel;
280                          }                          }
# Line 285  ui_create_window(char *title) Line 289  ui_create_window(char *title)
289                  return False;                  return False;
290          }          }
291    
292          if (depth <= 8)          xcolmap = DefaultColormapOfScreen(screen);
293                  owncolmap = True;          gc = XCreateGC(display, RootWindowOfScreen(screen), 0, NULL);
294          else  
295                  xcolmap = DefaultColormapOfScreen(screen);          if (DoesBackingStore(screen) != Always)
296                    ownbackstore = True;
297    
298          test = 1;          test = 1;
299          host_be = !(BOOL) (*(uint8 *) (&test));          host_be = !(BOOL) (*(uint8 *) (&test));
300          xserver_be = (ImageByteOrder(display) == MSBFirst);          xserver_be = (ImageByteOrder(display) == MSBFirst);
301    
302          white = WhitePixelOfScreen(screen);          if ((width == 0) || (height == 0))
303          attribs.background_pixel = BlackPixelOfScreen(screen);          {
304          attribs.backing_store = DoesBackingStore(screen);                  /* Fetch geometry from _NET_WORKAREA */
305                    uint32 xpos, ypos;
306    
307          if (attribs.backing_store == NotUseful)                  if (get_current_workarea(&xpos, &ypos, &width, &height) < 0)
308                  ownbackstore = True;                  {
309                            error("Failed to get workarea.\n");
310                            error("Perhaps your window manager does not support EWMH?\n");
311                            error("Defaulting to geometry 800x600\n");
312                            width = 800;
313                            height = 600;
314                    }
315            }
316    
317          if (fullscreen)          if (fullscreen)
318          {          {
                 attribs.override_redirect = True;  
319                  width = WidthOfScreen(screen);                  width = WidthOfScreen(screen);
320                  height = HeightOfScreen(screen);                  height = HeightOfScreen(screen);
321          }          }
322          else  
323            /* make sure width is a multiple of 4 */
324            width = (width + 3) & ~3;
325    
326            if (ownbackstore)
327          {          {
328                  attribs.override_redirect = False;                  backstore =
329                            XCreatePixmap(display, RootWindowOfScreen(screen), width, height, depth);
330    
331                    /* clear to prevent rubbish being exposed at startup */
332                    XSetForeground(display, gc, BlackPixelOfScreen(screen));
333                    XFillRectangle(display, backstore, gc, 0, 0, width, height);
334          }          }
335    
336          width = (width + 3) & ~3;       /* make width a multiple of 32 bits */          mod_map = XGetModifierMapping(display);
337    
338            if (enable_compose)
339                    IM = XOpenIM(display, NULL, NULL, NULL);
340    
341          wnd = XCreateWindow(display, RootWindowOfScreen(screen),          xkeymap_init();
342                              0, 0, width, height, 0, CopyFromParent,          return True;
343                              InputOutput, CopyFromParent,  }
344                              CWBackingStore | CWBackPixel | CWOverrideRedirect,  
345                              &attribs);  void
346    ui_deinit(void)
347    {
348            if (IM != NULL)
349                    XCloseIM(IM);
350    
351            XFreeModifiermap(mod_map);
352    
353            if (ownbackstore)
354                    XFreePixmap(display, backstore);
355    
356            XFreeGC(display, gc);
357            XCloseDisplay(display);
358            display = NULL;
359    }
360    
361    BOOL
362    ui_create_window(void)
363    {
364            XSetWindowAttributes attribs;
365            XClassHint *classhints;
366            XSizeHints *sizehints;
367            int wndwidth, wndheight;
368            long input_mask, ic_input_mask;
369            XEvent xevent;
370    
371            wndwidth = fullscreen ? WidthOfScreen(screen) : width;
372            wndheight = fullscreen ? HeightOfScreen(screen) : height;
373    
374            attribs.background_pixel = BlackPixelOfScreen(screen);
375            attribs.backing_store = ownbackstore ? NotUseful : Always;
376            attribs.override_redirect = fullscreen;
377    
378            wnd = XCreateWindow(display, RootWindowOfScreen(screen), 0, 0, wndwidth, wndheight,
379                                0, CopyFromParent, InputOutput, CopyFromParent,
380                                CWBackPixel | CWBackingStore | CWOverrideRedirect, &attribs);
381    
382          XStoreName(display, wnd, title);          XStoreName(display, wnd, title);
383    
384            if (hide_decorations)
385                    mwm_hide_decorations();
386    
387          classhints = XAllocClassHint();          classhints = XAllocClassHint();
388          if (classhints != NULL)          if (classhints != NULL)
389          {          {
# Line 340  ui_create_window(char *title) Line 402  ui_create_window(char *title)
402                  XFree(sizehints);                  XFree(sizehints);
403          }          }
404    
405          xkeymap_init();          input_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask |
406                    VisibilityChangeMask | FocusChangeMask;
407    
         input_mask = KeyPressMask | KeyReleaseMask |  
                 ButtonPressMask | ButtonReleaseMask |  
                 EnterWindowMask | LeaveWindowMask;  
408          if (sendmotion)          if (sendmotion)
409                  input_mask |= PointerMotionMask;                  input_mask |= PointerMotionMask;
   
410          if (ownbackstore)          if (ownbackstore)
411                  input_mask |= ExposureMask;                  input_mask |= ExposureMask;
412            if (fullscreen || grab_keyboard)
413                    input_mask |= EnterWindowMask;
414            if (grab_keyboard)
415                    input_mask |= LeaveWindowMask;
416    
417            if (IM != NULL)
418            {
419                    IC = XCreateIC(IM, XNInputStyle, (XIMPreeditNothing | XIMStatusNothing),
420                                   XNClientWindow, wnd, XNFocusWindow, wnd, NULL);
421    
422                    if ((IC != NULL)
423                        && (XGetICValues(IC, XNFilterEvents, &ic_input_mask, NULL) == NULL))
424                            input_mask |= ic_input_mask;
425            }
426    
427          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);  
   
428          XMapWindow(display, wnd);          XMapWindow(display, wnd);
429    
430          /* TODO: error texts... make them friendly. */          /* wait for VisibilityNotify */
431          xkb = XkbGetKeyboard(display, XkbAllComponentsMask, XkbUseCoreKbd);          do
         if ((int) xkb == BadAlloc || xkb == NULL)  
432          {          {
433                  error("XkbGetKeyboard failed.\n");                  XMaskEvent(display, VisibilityChangeMask, &xevent);
                 exit(0);  
434          }          }
435            while (xevent.type != VisibilityNotify);
436    
437          /* TODO: error texts... make them friendly. */          focused = False;
438          if (XkbSelectEvents          mouse_in_wnd = False;
             (display, xkb->device_spec, XkbAllEventsMask,  
              XkbAllEventsMask) == False)  
         {  
                 error("XkbSelectEvents failed.\n");  
                 exit(0);  
         }  
439    
440          xwin_get_numlock_mask();          /* handle the WM_DELETE_WINDOW protocol */
441            protocol_atom = XInternAtom(display, "WM_PROTOCOLS", True);
442            kill_atom = XInternAtom(display, "WM_DELETE_WINDOW", True);
443            XSetWMProtocols(display, wnd, &kill_atom, 1);
444    
445          return True;          return True;
446  }  }
447    
448  void  void
449  xwin_get_numlock_mask()  ui_destroy_window(void)
450  {  {
451          KeyCode numlockcode;          if (IC != NULL)
452          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");  
453    
454            XDestroyWindow(display, wnd);
455  }  }
456    
457  void  void
458  ui_destroy_window()  xwin_toggle_fullscreen(void)
459  {  {
460          if (xkb != NULL)          Pixmap contents = 0;
                 XkbFreeKeyboard(xkb, XkbAllControlsMask, True);  
461    
462          if (ownbackstore)          if (!ownbackstore)
463                  XFreePixmap(display, backstore);          {
464                    /* need to save contents of window */
465                    contents = XCreatePixmap(display, wnd, width, height, depth);
466                    XCopyArea(display, wnd, contents, gc, 0, 0, width, height, 0, 0);
467            }
468    
469          XFreeGC(display, gc);          ui_destroy_window();
470          XDestroyWindow(display, wnd);          fullscreen = !fullscreen;
471          XCloseDisplay(display);          ui_create_window();
472          display = NULL;  
473            XDefineCursor(display, wnd, current_cursor);
474    
475            if (!ownbackstore)
476            {
477                    XCopyArea(display, contents, wnd, gc, 0, 0, width, height, 0, 0);
478                    XFreePixmap(display, contents);
479            }
480  }  }
481    
482  static void  /* Process all events in Xlib queue
483  xwin_process_events()     Returns 0 after user quit, 1 otherwise */
484    static int
485    xwin_process_events(void)
486  {  {
487          XEvent xevent;          XEvent xevent;
   
488          KeySym keysym;          KeySym keysym;
         uint8 scancode;  
489          uint16 button, flags;          uint16 button, flags;
490          uint32 ev_time;          uint32 ev_time;
491          uint32 tmpmods;          key_translation tr;
492            char str[256];
493            Status status;
494            unsigned int state;
495            Window wdummy;
496            int dummy;
497    
498          while (XCheckMaskEvent(display, ~0, &xevent))          while (XPending(display) > 0)
499          {          {
500                  ev_time = time(NULL);                  XNextEvent(display, &xevent);
501    
502                    if ((IC != NULL) && (XFilterEvent(&xevent, None) == True))
503                    {
504                            DEBUG_KBD(("Filtering event\n"));
505                            continue;
506                    }
507    
508                  flags = 0;                  flags = 0;
509    
510                  switch (xevent.type)                  switch (xevent.type)
511                  {                  {
512                          case KeyRelease:                          case ClientMessage:
513                                  flags = KBD_FLAG_DOWN | KBD_FLAG_UP;                                  /* the window manager told us to quit */
514                                  /* fall through */                                  if ((xevent.xclient.message_type == protocol_atom)
515                                        && (xevent.xclient.data.l[0] == kill_atom))
516                                            /* Quit */
517                                            return 0;
518                                    break;
519    
520                          case KeyPress:                          case KeyPress:
521                                  if (XkbTranslateKeyCode                                  if (IC != NULL)
522                                      (xkb, xevent.xkey.keycode,                                          /* Multi_key compatible version */
523                                       xevent.xkey.state, &tmpmods,                                  {
524                                       &keysym) == False)                                          XmbLookupString(IC,
525                                                            (XKeyPressedEvent *) &
526                                                            xevent, str, sizeof(str), &keysym, &status);
527                                            if (!((status == XLookupKeySym) || (status == XLookupBoth)))
528                                            {
529                                                    error("XmbLookupString failed with status 0x%x\n",
530                                                          status);
531                                                    break;
532                                            }
533                                    }
534                                    else
535                                    {
536                                            /* Plain old XLookupString */
537                                            DEBUG_KBD(("\nNo input context, using XLookupString\n"));
538                                            XLookupString((XKeyEvent *) & xevent,
539                                                          str, sizeof(str), &keysym, NULL);
540                                    }
541    
542                                    DEBUG_KBD(("KeyPress for (keysym 0x%lx, %s)\n", keysym,
543                                               get_ksname(keysym)));
544    
545                                    ev_time = time(NULL);
546                                    if (handle_special_keys(keysym, xevent.xkey.state, ev_time, True))
547                                          break;                                          break;
                                 scancode =  
                                         xkeymap_translate_key(keysym,  
                                                               xevent.xkey.  
                                                               keycode,  
                                                               &flags);  
548    
549                                  if (scancode == 0)                                  tr = xkeymap_translate_key(keysym,
550                                                               xevent.xkey.keycode, xevent.xkey.state);
551    
552                                    if (tr.scancode == 0)
553                                          break;                                          break;
554    
555                                  /* keep track of the modifiers -- needed for stickykeys... */                                  ensure_remote_modifiers(ev_time, tr);
                                 if (xevent.type == KeyPress)  
                                         xwin_mod_press(xevent.xkey.state,  
                                                        ev_time, scancode);  
   
                                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE,  
                                                flags, scancode, 0);  
   
                                 if (xevent.type == KeyRelease)  
                                         xwin_mod_release(xevent.xkey.state,  
                                                          ev_time, scancode);  
556    
557                                    rdp_send_scancode(ev_time, RDP_KEYPRESS, tr.scancode);
558                                    break;
559    
560                            case KeyRelease:
561                                    XLookupString((XKeyEvent *) & xevent, str,
562                                                  sizeof(str), &keysym, NULL);
563    
564                                    DEBUG_KBD(("\nKeyRelease for (keysym 0x%lx, %s)\n", keysym,
565                                               get_ksname(keysym)));
566    
567                                    ev_time = time(NULL);
568                                    if (handle_special_keys(keysym, xevent.xkey.state, ev_time, False))
569                                            break;
570    
571                                    tr = xkeymap_translate_key(keysym,
572                                                               xevent.xkey.keycode, xevent.xkey.state);
573    
574                                    if (tr.scancode == 0)
575                                            break;
576    
577                                    rdp_send_scancode(ev_time, RDP_KEYRELEASE, tr.scancode);
578                                  break;                                  break;
579    
580                          case ButtonPress:                          case ButtonPress:
# Line 499  xwin_process_events() Line 582  xwin_process_events()
582                                  /* fall through */                                  /* fall through */
583    
584                          case ButtonRelease:                          case ButtonRelease:
585                                  button = xkeymap_translate_button(xevent.                                  button = xkeymap_translate_button(xevent.xbutton.button);
                                                                   xbutton.  
                                                                   button);  
586                                  if (button == 0)                                  if (button == 0)
587                                          break;                                          break;
588    
589                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
590                                                 flags | button,                                                 flags | button, xevent.xbutton.x, xevent.xbutton.y);
                                                xevent.xbutton.x,  
                                                xevent.xbutton.y);  
591                                  break;                                  break;
592    
593                          case MotionNotify:                          case MotionNotify:
594                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
595                                                 MOUSE_FLAG_MOVE,                                                 MOUSE_FLAG_MOVE, xevent.xmotion.x, xevent.xmotion.y);
                                                xevent.xmotion.x,  
                                                xevent.xmotion.y);  
596                                  break;                                  break;
597    
598                          case EnterNotify:                          case FocusIn:
599                                  XGrabKeyboard(display, wnd, True,                                  if (xevent.xfocus.mode == NotifyGrab)
600                                                GrabModeAsync, GrabModeAsync,                                          break;
601                                                CurrentTime);                                  focused = True;
602                                    XQueryPointer(display, wnd, &wdummy, &wdummy, &dummy, &dummy,
603                                                  &dummy, &dummy, &state);
604                                    reset_modifier_keys(state);
605                                    if (grab_keyboard && mouse_in_wnd)
606                                            XGrabKeyboard(display, wnd, True,
607                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
608                                    break;
609    
610                                  xwin_mod_update(xevent.xcrossing.state,                          case FocusOut:
611                                                  ev_time);                                  if (xevent.xfocus.mode == NotifyUngrab)
612                                            break;
613                                    focused = False;
614                                    if (xevent.xfocus.mode == NotifyWhileGrabbed)
615                                            XUngrabKeyboard(display, CurrentTime);
616                                    break;
617    
618                            case EnterNotify:
619                                    /* we only register for this event when in fullscreen mode */
620                                    /* or grab_keyboard */
621                                    mouse_in_wnd = True;
622                                    if (fullscreen)
623                                    {
624                                            XSetInputFocus(display, wnd, RevertToPointerRoot,
625                                                           CurrentTime);
626                                            break;
627                                    }
628                                    if (focused)
629                                            XGrabKeyboard(display, wnd, True,
630                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
631                                  break;                                  break;
632    
633                          case LeaveNotify:                          case LeaveNotify:
634                                    /* we only register for this event when grab_keyboard */
635                                    mouse_in_wnd = False;
636                                  XUngrabKeyboard(display, CurrentTime);                                  XUngrabKeyboard(display, CurrentTime);
637                                  break;                                  break;
638    
# Line 538  xwin_process_events() Line 643  xwin_process_events()
643                                            xevent.xexpose.height,                                            xevent.xexpose.height,
644                                            xevent.xexpose.x, xevent.xexpose.y);                                            xevent.xexpose.x, xevent.xexpose.y);
645                                  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;  
   
         }  
646    
647                            case MappingNotify:
648                                    /* Refresh keyboard mapping if it has changed. This is important for
649                                       Xvnc, since it allocates keycodes dynamically */
650                                    if (xevent.xmapping.request == MappingKeyboard
651                                        || xevent.xmapping.request == MappingModifier)
652                                            XRefreshKeyboardMapping(&xevent.xmapping);
653    
654          if ((ShiftMask & state)                                  if (xevent.xmapping.request == MappingModifier)
655              && !((key_down_state & DShift1Mask)                                  {
656                   || (key_down_state & DShift2Mask)))                                          XFreeModifiermap(mod_map);
657          {                                          mod_map = XGetModifierMapping(display);
658                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN,                                  }
659                                 0x2a, 0);                                  break;
                 key_down_state |= DShift1Mask;  
   
         }  
   
         if ((ControlMask & state)  
             && !((key_down_state & DControl1Mask)  
                  || (key_down_state & DControl2Mask)))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN,  
                                0x1d, 0);  
                 key_down_state |= DControl1Mask;  
   
         }  
   
         if ((Mod1Mask & state) && !(key_down_state & DMod1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN,  
                                0x38, 0);  
                 key_down_state |= DMod1Mask;  
   
         }  
   
         if ((Mod2Mask & state) && !(key_down_state & DMod2Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN,  
                                0xb8, 0);  
                 key_down_state |= DMod2Mask;  
660    
661                    }
662          }          }
663            /* Keep going */
664            return 1;
665  }  }
666    
667  void  /* Returns 0 after user quit, 1 otherwise */
668    int
669  ui_select(int rdp_socket)  ui_select(int rdp_socket)
670  {  {
671          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 675  ui_select(int rdp_socket)
675    
676          while (True)          while (True)
677          {          {
678                    /* Process any events already waiting */
679                    if (!xwin_process_events())
680                            /* User quit */
681                            return 0;
682    
683                  FD_ZERO(&rfds);                  FD_ZERO(&rfds);
684                  FD_SET(rdp_socket, &rfds);                  FD_SET(rdp_socket, &rfds);
685                  if (display != NULL)                  FD_SET(x_socket, &rfds);
                 {  
                         FD_SET(x_socket, &rfds);  
                         XFlush(display);  
                 }  
686    
687                  switch (select(n, &rfds, NULL, NULL, NULL))                  switch (select(n, &rfds, NULL, NULL, NULL))
688                  {                  {
# Line 741  ui_select(int rdp_socket) Line 693  ui_select(int rdp_socket)
693                                  continue;                                  continue;
694                  }                  }
695    
                 if (FD_ISSET(x_socket, &rfds))  
                         xwin_process_events();  
   
696                  if (FD_ISSET(rdp_socket, &rfds))                  if (FD_ISSET(rdp_socket, &rfds))
697                          return;                          return 1;
698          }          }
699  }  }
700    
# Line 762  ui_create_bitmap(int width, int height, Line 711  ui_create_bitmap(int width, int height,
711          Pixmap bitmap;          Pixmap bitmap;
712          uint8 *tdata;          uint8 *tdata;
713    
714          tdata = (owncolmap ? data : translate_image(width, height, data));          tdata = translate_image(width, height, data);
715          bitmap = XCreatePixmap(display, wnd, width, height, depth);          bitmap = XCreatePixmap(display, wnd, width, height, depth);
716          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
717                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
718    
719          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);
720    
721          XFree(image);          XFree(image);
722          if (!owncolmap)          xfree(tdata);
                 xfree(tdata);  
723          return (HBITMAP) bitmap;          return (HBITMAP) bitmap;
724  }  }
725    
726  void  void
727  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)  
728  {  {
729          XImage *image;          XImage *image;
730          uint8 *tdata;          uint8 *tdata;
731    
732          tdata = (owncolmap ? data : translate_image(width, height, data));          tdata = translate_image(width, height, data);
733          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
734                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
735    
736          if (ownbackstore)          if (ownbackstore)
737          {          {
# Line 797  ui_paint_bitmap(int x, int y, int cx, in Line 744  ui_paint_bitmap(int x, int y, int cx, in
744          }          }
745    
746          XFree(image);          XFree(image);
747          if (!owncolmap)          xfree(tdata);
                 xfree(tdata);  
748  }  }
749    
750  void  void
# Line 820  ui_create_glyph(int width, int height, u Line 766  ui_create_glyph(int width, int height, u
766          bitmap = XCreatePixmap(display, wnd, width, height, 1);          bitmap = XCreatePixmap(display, wnd, width, height, 1);
767          gc = XCreateGC(display, bitmap, 0, NULL);          gc = XCreateGC(display, bitmap, 0, NULL);
768    
769          image = XCreateImage(display, visual, 1, ZPixmap, 0,          image = XCreateImage(display, visual, 1, ZPixmap, 0, (char *) data,
770                               data, width, height, 8, scanline);                               width, height, 8, scanline);
771          image->byte_order = MSBFirst;          image->byte_order = MSBFirst;
772          image->bitmap_bit_order = MSBFirst;          image->bitmap_bit_order = MSBFirst;
773          XInitImage(image);          XInitImage(image);
# Line 840  ui_destroy_glyph(HGLYPH glyph) Line 786  ui_destroy_glyph(HGLYPH glyph)
786  }  }
787    
788  HCURSOR  HCURSOR
789  ui_create_cursor(unsigned int x, unsigned int y, int width,  ui_create_cursor(unsigned int x, unsigned int y, int width, int height,
790                   int height, uint8 * andmask, uint8 * xormask)                   uint8 * andmask, uint8 * xormask)
791  {  {
792          HGLYPH maskglyph, cursorglyph;          HGLYPH maskglyph, cursorglyph;
793          XColor bg, fg;          XColor bg, fg;
# Line 899  ui_create_cursor(unsigned int x, unsigne Line 845  ui_create_cursor(unsigned int x, unsigne
845          cursorglyph = ui_create_glyph(width, height, cursor);          cursorglyph = ui_create_glyph(width, height, cursor);
846          maskglyph = ui_create_glyph(width, height, mask);          maskglyph = ui_create_glyph(width, height, mask);
847    
848          xcursor = XCreatePixmapCursor(display, (Pixmap) cursorglyph,          xcursor =
849                                        (Pixmap) maskglyph, &fg, &bg, x, y);                  XCreatePixmapCursor(display, (Pixmap) cursorglyph,
850                                        (Pixmap) maskglyph, &fg, &bg, x, y);
851    
852          ui_destroy_glyph(maskglyph);          ui_destroy_glyph(maskglyph);
853          ui_destroy_glyph(cursorglyph);          ui_destroy_glyph(cursorglyph);
# Line 912  ui_create_cursor(unsigned int x, unsigne Line 859  ui_create_cursor(unsigned int x, unsigne
859  void  void
860  ui_set_cursor(HCURSOR cursor)  ui_set_cursor(HCURSOR cursor)
861  {  {
862          XDefineCursor(display, wnd, (Cursor) cursor);          current_cursor = (Cursor) cursor;
863            XDefineCursor(display, wnd, current_cursor);
864  }  }
865    
866  void  void
# Line 932  ui_create_colourmap(COLOURMAP * colours) Line 880  ui_create_colourmap(COLOURMAP * colours)
880  {  {
881          COLOURENTRY *entry;          COLOURENTRY *entry;
882          int i, ncolours = colours->ncolours;          int i, ncolours = colours->ncolours;
883            uint32 *map = xmalloc(sizeof(*colmap) * ncolours);
884          if (owncolmap)          XColor xentry;
885            XColor xc_cache[256];
886            uint32 colour;
887            int colLookup = 256;
888            for (i = 0; i < ncolours; i++)
889          {          {
890                  XColor *xcolours, *xentry;                  entry = &colours->colours[i];
891                  Colormap map;                  MAKE_XCOLOR(&xentry, entry);
892    
893                  xcolours = xmalloc(sizeof(XColor) * ncolours);                  if (XAllocColor(display, xcolmap, &xentry) == 0)
                 for (i = 0; i < ncolours; i++)  
894                  {                  {
895                          entry = &colours->colours[i];                          /* Allocation failed, find closest match. */
896                          xentry = &xcolours[i];                          int j = 256;
897                          xentry->pixel = i;                          int nMinDist = 3 * 256 * 256;
898                          MAKE_XCOLOR(xentry, entry);                          long nDist = nMinDist;
                 }  
899    
900                  map = XCreateColormap(display, wnd, visual, AllocAll);                          /* only get the colors once */
901                  XStoreColors(display, map, xcolours, ncolours);                          while (colLookup--)
902                            {
903                                    xc_cache[colLookup].pixel = colLookup;
904                                    xc_cache[colLookup].red = xc_cache[colLookup].green =
905                                            xc_cache[colLookup].blue = 0;
906                                    xc_cache[colLookup].flags = 0;
907                                    XQueryColor(display,
908                                                DefaultColormap(display, DefaultScreen(display)),
909                                                &xc_cache[colLookup]);
910                            }
911                            colLookup = 0;
912    
913                  xfree(xcolours);                          /* approximate the pixel */
914                  return (HCOLOURMAP) map;                          while (j--)
915          }                          {
916          else                                  if (xc_cache[j].flags)
917          {                                  {
918                  uint32 *map = xmalloc(sizeof(*colmap) * ncolours);                                          nDist = ((long) (xc_cache[j].red >> 8) -
919                  XColor xentry;                                                   (long) (xentry.red >> 8)) *
920                  uint32 colour;                                                  ((long) (xc_cache[j].red >> 8) -
921                                                     (long) (xentry.red >> 8)) +
922                                                    ((long) (xc_cache[j].green >> 8) -
923                                                     (long) (xentry.green >> 8)) *
924                                                    ((long) (xc_cache[j].green >> 8) -
925                                                     (long) (xentry.green >> 8)) +
926                                                    ((long) (xc_cache[j].blue >> 8) -
927                                                     (long) (xentry.blue >> 8)) *
928                                                    ((long) (xc_cache[j].blue >> 8) -
929                                                     (long) (xentry.blue >> 8));
930                                    }
931                                    if (nDist < nMinDist)
932                                    {
933                                            nMinDist = nDist;
934                                            xentry.pixel = j;
935                                    }
936                            }
937                    }
938                    colour = xentry.pixel;
939    
940                  for (i = 0; i < ncolours; i++)                  /* update our cache */
941                    if (xentry.pixel < 256)
942                  {                  {
943                          entry = &colours->colours[i];                          xc_cache[xentry.pixel].red = xentry.red;
944                          MAKE_XCOLOR(&xentry, entry);                          xc_cache[xentry.pixel].green = xentry.green;
945                            xc_cache[xentry.pixel].blue = xentry.blue;
946    
                         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);  
947                  }                  }
948    
949                  return map;  
950                    /* byte swap here to make translate_image faster */
951                    map[i] = translate_colour(colour);
952          }          }
953    
954            return map;
955  }  }
956    
957  void  void
958  ui_destroy_colourmap(HCOLOURMAP map)  ui_destroy_colourmap(HCOLOURMAP map)
959  {  {
960          if (owncolmap)          xfree(map);
                 XFreeColormap(display, (Colormap) map);  
         else  
                 xfree(map);  
961  }  }
962    
963  void  void
964  ui_set_colourmap(HCOLOURMAP map)  ui_set_colourmap(HCOLOURMAP map)
965  {  {
966          if (owncolmap)          colmap = map;
                 XSetWindowColormap(display, wnd, (Colormap) map);  
         else  
                 colmap = map;  
967  }  }
968    
969  void  void
# Line 1008  ui_set_clip(int x, int y, int cx, int cy Line 979  ui_set_clip(int x, int y, int cx, int cy
979  }  }
980    
981  void  void
982  ui_reset_clip()  ui_reset_clip(void)
983  {  {
984          XRectangle rect;          XRectangle rect;
985    
# Line 1020  ui_reset_clip() Line 991  ui_reset_clip()
991  }  }
992    
993  void  void
994  ui_bell()  ui_bell(void)
995  {  {
996          XBell(display, 0);          XBell(display, 0);
997  }  }
# Line 1060  ui_patblt(uint8 opcode, Line 1031  ui_patblt(uint8 opcode,
1031                          SET_BACKGROUND(fgcolour);                          SET_BACKGROUND(fgcolour);
1032                          XSetFillStyle(display, gc, FillOpaqueStippled);                          XSetFillStyle(display, gc, FillOpaqueStippled);
1033                          XSetStipple(display, gc, fill);                          XSetStipple(display, gc, fill);
1034                          XSetTSOrigin(display, gc, brush->xorigin,                          XSetTSOrigin(display, gc, brush->xorigin, brush->yorigin);
                                      brush->yorigin);  
1035    
1036                          FILL_RECTANGLE(x, y, cx, cy);                          FILL_RECTANGLE(x, y, cx, cy);
1037    
1038                          XSetFillStyle(display, gc, FillSolid);                          XSetFillStyle(display, gc, FillSolid);
1039                            XSetTSOrigin(display, gc, 0, 0);
1040                          ui_destroy_glyph((HGLYPH) fill);                          ui_destroy_glyph((HGLYPH) fill);
1041                          break;                          break;
1042    
# Line 1084  ui_screenblt(uint8 opcode, Line 1055  ui_screenblt(uint8 opcode,
1055          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1056          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);
1057          if (ownbackstore)          if (ownbackstore)
1058                  XCopyArea(display, backstore, backstore, gc, srcx, srcy,                  XCopyArea(display, backstore, backstore, gc, srcx, srcy, cx, cy, x, y);
                           cx, cy, x, y);  
1059          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1060  }  }
1061    
# Line 1097  ui_memblt(uint8 opcode, Line 1067  ui_memblt(uint8 opcode,
1067          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1068          XCopyArea(display, (Pixmap) src, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, (Pixmap) src, wnd, gc, srcx, srcy, cx, cy, x, y);
1069          if (ownbackstore)          if (ownbackstore)
1070                  XCopyArea(display, (Pixmap) src, backstore, gc, srcx, srcy,                  XCopyArea(display, (Pixmap) src, backstore, gc, srcx, srcy, cx, cy, x, y);
                           cx, cy, x, y);  
1071          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1072  }  }
1073    
# Line 1115  ui_triblt(uint8 opcode, Line 1084  ui_triblt(uint8 opcode,
1084          {          {
1085                  case 0x69:      /* PDSxxn */                  case 0x69:      /* PDSxxn */
1086                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);
1087                          ui_patblt(ROP2_NXOR, x, y, cx, cy,                          ui_patblt(ROP2_NXOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1088                          break;                          break;
1089    
1090                  case 0xb8:      /* PSDPxax */                  case 0xb8:      /* PSDPxax */
1091                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1092                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);
1093                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1094                          break;                          break;
1095    
1096                  case 0xc0:      /* PSa */                  case 0xc0:      /* PSa */
1097                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);
1098                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour,                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   fgcolour);  
1099                          break;                          break;
1100    
1101                  default:                  default:
# Line 1161  ui_rect( Line 1126  ui_rect(
1126          FILL_RECTANGLE(x, y, cx, cy);          FILL_RECTANGLE(x, y, cx, cy);
1127  }  }
1128    
1129    /* warning, this function only draws on wnd or backstore, not both */
1130  void  void
1131  ui_draw_glyph(int mixmode,  ui_draw_glyph(int mixmode,
1132                /* dest */ int x, int y, int cx, int cy,                /* dest */ int x, int y, int cx, int cy,
1133                /* src */ HGLYPH glyph, int srcx, int srcy, int bgcolour,                /* src */ HGLYPH glyph, int srcx, int srcy,
1134                int fgcolour)                int bgcolour, int fgcolour)
1135  {  {
1136          SET_FOREGROUND(fgcolour);          SET_FOREGROUND(fgcolour);
1137          SET_BACKGROUND(bgcolour);          SET_BACKGROUND(bgcolour);
1138    
1139          XSetFillStyle(display, gc, (mixmode == MIX_TRANSPARENT)          XSetFillStyle(display, gc,
1140                        ? FillStippled : FillOpaqueStippled);                        (mixmode == MIX_TRANSPARENT) ? FillStippled : FillOpaqueStippled);
1141          XSetStipple(display, gc, (Pixmap) glyph);          XSetStipple(display, gc, (Pixmap) glyph);
1142          XSetTSOrigin(display, gc, x, y);          XSetTSOrigin(display, gc, x, y);
1143    
1144          FILL_RECTANGLE(x, y, cx, cy);          if (ownbackstore)
1145                    XFillRectangle(display, backstore, gc, x, y, cx, cy);
1146            else
1147                    XFillRectangle(display, wnd, gc, x, y, cx, cy);
1148    
1149          XSetFillStyle(display, gc, FillSolid);          XSetFillStyle(display, gc, FillSolid);
1150  }  }
# Line 1189  ui_draw_glyph(int mixmode, Line 1158  ui_draw_glyph(int mixmode,
1158        if ((xyoffset & 0x80))\        if ((xyoffset & 0x80))\
1159          {\          {\
1160            if (flags & TEXT2_VERTICAL) \            if (flags & TEXT2_VERTICAL) \
1161              y += ttext[++idx] | (ttext[++idx] << 8);\              y += ttext[idx+1] | (ttext[idx+2] << 8);\
1162            else\            else\
1163              x += ttext[++idx] | (ttext[++idx] << 8);\              x += ttext[idx+1] | (ttext[idx+2] << 8);\
1164              idx += 2;\
1165          }\          }\
1166        else\        else\
1167          {\          {\
# Line 1214  ui_draw_glyph(int mixmode, Line 1184  ui_draw_glyph(int mixmode,
1184    
1185  void  void
1186  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,
1187               int clipx, int clipy, int clipcx, int clipcy, int boxx,               int clipx, int clipy, int clipcx, int clipcy,
1188               int boxy, int boxcx, int boxcy, int bgcolour,               int boxx, int boxy, int boxcx, int boxcy, int bgcolour,
1189               int fgcolour, uint8 * text, uint8 length)               int fgcolour, uint8 * text, uint8 length)
1190  {  {
1191          FONTGLYPH *glyph;          FONTGLYPH *glyph;
# Line 1240  ui_draw_text(uint8 font, uint8 flags, in Line 1210  ui_draw_text(uint8 font, uint8 flags, in
1210                  {                  {
1211                          case 0xff:                          case 0xff:
1212                                  if (i + 2 < length)                                  if (i + 2 < length)
1213                                          cache_put_text(text[i + 1], text,                                          cache_put_text(text[i + 1], text, text[i + 2]);
                                                        text[i + 2]);  
1214                                  else                                  else
1215                                  {                                  {
1216                                          error("this shouldn't be happening\n");                                          error("this shouldn't be happening\n");
1217                                          break;                                          exit(1);
1218                                  }                                  }
1219                                  /* this will move pointer from start to first character after FF command */                                  /* this will move pointer from start to first character after FF command */
1220                                  length -= i + 3;                                  length -= i + 3;
# Line 1258  ui_draw_text(uint8 font, uint8 flags, in Line 1227  ui_draw_text(uint8 font, uint8 flags, in
1227                                  if (entry != NULL)                                  if (entry != NULL)
1228                                  {                                  {
1229                                          if ((((uint8 *) (entry->data))[1] ==                                          if ((((uint8 *) (entry->data))[1] ==
1230                                               0)                                               0) && (!(flags & TEXT2_IMPLICIT_X)))
                                             && (!(flags & TEXT2_IMPLICIT_X)))  
1231                                          {                                          {
1232                                                  if (flags & TEXT2_VERTICAL)                                                  if (flags & TEXT2_VERTICAL)
1233                                                          y += text[i + 2];                                                          y += text[i + 2];
# Line 1275  ui_draw_text(uint8 font, uint8 flags, in Line 1243  ui_draw_text(uint8 font, uint8 flags, in
1243                                          text = &(text[i]);                                          text = &(text[i]);
1244                                          i = 0;                                          i = 0;
1245                                          for (j = 0; j < entry->size; j++)                                          for (j = 0; j < entry->size; j++)
1246                                                  DO_GLYPH(((uint8 *) (entry->                                                  DO_GLYPH(((uint8 *) (entry->data)), j);
                                                                      data)),  
                                                          j);  
1247                                  }                                  }
1248                                  break;                                  break;
1249    
# Line 1287  ui_draw_text(uint8 font, uint8 flags, in Line 1253  ui_draw_text(uint8 font, uint8 flags, in
1253                                  break;                                  break;
1254                  }                  }
1255          }          }
1256            if (ownbackstore)
1257            {
1258                    if (boxcx > 1)
1259                            XCopyArea(display, backstore, wnd, gc, boxx,
1260                                      boxy, boxcx, boxcy, boxx, boxy);
1261                    else
1262                            XCopyArea(display, backstore, wnd, gc, clipx,
1263                                      clipy, clipcx, clipcy, clipx, clipy);
1264            }
1265  }  }
1266    
1267  void  void
# Line 1299  ui_desktop_save(uint32 offset, int x, in Line 1272  ui_desktop_save(uint32 offset, int x, in
1272    
1273          if (ownbackstore)          if (ownbackstore)
1274          {          {
1275                  image = XGetImage(display, backstore, x, y, cx, cy, AllPlanes,                  image = XGetImage(display, backstore, x, y, cx, cy, AllPlanes, ZPixmap);
                                   ZPixmap);  
1276          }          }
1277          else          else
1278          {          {
1279                  pix = XCreatePixmap(display, wnd, cx, cy, depth);                  pix = XCreatePixmap(display, wnd, cx, cy, depth);
1280                  XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);                  XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);
1281                  image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes,                  image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes, ZPixmap);
                                   ZPixmap);  
1282                  XFreePixmap(display, pix);                  XFreePixmap(display, pix);
1283          }          }
1284    
1285          offset *= bpp / 8;          offset *= bpp / 8;
1286          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);  
1287    
1288          XDestroyImage(image);          XDestroyImage(image);
1289  }  }
# Line 1329  ui_desktop_restore(uint32 offset, int x, Line 1299  ui_desktop_restore(uint32 offset, int x,
1299          if (data == NULL)          if (data == NULL)
1300                  return;                  return;
1301    
1302          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
1303                               0, data, cx, cy, BitmapPad(display),                               (char *) data, cx, cy, BitmapPad(display), cx * bpp / 8);
                              cx * bpp / 8);  
1304    
1305          if (ownbackstore)          if (ownbackstore)
1306          {          {

Legend:
Removed from v.64  
changed lines
  Added in v.278

  ViewVC Help
Powered by ViewVC 1.1.26