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

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

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

revision 52 by n-ki, Fri Apr 26 08:22:39 2002 UTC revision 196 by astrand, Wed Sep 25 11:07:12 2002 UTC
# 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 char title[];
33    BOOL enable_compose = False;
34    
35  Display *display;  Display *display;
 XkbDescPtr xkb;  
36  static int x_socket;  static int x_socket;
37    static Screen *screen;
38  static Window wnd;  static Window wnd;
39  static GC gc;  static GC gc;
40  static Visual *visual;  static Visual *visual;
41  static int depth;  static int depth;
42  static int bpp;  static int bpp;
43    static XIM IM;
44    static XIC IC;
45    static Cursor current_cursor;
46    
47  /* endianness */  /* endianness */
48  static BOOL host_be;  static BOOL host_be;
# Line 49  static BOOL xserver_be; Line 52  static BOOL xserver_be;
52  static BOOL ownbackstore;  static BOOL ownbackstore;
53  static Pixmap backstore;  static Pixmap backstore;
54    
 /* needed to keep track of the modifiers */  
 static unsigned int key_modifier_state = 0;  
 static unsigned int key_down_state = 0;  
   
 #define DShift1Mask   (1<<0)  
 #define DShift2Mask   (1<<1)  
 #define DControl1Mask (1<<2)  
 #define DControl2Mask (1<<3)  
 #define DMod1Mask     (1<<4)  
 #define DMod2Mask     (1<<5)  
   
55  #define FILL_RECTANGLE(x,y,cx,cy)\  #define FILL_RECTANGLE(x,y,cx,cy)\
56  { \  { \
57          XFillRectangle(display, wnd, gc, x, y, cx, cy); \          XFillRectangle(display, wnd, gc, x, y, cx, cy); \
# Line 68  static unsigned int key_down_state = 0; Line 60  static unsigned int key_down_state = 0;
60  }  }
61    
62  /* colour maps */  /* colour maps */
 static BOOL owncolmap;  
63  static Colormap xcolmap;  static Colormap xcolmap;
 static uint32 white;  
64  static uint32 *colmap;  static uint32 *colmap;
65    
66  #define TRANSLATE(col)          ( owncolmap ? col : translate_colour(colmap[col]) )  #define SET_FOREGROUND(col)     XSetForeground(display, gc, translate_colour(colmap[col]));
67  #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));  
68    
69  static int rop2_map[] = {  static int rop2_map[] = {
70          GXclear,                /* 0 */          GXclear,                /* 0 */
# Line 99  static int rop2_map[] = { Line 88  static int rop2_map[] = {
88  #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]); }
89  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }
90    
 void xwin_release_modifiers(XKeyEvent* ev, uint32 ev_time, uint32 scancode);  
 void xwin_press_modifiers(XKeyEvent* ev, uint32 ev_time, uint32 scancode);  
   
91  static void  static void
92  translate8(uint8 *data, uint8 *out, uint8 *end)  translate8(uint8 * data, uint8 * out, uint8 * end)
93  {  {
94          while (out < end)          while (out < end)
95                  *(out++) = (uint8)colmap[*(data++)];                  *(out++) = (uint8) colmap[*(data++)];
96  }  }
97    
98  static void  static void
99  translate16(uint8 *data, uint16 *out, uint16 *end)  translate16(uint8 * data, uint16 * out, uint16 * end)
100  {  {
101          while (out < end)          while (out < end)
102                  *(out++) = (uint16)colmap[*(data++)];                  *(out++) = (uint16) colmap[*(data++)];
103  }  }
104    
105  /* little endian - conversion happens when colourmap is built */  /* little endian - conversion happens when colourmap is built */
106  static void  static void
107  translate24(uint8 *data, uint8 *out, uint8 *end)  translate24(uint8 * data, uint8 * out, uint8 * end)
108  {  {
109          uint32 value;          uint32 value;
110    
# Line 132  translate24(uint8 *data, uint8 *out, uin Line 118  translate24(uint8 *data, uint8 *out, uin
118  }  }
119    
120  static void  static void
121  translate32(uint8 *data, uint32 *out, uint32 *end)  translate32(uint8 * data, uint32 * out, uint32 * end)
122  {  {
123          while (out < end)          while (out < end)
124                  *(out++) = colmap[*(data++)];                  *(out++) = colmap[*(data++)];
125  }  }
126    
127  static uint8 *  static uint8 *
128  translate_image(int width, int height, uint8 *data)  translate_image(int width, int height, uint8 * data)
129  {  {
130          int size = width * height * bpp/8;          int size = width * height * bpp / 8;
131          uint8 *out = xmalloc(size);          uint8 *out = xmalloc(size);
132          uint8 *end = out + size;          uint8 *end = out + size;
133    
# Line 152  translate_image(int width, int height, u Line 138  translate_image(int width, int height, u
138                          break;                          break;
139    
140                  case 16:                  case 16:
141                          translate16(data, (uint16 *)out, (uint16 *)end);                          translate16(data, (uint16 *) out, (uint16 *) end);
142                          break;                          break;
143    
144                  case 24:                  case 24:
# Line 160  translate_image(int width, int height, u Line 146  translate_image(int width, int height, u
146                          break;                          break;
147    
148                  case 32:                  case 32:
149                          translate32(data, (uint32 *)out, (uint32 *)end);                          translate32(data, (uint32 *) out, (uint32 *) end);
150                          break;                          break;
151          }          }
152    
# Line 197  translate_colour(uint32 colour) Line 183  translate_colour(uint32 colour)
183  }  }
184    
185  BOOL  BOOL
186  ui_create_window(char *title)  get_key_state(int keysym)
187  {  {
188          XSetWindowAttributes attribs;          int keysymMask = 0, modifierpos, key;
189          XClassHint *classhints;          Window wDummy1, wDummy2;
190          XSizeHints *sizehints;          int iDummy3, iDummy4, iDummy5, iDummy6;
191          unsigned long input_mask;          unsigned int current_state;
192          XPixmapFormatValues *pfm;          int offset;
193          Screen *screen;  
194          uint16 test;          XModifierKeymap *map = XGetModifierMapping(display);
195          int i;          KeyCode keycode = XKeysymToKeycode(display, keysym);
           
         int xkb_minor, xkb_major;  
         int xkb_event, xkb_error, xkb_reason;  
   
         /* compare compiletime libs with runtime libs. */  
         xkb_major = XkbMajorVersion;  
         xkb_minor = XkbMinorVersion;  
         if( XkbLibraryVersion( &xkb_major, &xkb_minor ) == False )  
         {  
                 error("please re-compile rdesktop\ncompile time version of xkb is not compatible with\nyour runtime version of the library\n");  
                 return False;  
         }  
196    
197            if (keycode == NoSymbol)
198                    return False;
199    
200          /* XKB is the 'new' keyboard handler in x.. ( the xkb code in Xfree86 originates from SGI, years 1993 and 1995 from what I could tell. )          for (modifierpos = 0; modifierpos < 8; modifierpos++)
          * it makes it possible for people with disabilities to use rdesktop, stickykeys, bouncekeys etc. VERY MUCH useful.  
          * XFree86 has had support for it since it's earliest incarnation. I believe it is a reasonable dependency.  
          */  
         display = XkbOpenDisplay( NULL, &xkb_event, &xkb_error, &xkb_major, &xkb_minor, &xkb_reason );  
         switch(xkb_reason)  
201          {          {
202                  case XkbOD_BadLibraryVersion:                  offset = map->max_keypermod * modifierpos;
203                          error("XkbOD_BadLibraryVersion: XKB extensions in server and the library rdesktop is linked against aren't compatible with each other.\n");  
204                          break;                  for (key = 0; key < map->max_keypermod; key++)
205                  case XkbOD_ConnectionRefused:                  {
206                          error("XkbOD_ConnectionRefused\n");                          if (map->modifiermap[offset + key] == keycode)
207                          break;                                  keysymMask = 1 << modifierpos;
208                  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;  
209          }          }
210    
211            XQueryPointer(display, DefaultRootWindow(display), &wDummy1,
212                          &wDummy2, &iDummy3, &iDummy4, &iDummy5, &iDummy6, &current_state);
213    
214            XFreeModifiermap(map);
215    
216            return (current_state & keysymMask) ? True : False;
217    }
218    
219    BOOL
220    ui_init(void)
221    {
222            XPixmapFormatValues *pfm;
223            uint16 test;
224            int i;
225    
226            display = XOpenDisplay(NULL);
227          if (display == NULL)          if (display == NULL)
228          {          {
229                  error("Failed to open display\n");                  error("Failed to open display\n");
# Line 255  ui_create_window(char *title) Line 234  ui_create_window(char *title)
234          screen = DefaultScreenOfDisplay(display);          screen = DefaultScreenOfDisplay(display);
235          visual = DefaultVisualOfScreen(screen);          visual = DefaultVisualOfScreen(screen);
236          depth = DefaultDepthOfScreen(screen);          depth = DefaultDepthOfScreen(screen);
237            
238          pfm = XListPixmapFormats(display, &i);          pfm = XListPixmapFormats(display, &i);
239          if (pfm != NULL)          if (pfm != NULL)
240          {          {
# Line 263  ui_create_window(char *title) Line 242  ui_create_window(char *title)
242                     desirable, e.g. 24 bits->32 bits. */                     desirable, e.g. 24 bits->32 bits. */
243                  while (i--)                  while (i--)
244                  {                  {
245                          if ((pfm[i].depth == depth)                          if ((pfm[i].depth == depth) && (pfm[i].bits_per_pixel > bpp))
                             && (pfm[i].bits_per_pixel > bpp))  
246                          {                          {
247                                  bpp = pfm[i].bits_per_pixel;                                  bpp = pfm[i].bits_per_pixel;
248                          }                          }
# Line 279  ui_create_window(char *title) Line 257  ui_create_window(char *title)
257                  return False;                  return False;
258          }          }
259    
260          if (depth <= 8)          xcolmap = DefaultColormapOfScreen(screen);
261                  owncolmap = True;          gc = XCreateGC(display, RootWindowOfScreen(screen), 0, NULL);
262          else  
263                  xcolmap = DefaultColormapOfScreen(screen);          if (DoesBackingStore(screen) != Always)
264                    ownbackstore = True;
265    
266          test = 1;          test = 1;
267          host_be = !(BOOL)(*(uint8 *)(&test));          host_be = !(BOOL) (*(uint8 *) (&test));
268          xserver_be = (ImageByteOrder(display) == MSBFirst);          xserver_be = (ImageByteOrder(display) == MSBFirst);
269    
         white = WhitePixelOfScreen(screen);  
         attribs.background_pixel = BlackPixelOfScreen(screen);  
         attribs.backing_store = DoesBackingStore(screen);  
   
         if (attribs.backing_store == NotUseful)  
                 ownbackstore = True;  
   
270          if (fullscreen)          if (fullscreen)
271          {          {
                 attribs.override_redirect = True;  
272                  width = WidthOfScreen(screen);                  width = WidthOfScreen(screen);
273                  height = HeightOfScreen(screen);                  height = HeightOfScreen(screen);
274          }          }
275          else  
276            /* make sure width is a multiple of 4 */
277            width = (width + 3) & ~3;
278    
279            if (ownbackstore)
280          {          {
281                  attribs.override_redirect = False;                  backstore =
282                            XCreatePixmap(display, RootWindowOfScreen(screen), width, height, depth);
283    
284                    /* clear to prevent rubbish being exposed at startup */
285                    XSetForeground(display, gc, BlackPixelOfScreen(screen));
286                    XFillRectangle(display, backstore, gc, 0, 0, width, height);
287          }          }
288    
289          width = (width + 3) & ~3; /* make width a multiple of 32 bits */          if (enable_compose)
290                    IM = XOpenIM(display, NULL, NULL, NULL);
291    
292          wnd = XCreateWindow(display, RootWindowOfScreen(screen),          xkeymap_init();
293                              0, 0, width, height, 0, CopyFromParent,          return True;
294                              InputOutput, CopyFromParent,  }
295                              CWBackingStore | CWBackPixel | CWOverrideRedirect,  
296                              &attribs);  void
297    ui_deinit(void)
298    {
299            if (IM != NULL)
300                    XCloseIM(IM);
301    
302            if (ownbackstore)
303                    XFreePixmap(display, backstore);
304    
305            XFreeGC(display, gc);
306            XCloseDisplay(display);
307            display = NULL;
308    }
309    
310    BOOL
311    ui_create_window(void)
312    {
313            XSetWindowAttributes attribs;
314            XClassHint *classhints;
315            XSizeHints *sizehints;
316            int wndwidth, wndheight;
317            long input_mask, ic_input_mask;
318            XEvent xevent;
319    
320            wndwidth = fullscreen ? WidthOfScreen(screen) : width;
321            wndheight = fullscreen ? HeightOfScreen(screen) : height;
322    
323            attribs.background_pixel = BlackPixelOfScreen(screen);
324            attribs.backing_store = ownbackstore ? NotUseful : Always;
325            attribs.override_redirect = fullscreen;
326    
327            wnd = XCreateWindow(display, RootWindowOfScreen(screen), 0, 0, wndwidth, wndheight,
328                                0, CopyFromParent, InputOutput, CopyFromParent,
329                                CWBackPixel | CWBackingStore | CWOverrideRedirect, &attribs);
330    
331          XStoreName(display, wnd, title);          XStoreName(display, wnd, title);
332    
# Line 334  ui_create_window(char *title) Line 348  ui_create_window(char *title)
348                  XFree(sizehints);                  XFree(sizehints);
349          }          }
350    
351          xkeymap_init();          input_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask |
352                    StructureNotifyMask | FocusChangeMask;
353    
         input_mask = KeyPressMask | KeyReleaseMask |  
                          ButtonPressMask | ButtonReleaseMask |  
                          EnterWindowMask | LeaveWindowMask | KeymapStateMask;  
354          if (sendmotion)          if (sendmotion)
355                  input_mask |= PointerMotionMask;                  input_mask |= PointerMotionMask;
   
356          if (ownbackstore)          if (ownbackstore)
357                  input_mask |= ExposureMask;                  input_mask |= ExposureMask;
358    
359          XSelectInput(display, wnd, input_mask);          if (IM != NULL)
360          gc = XCreateGC(display, wnd, 0, NULL);          {
361                    IC = XCreateIC(IM, XNInputStyle, (XIMPreeditNothing | XIMStatusNothing),
362                                   XNClientWindow, wnd, XNFocusWindow, wnd, NULL);
363    
364          if (ownbackstore)                  if ((IC != NULL)
365                  backstore = XCreatePixmap(display, wnd, width, height, depth);                      && (XGetICValues(IC, XNFilterEvents, &ic_input_mask, NULL) == NULL))
366                            input_mask |= ic_input_mask;
367            }
368    
369            XSelectInput(display, wnd, input_mask);
370          XMapWindow(display, wnd);          XMapWindow(display, wnd);
371    
372          /* TODO: error texts... make them friendly. */          /* wait for MapNotify */
373          xkb = XkbGetKeyboard(display, XkbAllComponentsMask, XkbUseCoreKbd);          do
         if ((int)xkb == BadAlloc || xkb == NULL)  
         {  
                         error( "XkbGetKeyboard failed.\n");  
                         exit(0);  
         }  
   
         /* TODO: error texts... make them friendly. */  
         if( XkbSelectEvents(display, xkb->device_spec, XkbAllEventsMask, XkbAllEventsMask) == False )  
374          {          {
375                          error( "XkbSelectEvents failed.\n");                  XMaskEvent(display, StructureNotifyMask, &xevent);
                         exit(0);  
376          }          }
377            while (xevent.type != MapNotify);
378    
379            if (fullscreen)
380                    XSetInputFocus(display, wnd, RevertToPointerRoot, CurrentTime);
381    
382          return True;          return True;
383  }  }
384    
385  void  void
386  ui_destroy_window()  ui_destroy_window(void)
387  {  {
388          if( xkb != NULL )          if (IC != NULL)
389                  XkbFreeKeyboard(xkb, XkbAllControlsMask, True);                  XDestroyIC(IC);
390    
         if (ownbackstore)  
                 XFreePixmap(display, backstore);  
   
         XFreeGC(display, gc);  
391          XDestroyWindow(display, wnd);          XDestroyWindow(display, wnd);
         XCloseDisplay(display);  
         display = NULL;  
392  }  }
393    
394    void
395    xwin_toggle_fullscreen(void)
396    {
397            Pixmap contents = 0;
398    
399            if (!ownbackstore)
400            {
401                    /* need to save contents of window */
402                    contents = XCreatePixmap(display, wnd, width, height, depth);
403                    XCopyArea(display, wnd, contents, gc, 0, 0, width, height, 0, 0);
404            }
405    
406            ui_destroy_window();
407            fullscreen = !fullscreen;
408            ui_create_window();
409    
410            XDefineCursor(display, wnd, current_cursor);
411    
412            if (!ownbackstore)
413            {
414                    XCopyArea(display, contents, wnd, gc, 0, 0, width, height, 0, 0);
415                    XFreePixmap(display, contents);
416            }
417    }
418    
419    /* Process all events in Xlib queue */
420  static void  static void
421  xwin_process_events()  xwin_process_events(void)
422  {  {
423          XkbEvent xkbevent;          XEvent xevent;
           
424          KeySym keysym;          KeySym keysym;
         uint8 scancode;  
425          uint16 button, flags;          uint16 button, flags;
426          uint32 ev_time;          uint32 ev_time;
427          uint32 tmpmods;          key_translation tr;
428            char *ksname = NULL;
429          if (display == NULL)          char str[256];
430                  return;          Status status;
431    
432          while (XCheckMaskEvent(display, ~0, &xkbevent.core))          while (XPending(display) > 0)
433          {          {
434                    XNextEvent(display, &xevent);
435    
436                    if ((IC != NULL) && (XFilterEvent(&xevent, None) == True))
437                    {
438                            DEBUG_KBD(("Filtering event\n"));
439                            continue;
440                    }
441    
442                  ev_time = time(NULL);                  ev_time = time(NULL);
443                  flags = 0;                  flags = 0;
444    
445                  switch (xkbevent.type)                  switch (xevent.type)
446                  {                  {
447                          case KeymapNotify:                          case KeyPress:
448                                  /* TODO:                                  if (IC != NULL)
449                                   * read modifier status at focus in, and update the local masks, and the other end as well..                                          /* Multi_key compatible version */
450                                   * if not, we may get out of sync.                                  {
451                                   * xkbevent.core.xkeymap.key_vector                                          XmbLookupString(IC,
452                                   * char key_vector[32];                                                          (XKeyPressedEvent *) &
453                                   */                                                          xevent, str, sizeof(str), &keysym, &status);
454                                  break;                                          if (!((status == XLookupKeySym) || (status == XLookupBoth)))
455                                            {
456                                                    error("XmbLookupString failed with status 0x%x\n",
457                                                          status);
458                                                    break;
459                                            }
460                                    }
461                                    else
462                                    {
463                                            /* Plain old XLookupString */
464                                            DEBUG_KBD(("\nNo input context, using XLookupString\n"));
465                                            XLookupString((XKeyEvent *) & xevent,
466                                                          str, sizeof(str), &keysym, NULL);
467                                    }
468    
469                          case KeyRelease:                                  ksname = get_ksname(keysym);
470                                  flags = KBD_FLAG_DOWN | KBD_FLAG_UP;                                  DEBUG_KBD(("KeyPress for (keysym 0x%lx, %s)\n", keysym, ksname));
                                 /* fall through */  
471    
472                          case KeyPress:                                  if (handle_special_keys(keysym, ev_time, True))
                                 if( XkbTranslateKeyCode(xkb, xkbevent.core.xkey.keycode, xkbevent.core.xkey.state, &tmpmods, &keysym) == False )  
473                                          break;                                          break;
                                 scancode = xkeymap_translate_key(keysym, xkbevent.core.xkey.keycode, &flags);  
474    
475                                  if (scancode == 0 )                                  tr = xkeymap_translate_key(keysym,
476                                                               xevent.xkey.keycode, xevent.xkey.state);
477    
478                                    if (tr.scancode == 0)
479                                          break;                                          break;
480    
481                                  /* keep track of the modifiers -- needed for stickykeys... */                                  ensure_remote_modifiers(ev_time, tr);
482                                  if( xkbevent.type == KeyPress )  
483                                          xwin_press_modifiers( &xkbevent.core.xkey, ev_time, scancode );                                  rdp_send_scancode(ev_time, RDP_KEYPRESS, tr.scancode);
484                                    break;
485                            case KeyRelease:
486                                    XLookupString((XKeyEvent *) & xevent, str,
487                                                  sizeof(str), &keysym, NULL);
488    
489                                    ksname = get_ksname(keysym);
490                                    DEBUG_KBD(("\nKeyRelease for (keysym 0x%lx, %s)\n", keysym,
491                                               ksname));
492    
493                                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE, flags, scancode, 0);                                  if (handle_special_keys(keysym, ev_time, False))
494                                            break;
495    
496                                  if( xkbevent.type == KeyRelease )                                  tr = xkeymap_translate_key(keysym,
497                                          xwin_release_modifiers( &xkbevent.core.xkey, ev_time, scancode );                                                             xevent.xkey.keycode, xevent.xkey.state);
498    
499                                    if (tr.scancode == 0)
500                                            break;
501    
502                                    rdp_send_scancode(ev_time, RDP_KEYRELEASE, tr.scancode);
503                                  break;                                  break;
504    
505                          case ButtonPress:                          case ButtonPress:
# Line 444  xwin_process_events() Line 507  xwin_process_events()
507                                  /* fall through */                                  /* fall through */
508    
509                          case ButtonRelease:                          case ButtonRelease:
510                                  button = xkeymap_translate_button(xkbevent.core.xbutton.button);                                  button = xkeymap_translate_button(xevent.xbutton.button);
511                                  if (button == 0)                                  if (button == 0)
512                                          break;                                          break;
513    
514                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,
515                                                 flags | button,                                                 flags | button, xevent.xbutton.x, xevent.xbutton.y);
                                                xkbevent.core.xbutton.x,  
                                                xkbevent.core.xbutton.y);  
516                                  break;                                  break;
517    
518                          case MotionNotify:                          case MotionNotify:
519                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,
520                                                 MOUSE_FLAG_MOVE,                                                 MOUSE_FLAG_MOVE, xevent.xmotion.x, xevent.xmotion.y);
                                                xkbevent.core.xmotion.x,  
                                                xkbevent.core.xmotion.y);  
521                                  break;                                  break;
522    
523                          case EnterNotify:                          case FocusIn:
524                                  XGrabKeyboard(display, wnd, True, GrabModeAsync,                                  reset_modifier_keys();
525                                                GrabModeAsync, CurrentTime);                                  if (grab_keyboard)
526                                            XGrabKeyboard(display, wnd, True,
527                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
528                                  break;                                  break;
529    
530                          case LeaveNotify:                          case FocusOut:
531                                  XUngrabKeyboard(display, CurrentTime);                                  if (grab_keyboard)
532                                            XUngrabKeyboard(display, CurrentTime);
533                                  break;                                  break;
534    
535                          case Expose:                          case Expose:
536                                  XCopyArea(display, backstore, wnd, gc,                                  XCopyArea(display, backstore, wnd, gc,
537                                            xkbevent.core.xexpose.x, xkbevent.core.xexpose.y,                                            xevent.xexpose.x, xevent.xexpose.y,
538                                            xkbevent.core.xexpose.width, xkbevent.core.xexpose.height,                                            xevent.xexpose.width,
539                                            xkbevent.core.xexpose.x, xkbevent.core.xexpose.y);                                            xevent.xexpose.height,
540                                              xevent.xexpose.x, xevent.xexpose.y);
541                                  break;                                  break;
                 }  
         }  
 }  
   
 void  
 xwin_release_modifiers(XKeyEvent* ev, uint32 ev_time, uint32 scancode)  
 {  
         switch (scancode) {  
         case 0x2a:  
                 key_down_state &= ~DShift1Mask;  
                 break;  
         case 0x36:  
                 key_down_state &= ~DShift2Mask;  
                 break;  
         case 0x1d:  
                 key_down_state &= ~DControl1Mask;  
                 break;  
         case 0x9d:  
                 key_down_state &= ~DControl2Mask;  
                 break;  
         case 0x38:  
                 key_down_state &= ~DMod1Mask;  
                 break;  
         case 0xb8:  
                 key_down_state &= ~DMod2Mask;  
                 break;  
         }  
   
         if( !(ShiftMask & ev->state) && (key_down_state & DShift1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0x2a, 0);  
                 key_down_state &= ~DShift1Mask;  
   
         }  
   
         if( !(ControlMask & ev->state) && (key_down_state & DControl1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0x1d, 0);  
                 key_down_state &= ~DControl1Mask;  
   
         }  
           
         if( !(Mod1Mask & ev->state) && (key_down_state & DMod1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0x38, 0);  
                 key_down_state &= ~DMod1Mask;  
   
         }  
           
         if( !(Mod2Mask & ev->state) && (key_down_state & DMod2Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_UP, 0xb8, 0);  
                 key_down_state &= ~DMod2Mask;  
         }  
 }  
542    
543                            case MappingNotify:
544                                    /* Refresh keyboard mapping if it has changed. This is important for
545                                       Xvnc, since it allocates keycodes dynamically */
546                                    if (xevent.xmapping.request == MappingKeyboard
547                                        || xevent.xmapping.request == MappingModifier)
548                                            XRefreshKeyboardMapping(&xevent.xmapping);
549                                    break;
550    
551  void                  }
 xwin_press_modifiers(XKeyEvent* ev, uint32 ev_time, uint32 scancode)  
 {  
         key_modifier_state = ev->state;  
   
         switch (scancode) {  
         case 0x2a:  
                 key_down_state |= DShift1Mask;  
                 break;  
         case 0x36:  
                 key_down_state |= DShift2Mask;  
                 break;  
         case 0x1d:  
                 key_down_state |= DControl1Mask;  
                 break;  
         case 0x9d:  
                 key_down_state |= DControl2Mask;  
                 break;  
         case 0x38:  
                 key_down_state |= DMod1Mask;  
                 break;  
         case 0xb8:  
                 key_down_state |= DMod2Mask;  
                 break;  
         }  
   
         if( (ShiftMask & ev->state) && !((key_down_state & DShift1Mask) || (key_down_state & DShift2Mask)))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0x2a, 0);  
                 key_down_state |= DShift1Mask;  
   
         }  
   
         if( (ControlMask & ev->state) && !((key_down_state & DControl1Mask) || (key_down_state & DControl2Mask)))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0x1d, 0);  
                 key_down_state |= DControl1Mask;  
   
         }  
   
         if( (Mod1Mask & ev->state) && !(key_down_state & DMod1Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0x38, 0);  
                 key_down_state |= DMod1Mask;  
   
         }  
   
         if( (Mod2Mask & ev->state) && !(key_down_state & DMod2Mask))  
         {  
                 rdp_send_input(ev_time, RDP_INPUT_SCANCODE, KBD_FLAG_DOWN, 0xb8, 0);  
                 key_down_state |= DMod2Mask;  
   
552          }          }
553  }  }
554    
555  void  void
556  ui_select(int rdp_socket)  ui_select(int rdp_socket)
557  {  {
558          int n = (rdp_socket > x_socket) ? rdp_socket+1 : x_socket+1;          int n = (rdp_socket > x_socket) ? rdp_socket + 1 : x_socket + 1;
559          fd_set rfds;          fd_set rfds;
560    
         XFlush(display);  
   
561          FD_ZERO(&rfds);          FD_ZERO(&rfds);
562    
563          while (True)          while (True)
564          {          {
565                    /* Process any events already waiting */
566                    xwin_process_events();
567    
568                  FD_ZERO(&rfds);                  FD_ZERO(&rfds);
569                  FD_SET(rdp_socket, &rfds);                  FD_SET(rdp_socket, &rfds);
570                  FD_SET(x_socket, &rfds);                  FD_SET(x_socket, &rfds);
# Line 613  ui_select(int rdp_socket) Line 578  ui_select(int rdp_socket)
578                                  continue;                                  continue;
579                  }                  }
580    
                 if (FD_ISSET(x_socket, &rfds))  
                         xwin_process_events();  
   
581                  if (FD_ISSET(rdp_socket, &rfds))                  if (FD_ISSET(rdp_socket, &rfds))
582                          return;                          return;
583          }          }
# Line 628  ui_move_pointer(int x, int y) Line 590  ui_move_pointer(int x, int y)
590  }  }
591    
592  HBITMAP  HBITMAP
593  ui_create_bitmap(int width, int height, uint8 *data)  ui_create_bitmap(int width, int height, uint8 * data)
594  {  {
595          XImage *image;          XImage *image;
596          Pixmap bitmap;          Pixmap bitmap;
597          uint8 *tdata;          uint8 *tdata;
598    
599          tdata = (owncolmap ? data : translate_image(width, height, data));          tdata = translate_image(width, height, data);
600          bitmap = XCreatePixmap(display, wnd, width, height, depth);          bitmap = XCreatePixmap(display, wnd, width, height, depth);
601          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
602                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
603    
604          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);
605    
606          XFree(image);          XFree(image);
607          if (!owncolmap)          xfree(tdata);
                 xfree(tdata);  
608          return (HBITMAP) bitmap;          return (HBITMAP) bitmap;
609  }  }
610    
611  void  void
612  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)  
613  {  {
614          XImage *image;          XImage *image;
615          uint8 *tdata;          uint8 *tdata;
616    
617          tdata = (owncolmap ? data : translate_image(width, height, data));          tdata = translate_image(width, height, data);
618          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
619                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
620    
621          if (ownbackstore)          if (ownbackstore)
622          {          {
# Line 669  ui_paint_bitmap(int x, int y, int cx, in Line 629  ui_paint_bitmap(int x, int y, int cx, in
629          }          }
630    
631          XFree(image);          XFree(image);
632          if (!owncolmap)          xfree(tdata);
                 xfree(tdata);  
633  }  }
634    
635  void  void
636  ui_destroy_bitmap(HBITMAP bmp)  ui_destroy_bitmap(HBITMAP bmp)
637  {  {
638          XFreePixmap(display, (Pixmap)bmp);          XFreePixmap(display, (Pixmap) bmp);
639  }  }
640    
641  HGLYPH  HGLYPH
642  ui_create_glyph(int width, int height, uint8 *data)  ui_create_glyph(int width, int height, uint8 * data)
643  {  {
644          XImage *image;          XImage *image;
645          Pixmap bitmap;          Pixmap bitmap;
# Line 692  ui_create_glyph(int width, int height, u Line 651  ui_create_glyph(int width, int height, u
651          bitmap = XCreatePixmap(display, wnd, width, height, 1);          bitmap = XCreatePixmap(display, wnd, width, height, 1);
652          gc = XCreateGC(display, bitmap, 0, NULL);          gc = XCreateGC(display, bitmap, 0, NULL);
653    
654          image = XCreateImage(display, visual, 1, ZPixmap, 0,          image = XCreateImage(display, visual, 1, ZPixmap, 0, (char *) data,
655                               data, width, height, 8, scanline);                               width, height, 8, scanline);
656          image->byte_order = MSBFirst;          image->byte_order = MSBFirst;
657          image->bitmap_bit_order = MSBFirst;          image->bitmap_bit_order = MSBFirst;
658          XInitImage(image);          XInitImage(image);
# Line 702  ui_create_glyph(int width, int height, u Line 661  ui_create_glyph(int width, int height, u
661    
662          XFree(image);          XFree(image);
663          XFreeGC(display, gc);          XFreeGC(display, gc);
664          return (HGLYPH)bitmap;          return (HGLYPH) bitmap;
665  }  }
666    
667  void  void
668  ui_destroy_glyph(HGLYPH glyph)  ui_destroy_glyph(HGLYPH glyph)
669  {  {
670          XFreePixmap(display, (Pixmap)glyph);          XFreePixmap(display, (Pixmap) glyph);
671  }  }
672    
673  HCURSOR  HCURSOR
674  ui_create_cursor(unsigned int x, unsigned int y, int width,  ui_create_cursor(unsigned int x, unsigned int y, int width, int height,
675                   int height, uint8 *andmask, uint8 *xormask)                   uint8 * andmask, uint8 * xormask)
676  {  {
677          HGLYPH maskglyph, cursorglyph;          HGLYPH maskglyph, cursorglyph;
678          XColor bg, fg;          XColor bg, fg;
# Line 770  ui_create_cursor(unsigned int x, unsigne Line 729  ui_create_cursor(unsigned int x, unsigne
729    
730          cursorglyph = ui_create_glyph(width, height, cursor);          cursorglyph = ui_create_glyph(width, height, cursor);
731          maskglyph = ui_create_glyph(width, height, mask);          maskglyph = ui_create_glyph(width, height, mask);
732            
733          xcursor = XCreatePixmapCursor(display, (Pixmap)cursorglyph,          xcursor =
734                                  (Pixmap)maskglyph, &fg, &bg, x, y);                  XCreatePixmapCursor(display, (Pixmap) cursorglyph,
735                                        (Pixmap) maskglyph, &fg, &bg, x, y);
736    
737          ui_destroy_glyph(maskglyph);          ui_destroy_glyph(maskglyph);
738          ui_destroy_glyph(cursorglyph);          ui_destroy_glyph(cursorglyph);
739          xfree(mask);          xfree(mask);
740          xfree(cursor);          xfree(cursor);
741          return (HCURSOR)xcursor;          return (HCURSOR) xcursor;
742  }  }
743    
744  void  void
745  ui_set_cursor(HCURSOR cursor)  ui_set_cursor(HCURSOR cursor)
746  {  {
747          XDefineCursor(display, wnd, (Cursor)cursor);          current_cursor = (Cursor) cursor;
748            XDefineCursor(display, wnd, current_cursor);
749  }  }
750    
751  void  void
752  ui_destroy_cursor(HCURSOR cursor)  ui_destroy_cursor(HCURSOR cursor)
753  {  {
754          XFreeCursor(display, (Cursor)cursor);          XFreeCursor(display, (Cursor) cursor);
755  }  }
756    
757  #define MAKE_XCOLOR(xc,c) \  #define MAKE_XCOLOR(xc,c) \
# Line 800  ui_destroy_cursor(HCURSOR cursor) Line 761  ui_destroy_cursor(HCURSOR cursor)
761                  (xc)->flags = DoRed | DoGreen | DoBlue;                  (xc)->flags = DoRed | DoGreen | DoBlue;
762    
763  HCOLOURMAP  HCOLOURMAP
764  ui_create_colourmap(COLOURMAP *colours)  ui_create_colourmap(COLOURMAP * colours)
765  {  {
766          COLOURENTRY *entry;          COLOURENTRY *entry;
767          int i, ncolours = colours->ncolours;          int i, ncolours = colours->ncolours;
768            uint32 *map = xmalloc(sizeof(*colmap) * ncolours);
769          if (owncolmap)          XColor xentry;
770            XColor xc_cache[256];
771            uint32 colour;
772            int colLookup = 256;
773            for (i = 0; i < ncolours; i++)
774          {          {
775                  XColor *xcolours, *xentry;                  entry = &colours->colours[i];
776                  Colormap map;                  MAKE_XCOLOR(&xentry, entry);
777    
778                  xcolours = xmalloc(sizeof(XColor) * ncolours);                  if (XAllocColor(display, xcolmap, &xentry) == 0)
                 for (i = 0; i < ncolours; i++)  
779                  {                  {
780                          entry = &colours->colours[i];                          /* Allocation failed, find closest match. */
781                          xentry = &xcolours[i];                          int j = 256;
782                          xentry->pixel = i;                          int nMinDist = 3 * 256 * 256;
783                          MAKE_XCOLOR(xentry, entry);                          long nDist = nMinDist;
                 }  
784    
785                  map = XCreateColormap(display, wnd, visual, AllocAll);                          /* only get the colors once */
786                  XStoreColors(display, map, xcolours, ncolours);                          while (colLookup--)
787                            {
788                                    xc_cache[colLookup].pixel = colLookup;
789                                    xc_cache[colLookup].red = xc_cache[colLookup].green =
790                                            xc_cache[colLookup].blue = 0;
791                                    xc_cache[colLookup].flags = 0;
792                                    XQueryColor(display,
793                                                DefaultColormap(display, DefaultScreen(display)),
794                                                &xc_cache[colLookup]);
795                            }
796                            colLookup = 0;
797    
798                  xfree(xcolours);                          /* approximate the pixel */
799                  return (HCOLOURMAP)map;                          while (j--)
800          }                          {
801          else                                  if (xc_cache[j].flags)
802          {                                  {
803                  uint32 *map = xmalloc(sizeof(*colmap) * ncolours);                                          nDist = ((long) (xc_cache[j].red >> 8) -
804                  XColor xentry;                                                   (long) (xentry.red >> 8)) *
805                  uint32 colour;                                                  ((long) (xc_cache[j].red >> 8) -
806                                                     (long) (xentry.red >> 8)) +
807                                                    ((long) (xc_cache[j].green >> 8) -
808                                                     (long) (xentry.green >> 8)) *
809                                                    ((long) (xc_cache[j].green >> 8) -
810                                                     (long) (xentry.green >> 8)) +
811                                                    ((long) (xc_cache[j].blue >> 8) -
812                                                     (long) (xentry.blue >> 8)) *
813                                                    ((long) (xc_cache[j].blue >> 8) -
814                                                     (long) (xentry.blue >> 8));
815                                    }
816                                    if (nDist < nMinDist)
817                                    {
818                                            nMinDist = nDist;
819                                            xentry.pixel = j;
820                                    }
821                            }
822                    }
823                    colour = xentry.pixel;
824    
825                  for (i = 0; i < ncolours; i++)                  /* update our cache */
826                    if (xentry.pixel < 256)
827                  {                  {
828                          entry = &colours->colours[i];                          xc_cache[xentry.pixel].red = xentry.red;
829                          MAKE_XCOLOR(&xentry, entry);                          xc_cache[xentry.pixel].green = xentry.green;
830                            xc_cache[xentry.pixel].blue = xentry.blue;
831    
                         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);  
832                  }                  }
833    
834                  return map;  
835                    /* byte swap here to make translate_image faster */
836                    map[i] = translate_colour(colour);
837          }          }
838    
839            return map;
840  }  }
841    
842  void  void
843  ui_destroy_colourmap(HCOLOURMAP map)  ui_destroy_colourmap(HCOLOURMAP map)
844  {  {
845          if (owncolmap)          xfree(map);
                 XFreeColormap(display, (Colormap)map);  
         else  
                 xfree(map);  
846  }  }
847    
848  void  void
849  ui_set_colourmap(HCOLOURMAP map)  ui_set_colourmap(HCOLOURMAP map)
850  {  {
851          if (owncolmap)          colmap = map;
                 XSetWindowColormap(display, wnd, (Colormap)map);  
         else  
                 colmap = map;  
852  }  }
853    
854  void  void
# Line 880  ui_set_clip(int x, int y, int cx, int cy Line 864  ui_set_clip(int x, int y, int cx, int cy
864  }  }
865    
866  void  void
867  ui_reset_clip()  ui_reset_clip(void)
868  {  {
869          XRectangle rect;          XRectangle rect;
870    
# Line 892  ui_reset_clip() Line 876  ui_reset_clip()
876  }  }
877    
878  void  void
879  ui_bell()  ui_bell(void)
880  {  {
881          XBell(display, 0);          XBell(display, 0);
882  }  }
# Line 909  ui_destblt(uint8 opcode, Line 893  ui_destblt(uint8 opcode,
893  void  void
894  ui_patblt(uint8 opcode,  ui_patblt(uint8 opcode,
895            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
896            /* brush */ BRUSH *brush, int bgcolour, int fgcolour)            /* brush */ BRUSH * brush, int bgcolour, int fgcolour)
897  {  {
898          Pixmap fill;          Pixmap fill;
899            uint8 i, ipattern[8];
900    
901          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
902    
# Line 923  ui_patblt(uint8 opcode, Line 908  ui_patblt(uint8 opcode,
908                          break;                          break;
909    
910                  case 3: /* Pattern */                  case 3: /* Pattern */
911                          fill = (Pixmap)ui_create_glyph(8, 8, brush->pattern);                          for (i = 0; i != 8; i++)
912                                    ipattern[7 - i] = brush->pattern[i];
913                            fill = (Pixmap) ui_create_glyph(8, 8, ipattern);
914    
915                          SET_FOREGROUND(bgcolour);                          SET_FOREGROUND(bgcolour);
916                          SET_BACKGROUND(fgcolour);                          SET_BACKGROUND(fgcolour);
# Line 934  ui_patblt(uint8 opcode, Line 921  ui_patblt(uint8 opcode,
921                          FILL_RECTANGLE(x, y, cx, cy);                          FILL_RECTANGLE(x, y, cx, cy);
922    
923                          XSetFillStyle(display, gc, FillSolid);                          XSetFillStyle(display, gc, FillSolid);
924                          ui_destroy_glyph((HGLYPH)fill);                          XSetTSOrigin(display, gc, 0, 0);
925                            ui_destroy_glyph((HGLYPH) fill);
926                          break;                          break;
927    
928                  default:                  default:
# Line 952  ui_screenblt(uint8 opcode, Line 940  ui_screenblt(uint8 opcode,
940          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
941          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);
942          if (ownbackstore)          if (ownbackstore)
943                  XCopyArea(display, backstore, backstore, gc, srcx, srcy,                  XCopyArea(display, backstore, backstore, gc, srcx, srcy, cx, cy, x, y);
                           cx, cy, x, y);  
944          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
945  }  }
946    
# Line 963  ui_memblt(uint8 opcode, Line 950  ui_memblt(uint8 opcode,
950            /* src */ HBITMAP src, int srcx, int srcy)            /* src */ HBITMAP src, int srcx, int srcy)
951  {  {
952          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
953          XCopyArea(display, (Pixmap)src, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, (Pixmap) src, wnd, gc, srcx, srcy, cx, cy, x, y);
954          if (ownbackstore)          if (ownbackstore)
955                  XCopyArea(display, (Pixmap)src, backstore, gc, srcx, srcy,                  XCopyArea(display, (Pixmap) src, backstore, gc, srcx, srcy, cx, cy, x, y);
                           cx, cy, x, y);  
956          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
957  }  }
958    
# Line 974  void Line 960  void
960  ui_triblt(uint8 opcode,  ui_triblt(uint8 opcode,
961            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
962            /* src */ HBITMAP src, int srcx, int srcy,            /* src */ HBITMAP src, int srcx, int srcy,
963            /* brush */ BRUSH *brush, int bgcolour, int fgcolour)            /* brush */ BRUSH * brush, int bgcolour, int fgcolour)
964  {  {
965          /* This is potentially difficult to do in general. Until someone          /* This is potentially difficult to do in general. Until someone
966             comes up with a more efficient way of doing it I am using cases. */             comes up with a more efficient way of doing it I am using cases. */
# Line 983  ui_triblt(uint8 opcode, Line 969  ui_triblt(uint8 opcode,
969          {          {
970                  case 0x69:      /* PDSxxn */                  case 0x69:      /* PDSxxn */
971                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);
972                          ui_patblt(ROP2_NXOR, x, y, cx, cy,                          ui_patblt(ROP2_NXOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
973                          break;                          break;
974    
975                  case 0xb8:      /* PSDPxax */                  case 0xb8:      /* PSDPxax */
976                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
977                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);
978                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
979                          break;                          break;
980    
981                  case 0xc0:      /* PSa */                  case 0xc0:      /* PSa */
982                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);
983                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour,                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   fgcolour);  
984                          break;                          break;
985    
986                  default:                  default:
# Line 1010  ui_triblt(uint8 opcode, Line 992  ui_triblt(uint8 opcode,
992  void  void
993  ui_line(uint8 opcode,  ui_line(uint8 opcode,
994          /* dest */ int startx, int starty, int endx, int endy,          /* dest */ int startx, int starty, int endx, int endy,
995          /* pen */ PEN *pen)          /* pen */ PEN * pen)
996  {  {
997          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
998          SET_FOREGROUND(pen->colour);          SET_FOREGROUND(pen->colour);
# Line 1032  ui_rect( Line 1014  ui_rect(
1014  void  void
1015  ui_draw_glyph(int mixmode,  ui_draw_glyph(int mixmode,
1016                /* dest */ int x, int y, int cx, int cy,                /* dest */ int x, int y, int cx, int cy,
1017                /* src */ HGLYPH glyph, int srcx, int srcy, int bgcolour,                /* src */ HGLYPH glyph, int srcx, int srcy,
1018                int fgcolour)                int bgcolour, int fgcolour)
1019  {  {
1020          SET_FOREGROUND(fgcolour);          SET_FOREGROUND(fgcolour);
1021          SET_BACKGROUND(bgcolour);          SET_BACKGROUND(bgcolour);
1022    
1023          XSetFillStyle(display, gc, (mixmode == MIX_TRANSPARENT)          XSetFillStyle(display, gc,
1024                        ? FillStippled : FillOpaqueStippled);                        (mixmode == MIX_TRANSPARENT) ? FillStippled : FillOpaqueStippled);
1025          XSetStipple(display, gc, (Pixmap)glyph);          XSetStipple(display, gc, (Pixmap) glyph);
1026          XSetTSOrigin(display, gc, x, y);          XSetTSOrigin(display, gc, x, y);
1027    
1028          FILL_RECTANGLE(x, y, cx, cy);          FILL_RECTANGLE(x, y, cx, cy);
# Line 1057  ui_draw_glyph(int mixmode, Line 1039  ui_draw_glyph(int mixmode,
1039        if ((xyoffset & 0x80))\        if ((xyoffset & 0x80))\
1040          {\          {\
1041            if (flags & TEXT2_VERTICAL) \            if (flags & TEXT2_VERTICAL) \
1042              y += ttext[++idx] | (ttext[++idx] << 8);\              y += ttext[idx+1] | (ttext[idx+2] << 8);\
1043            else\            else\
1044              x += ttext[++idx] | (ttext[++idx] << 8);\              x += ttext[idx+1] | (ttext[idx+2] << 8);\
1045              idx += 2;\
1046          }\          }\
1047        else\        else\
1048          {\          {\
# Line 1082  ui_draw_glyph(int mixmode, Line 1065  ui_draw_glyph(int mixmode,
1065    
1066  void  void
1067  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,
1068               int clipx, int clipy, int clipcx, int clipcy, int boxx,               int clipx, int clipy, int clipcx, int clipcy,
1069               int boxy, int boxcx, int boxcy, int bgcolour,               int boxx, int boxy, int boxcx, int boxcy, int bgcolour,
1070               int fgcolour, uint8 * text, uint8 length)               int fgcolour, uint8 * text, uint8 length)
1071  {  {
1072          FONTGLYPH *glyph;          FONTGLYPH *glyph;
# Line 1102  ui_draw_text(uint8 font, uint8 flags, in Line 1085  ui_draw_text(uint8 font, uint8 flags, in
1085          }          }
1086    
1087          /* Paint text, character by character */          /* Paint text, character by character */
1088          for (i = 0; i < length;) {          for (i = 0; i < length;)
1089                  switch (text[i]) {          {
1090                  case 0xff:                  switch (text[i])
1091                          if (i + 2 < length)                  {
1092                                  cache_put_text(text[i + 1], text, text[i + 2]);                          case 0xff:
1093                          else {                                  if (i + 2 < length)
1094                                  error("this shouldn't be happening\n");                                          cache_put_text(text[i + 1], text, text[i + 2]);
1095                                    else
1096                                    {
1097                                            error("this shouldn't be happening\n");
1098                                            break;
1099                                    }
1100                                    /* this will move pointer from start to first character after FF command */
1101                                    length -= i + 3;
1102                                    text = &(text[i + 3]);
1103                                    i = 0;
1104                                  break;                                  break;
                         }  
                         /* this will move pointer from start to first character after FF command */  
                         length -= i + 3;  
                         text = &(text[i + 3]);  
                         i = 0;  
                         break;  
1105    
1106                  case 0xfe:                          case 0xfe:
1107                          entry = cache_get_text(text[i + 1]);                                  entry = cache_get_text(text[i + 1]);
1108                          if (entry != NULL) {                                  if (entry != NULL)
1109                                  if ((((uint8 *) (entry->data))[1] == 0)                                  {
1110                                      && (!(flags & TEXT2_IMPLICIT_X))) {                                          if ((((uint8 *) (entry->data))[1] ==
1111                                          if (flags & TEXT2_VERTICAL)                                               0) && (!(flags & TEXT2_IMPLICIT_X)))
1112                                                  y += text[i + 2];                                          {
1113                                                    if (flags & TEXT2_VERTICAL)
1114                                                            y += text[i + 2];
1115                                                    else
1116                                                            x += text[i + 2];
1117                                            }
1118                                            if (i + 2 < length)
1119                                                    i += 3;
1120                                          else                                          else
1121                                                  x += text[i + 2];                                                  i += 2;
1122                                            length -= i;
1123                                            /* this will move pointer from start to first character after FE command */
1124                                            text = &(text[i]);
1125                                            i = 0;
1126                                            for (j = 0; j < entry->size; j++)
1127                                                    DO_GLYPH(((uint8 *) (entry->data)), j);
1128                                  }                                  }
1129                                  if (i + 2 < length)                                  break;
                                         i += 3;  
                                 else  
                                         i += 2;  
                                 length -= i;  
                                 /* this will move pointer from start to first character after FE command */  
                                 text = &(text[i]);  
                                 i = 0;  
                                 for (j = 0; j < entry->size; j++)  
                                         DO_GLYPH(((uint8 *) (entry->data)), j);  
                         }  
                         break;  
1130    
1131                  default:                          default:
1132                          DO_GLYPH(text, i);                                  DO_GLYPH(text, i);
1133                          i++;                                  i++;
1134                          break;                                  break;
1135                  }                  }
1136          }          }
1137    
# Line 1158  ui_desktop_save(uint32 offset, int x, in Line 1146  ui_desktop_save(uint32 offset, int x, in
1146    
1147          if (ownbackstore)          if (ownbackstore)
1148          {          {
1149                  image = XGetImage(display, backstore, x, y, cx, cy, AllPlanes,                  image = XGetImage(display, backstore, x, y, cx, cy, AllPlanes, ZPixmap);
                                   ZPixmap);  
1150          }          }
1151          else          else
1152          {          {
1153                  pix = XCreatePixmap(display, wnd, cx, cy, depth);                  pix = XCreatePixmap(display, wnd, cx, cy, depth);
1154                  XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);                  XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);
1155                  image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes,                  image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes, ZPixmap);
                                   ZPixmap);  
1156                  XFreePixmap(display, pix);                  XFreePixmap(display, pix);
1157          }          }
1158    
1159          offset *= bpp/8;          offset *= bpp / 8;
1160          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);  
1161    
1162          XDestroyImage(image);          XDestroyImage(image);
1163  }  }
# Line 1183  ui_desktop_restore(uint32 offset, int x, Line 1168  ui_desktop_restore(uint32 offset, int x,
1168          XImage *image;          XImage *image;
1169          uint8 *data;          uint8 *data;
1170    
1171          offset *= bpp/8;          offset *= bpp / 8;
1172          data = cache_get_desktop(offset, cx, cy, bpp/8);          data = cache_get_desktop(offset, cx, cy, bpp / 8);
1173          if (data == NULL)          if (data == NULL)
1174                  return;                  return;
1175    
1176          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
1177                               0, data, cx, cy, BitmapPad(display),                               (char *) data, cx, cy, BitmapPad(display), cx * bpp / 8);
                              cx * bpp/8);  
1178    
1179          if (ownbackstore)          if (ownbackstore)
1180          {          {

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

  ViewVC Help
Powered by ViewVC 1.1.26