/[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 29 by matty, Fri Sep 14 03:38:39 2001 UTC revision 281 by jsorg71, Mon Dec 2 22:57:47 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-Windows     User interface services - X Window System
4     Copyright (C) Matthew Chapman 1999-2000     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
8     the Free Software Foundation; either version 2 of the License, or     the Free Software Foundation; either version 2 of the License, or
9     (at your option) any later version.     (at your option) any later version.
10      
11     This program is distributed in the hope that it will be useful,     This program is distributed in the hope that it will be useful,
12     but WITHOUT ANY WARRANTY; without even the implied warranty of     but WITHOUT ANY WARRANTY; without even the implied warranty of
13     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14     GNU General Public License for more details.     GNU General Public License for more details.
15      
16     You should have received a copy of the GNU General Public License     You should have received a copy of the GNU General Public License
17     along with this program; if not, write to the Free Software     along with this program; if not, write to the Free Software
18     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
# Line 21  Line 21 
21  #include <X11/Xlib.h>  #include <X11/Xlib.h>
22  #include <X11/Xutil.h>  #include <X11/Xutil.h>
23  #include <time.h>  #include <time.h>
24    #include <errno.h>
25  #include "rdesktop.h"  #include "rdesktop.h"
26    
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  static Display *display;  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;
39    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 BOOL backpixmap;  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 */
53    static BOOL host_be;
54    static BOOL xserver_be;
55    
56    /* software backing store */
57    static BOOL ownbackstore;
58    static Pixmap backstore;
59    
60    /* MWM decorations */
61    #define MWM_HINTS_DECORATIONS   (1L << 1)
62    #define PROP_MOTIF_WM_HINTS_ELEMENTS    5
63    typedef struct
64    {
65            unsigned long flags;
66            unsigned long functions;
67            unsigned long decorations;
68            long inputMode;
69            unsigned long status;
70    }
71    PropMotifWmHints;
72    
73  static BOOL owncolmap;  
74    #define FILL_RECTANGLE(x,y,cx,cy)\
75    { \
76            XFillRectangle(display, wnd, gc, x, y, cx, cy); \
77            if (ownbackstore) \
78                    XFillRectangle(display, backstore, gc, x, y, cx, cy); \
79    }
80    
81    #define FILL_RECTANGLE_FAST(x,y,cx,cy)\
82    { \
83            XFillRectangle(display, ownbackstore ? backstore : wnd, gc, x, y, cx, cy); \
84    }
85    
86    /* colour maps */
87    BOOL owncolmap = False;
88  static Colormap xcolmap;  static Colormap xcolmap;
 static uint32 white;  
89  static uint32 *colmap;  static uint32 *colmap;
90    
91  #define TRANSLATE(col)          ( owncolmap ? col : colmap[col] )  #define TRANSLATE(col)          ( owncolmap ? col : translate_colour(colmap[col]) )
92  #define SET_FOREGROUND(col)     XSetForeground(display, gc, TRANSLATE(col));  #define SET_FOREGROUND(col)     XSetForeground(display, gc, TRANSLATE(col));
93  #define SET_BACKGROUND(col)     XSetBackground(display, gc, TRANSLATE(col));  #define SET_BACKGROUND(col)     XSetBackground(display, gc, TRANSLATE(col));
94    
# Line 67  static int rop2_map[] = { Line 114  static int rop2_map[] = {
114  #define SET_FUNCTION(rop2)      { if (rop2 != ROP2_COPY) XSetFunction(display, gc, rop2_map[rop2]); }  #define SET_FUNCTION(rop2)      { if (rop2 != ROP2_COPY) XSetFunction(display, gc, rop2_map[rop2]); }
115  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }  #define RESET_FUNCTION(rop2)    { if (rop2 != ROP2_COPY) XSetFunction(display, gc, GXcopy); }
116    
117    void
118    mwm_hide_decorations(void)
119    {
120            PropMotifWmHints motif_hints;
121            Atom hintsatom;
122    
123            /* setup the property */
124            motif_hints.flags = MWM_HINTS_DECORATIONS;
125            motif_hints.decorations = 0;
126    
127            /* get the atom for the property */
128            hintsatom = XInternAtom(display, "_MOTIF_WM_HINTS", False);
129            if (!hintsatom)
130            {
131                    error("Failed to get atom _MOTIF_WM_HINTS\n");
132                    return;
133            }
134    
135            XChangeProperty(display, wnd, hintsatom, hintsatom, 32, PropModeReplace,
136                            (unsigned char *) &motif_hints, PROP_MOTIF_WM_HINTS_ELEMENTS);
137    }
138    
139  static void  static void
140  translate8(uint8 *data, uint8 *out, uint8 *end)  translate8(uint8 * data, uint8 * out, uint8 * end)
141  {  {
142          while (out < end)          while (out < end)
143                  *(out++) = (uint8)colmap[*(data++)];                  *(out++) = (uint8) colmap[*(data++)];
144  }  }
145    
146  static void  static void
147  translate16(uint8 *data, uint16 *out, uint16 *end)  translate16(uint8 * data, uint16 * out, uint16 * end)
148  {  {
149          while (out < end)          while (out < end)
150                  *(out++) = (uint16)colmap[*(data++)];                  *(out++) = (uint16) colmap[*(data++)];
151  }  }
152    
153  /* XXX endianness */  /* little endian - conversion happens when colourmap is built */
154  static void  static void
155  translate24(uint8 *data, uint8 *out, uint8 *end)  translate24(uint8 * data, uint8 * out, uint8 * end)
156  {  {
157          uint32 value;          uint32 value;
158    
# Line 97  translate24(uint8 *data, uint8 *out, uin Line 166  translate24(uint8 *data, uint8 *out, uin
166  }  }
167    
168  static void  static void
169  translate32(uint8 *data, uint32 *out, uint32 *end)  translate32(uint8 * data, uint32 * out, uint32 * end)
170  {  {
171          while (out < end)          while (out < end)
172                  *(out++) = colmap[*(data++)];                  *(out++) = colmap[*(data++)];
173  }  }
174    
175  static uint8 *  static uint8 *
176  translate(int width, int height, uint8 *data)  translate_image(int width, int height, uint8 * data)
177  {  {
178          int size = width * height * bpp/8;          int size = width * height * bpp / 8;
179          uint8 *out = xmalloc(size);          uint8 *out = xmalloc(size);
180          uint8 *end = out + size;          uint8 *end = out + size;
181    
# Line 117  translate(int width, int height, uint8 * Line 186  translate(int width, int height, uint8 *
186                          break;                          break;
187    
188                  case 16:                  case 16:
189                          translate16(data, (uint16 *)out, (uint16 *)end);                          translate16(data, (uint16 *) out, (uint16 *) end);
190                          break;                          break;
191    
192                  case 24:                  case 24:
# Line 125  translate(int width, int height, uint8 * Line 194  translate(int width, int height, uint8 *
194                          break;                          break;
195    
196                  case 32:                  case 32:
197                          translate32(data, (uint32 *)out, (uint32 *)end);                          translate32(data, (uint32 *) out, (uint32 *) end);
198                          break;                          break;
199          }          }
200    
201          return out;          return out;
202  }  }
203    
204  #define L_ENDIAN  #define BSWAP16(x) { x = (((x & 0xff) << 8) | (x >> 8)); }
205  int screen_msbfirst = 0;  #define BSWAP24(x) { x = (((x & 0xff) << 16) | (x >> 16) | ((x >> 8) & 0xff00)); }
206    #define BSWAP32(x) { x = (((x & 0xff00ff) << 8) | ((x >> 8) & 0xff00ff)); \
207                            x = (x << 16) | (x >> 16); }
208    
209    static uint32
210    translate_colour(uint32 colour)
211    {
212            switch (bpp)
213            {
214                    case 16:
215                            if (host_be != xserver_be)
216                                    BSWAP16(colour);
217                            break;
218    
219                    case 24:
220                            if (xserver_be)
221                                    BSWAP24(colour);
222                            break;
223    
224                    case 32:
225                            if (host_be != xserver_be)
226                                    BSWAP32(colour);
227                            break;
228            }
229    
230            return colour;
231    }
232    
233    BOOL
234    get_key_state(unsigned int state, uint32 keysym)
235    {
236            int modifierpos, key, keysymMask = 0;
237            int offset;
238    
239            KeyCode keycode = XKeysymToKeycode(display, keysym);
240    
241            if (keycode == NoSymbol)
242                    return False;
243    
244            for (modifierpos = 0; modifierpos < 8; modifierpos++)
245            {
246                    offset = mod_map->max_keypermod * modifierpos;
247    
248                    for (key = 0; key < mod_map->max_keypermod; key++)
249                    {
250                            if (mod_map->modifiermap[offset + key] == keycode)
251                                    keysymMask |= 1 << modifierpos;
252                    }
253            }
254    
255            return (state & keysymMask) ? True : False;
256    }
257    
258  BOOL  BOOL
259  ui_create_window(char *title)  ui_init(void)
260  {  {
         XSetWindowAttributes attribs;  
         XClassHint *classhints;  
         XSizeHints *sizehints;  
         unsigned long input_mask;  
261          XPixmapFormatValues *pfm;          XPixmapFormatValues *pfm;
262          Screen *screen;          uint16 test;
263          int i;          int i;
264    
   
265          display = XOpenDisplay(NULL);          display = XOpenDisplay(NULL);
266          if (display == NULL)          if (display == NULL)
267          {          {
268                  ERROR("Failed to open display\n");                  error("Failed to open display: %s\n", XDisplayName(NULL));
269                  return False;                  return False;
270          }          }
271    
272            x_socket = ConnectionNumber(display);
273          screen = DefaultScreenOfDisplay(display);          screen = DefaultScreenOfDisplay(display);
274          visual = DefaultVisualOfScreen(screen);          visual = DefaultVisualOfScreen(screen);
275          depth = DefaultDepthOfScreen(screen);          depth = DefaultDepthOfScreen(screen);
# Line 166  ui_create_window(char *title) Line 281  ui_create_window(char *title)
281                     desirable, e.g. 24 bits->32 bits. */                     desirable, e.g. 24 bits->32 bits. */
282                  while (i--)                  while (i--)
283                  {                  {
284                          if ((pfm[i].depth == depth)                          if ((pfm[i].depth == depth) && (pfm[i].bits_per_pixel > bpp))
                             && (pfm[i].bits_per_pixel > bpp))  
285                          {                          {
286                                  bpp = pfm[i].bits_per_pixel;                                  bpp = pfm[i].bits_per_pixel;
287                          }                          }
# Line 177  ui_create_window(char *title) Line 291  ui_create_window(char *title)
291    
292          if (bpp < 8)          if (bpp < 8)
293          {          {
294                  ERROR("Less than 8 bpp not currently supported.\n");                  error("Less than 8 bpp not currently supported.\n");
295                  XCloseDisplay(display);                  XCloseDisplay(display);
296                  return False;                  return False;
297          }          }
298    
299          if (depth <= 8)          if (owncolmap != True)
300                  owncolmap = True;          {
         else  
301                  xcolmap = DefaultColormapOfScreen(screen);                  xcolmap = DefaultColormapOfScreen(screen);
302                    if (depth <= 8)
303                    {
304                            printf("You're using a screen depth of 8-bits or lower\n");
305                            printf("If you get scewed colours, try the -C switch\n");
306                    }
307            }
308    
309          white = WhitePixelOfScreen(screen);          gc = XCreateGC(display, RootWindowOfScreen(screen), 0, NULL);
310          attribs.background_pixel = BlackPixelOfScreen(screen);  
311          attribs.backing_store = DoesBackingStore(screen);          if (DoesBackingStore(screen) != Always)
312                    ownbackstore = True;
313    
314          if (attribs.backing_store == NotUseful)          test = 1;
315                  backpixmap = True;          host_be = !(BOOL) (*(uint8 *) (&test));
316            xserver_be = (ImageByteOrder(display) == MSBFirst);
317    
318            if ((width == 0) || (height == 0))
319            {
320                    /* Fetch geometry from _NET_WORKAREA */
321                    uint32 xpos, ypos;
322    
323                    if (get_current_workarea(&xpos, &ypos, &width, &height) < 0)
324                    {
325                            error("Failed to get workarea.\n");
326                            error("Perhaps your window manager does not support EWMH?\n");
327                            error("Defaulting to geometry 800x600\n");
328                            width = 800;
329                            height = 600;
330                    }
331            }
332    
333          if (fullscreen)          if (fullscreen)
334          {          {
                 attribs.override_redirect = True;  
335                  width = WidthOfScreen(screen);                  width = WidthOfScreen(screen);
336                  height = HeightOfScreen(screen);                  height = HeightOfScreen(screen);
337          }          }
338          else  
339            /* make sure width is a multiple of 4 */
340            width = (width + 3) & ~3;
341    
342            if (ownbackstore)
343          {          {
344                  attribs.override_redirect = False;                  backstore =
345                            XCreatePixmap(display, RootWindowOfScreen(screen), width, height, depth);
346    
347                    /* clear to prevent rubbish being exposed at startup */
348                    XSetForeground(display, gc, BlackPixelOfScreen(screen));
349                    XFillRectangle(display, backstore, gc, 0, 0, width, height);
350          }          }
351    
352          width &= ~3; /* make width a multiple of 32 bits */          mod_map = XGetModifierMapping(display);
353    
354            if (enable_compose)
355                    IM = XOpenIM(display, NULL, NULL, NULL);
356    
357          wnd = XCreateWindow(display, RootWindowOfScreen(screen),          xkeymap_init();
358                              0, 0, width, height, 0, CopyFromParent,          return True;
359                              InputOutput, CopyFromParent,  }
360                              CWBackingStore | CWBackPixel | CWOverrideRedirect,  
361                              &attribs);  void
362    ui_deinit(void)
363    {
364            if (IM != NULL)
365                    XCloseIM(IM);
366    
367            XFreeModifiermap(mod_map);
368    
369            if (ownbackstore)
370                    XFreePixmap(display, backstore);
371    
372            XFreeGC(display, gc);
373            XCloseDisplay(display);
374            display = NULL;
375    }
376    
377    BOOL
378    ui_create_window(void)
379    {
380            XSetWindowAttributes attribs;
381            XClassHint *classhints;
382            XSizeHints *sizehints;
383            int wndwidth, wndheight;
384            long input_mask, ic_input_mask;
385            XEvent xevent;
386    
387            wndwidth = fullscreen ? WidthOfScreen(screen) : width;
388            wndheight = fullscreen ? HeightOfScreen(screen) : height;
389    
390            attribs.background_pixel = BlackPixelOfScreen(screen);
391            attribs.backing_store = ownbackstore ? NotUseful : Always;
392            attribs.override_redirect = fullscreen;
393    
394            wnd = XCreateWindow(display, RootWindowOfScreen(screen), 0, 0, wndwidth, wndheight,
395                                0, CopyFromParent, InputOutput, CopyFromParent,
396                                CWBackPixel | CWBackingStore | CWOverrideRedirect, &attribs);
397    
398          XStoreName(display, wnd, title);          XStoreName(display, wnd, title);
399    
400            if (hide_decorations)
401                    mwm_hide_decorations();
402    
403          classhints = XAllocClassHint();          classhints = XAllocClassHint();
404          if (classhints != NULL)          if (classhints != NULL)
405          {          {
# Line 233  ui_create_window(char *title) Line 418  ui_create_window(char *title)
418                  XFree(sizehints);                  XFree(sizehints);
419          }          }
420    
421          input_mask = KeyPressMask | KeyReleaseMask          input_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask |
422                          | ButtonPressMask | ButtonReleaseMask                  VisibilityChangeMask | FocusChangeMask;
                         | EnterWindowMask | LeaveWindowMask;  
423    
424          if (sendmotion)          if (sendmotion)
425                  input_mask |= PointerMotionMask;                  input_mask |= PointerMotionMask;
426            if (ownbackstore)
427                    input_mask |= ExposureMask;
428            if (fullscreen || grab_keyboard)
429                    input_mask |= EnterWindowMask;
430            if (grab_keyboard)
431                    input_mask |= LeaveWindowMask;
432    
433          XSelectInput(display, wnd, input_mask);          if (IM != NULL)
434          gc = XCreateGC(display, wnd, 0, NULL);          {
435                    IC = XCreateIC(IM, XNInputStyle, (XIMPreeditNothing | XIMStatusNothing),
436                                   XNClientWindow, wnd, XNFocusWindow, wnd, NULL);
437    
438                    if ((IC != NULL)
439                        && (XGetICValues(IC, XNFilterEvents, &ic_input_mask, NULL) == NULL))
440                            input_mask |= ic_input_mask;
441            }
442    
443            XSelectInput(display, wnd, input_mask);
444          XMapWindow(display, wnd);          XMapWindow(display, wnd);
445    
446            /* wait for VisibilityNotify */
447            do
448            {
449                    XMaskEvent(display, VisibilityChangeMask, &xevent);
450            }
451            while (xevent.type != VisibilityNotify);
452    
453            focused = False;
454            mouse_in_wnd = False;
455    
456            /* handle the WM_DELETE_WINDOW protocol */
457            protocol_atom = XInternAtom(display, "WM_PROTOCOLS", True);
458            kill_atom = XInternAtom(display, "WM_DELETE_WINDOW", True);
459            XSetWMProtocols(display, wnd, &kill_atom, 1);
460    
461          return True;          return True;
462  }  }
463    
464  void  void
465  ui_destroy_window()  ui_destroy_window(void)
466  {  {
467          XFreeGC(display, gc);          if (IC != NULL)
468                    XDestroyIC(IC);
469    
470          XDestroyWindow(display, wnd);          XDestroyWindow(display, wnd);
         XCloseDisplay(display);  
         display = NULL;  
471  }  }
472    
473  static uint8  void
474  xwin_translate_key(unsigned long key)  xwin_toggle_fullscreen(void)
475  {  {
476          DEBUG("KEY(code=0x%lx)\n", key);          Pixmap contents = 0;
477    
478          if ((key > 8) && (key <= 0x60))          if (!ownbackstore)
479                  return (key - 8);          {
480                    /* need to save contents of window */
481                    contents = XCreatePixmap(display, wnd, width, height, depth);
482                    XCopyArea(display, wnd, contents, gc, 0, 0, width, height, 0, 0);
483            }
484    
485          switch (key)          ui_destroy_window();
486          {          fullscreen = !fullscreen;
487                  case 0x61:      /* home */          ui_create_window();
                         return 0x47 | 0x80;  
                 case 0x62:      /* up arrow */  
                         return 0x48 | 0x80;  
                 case 0x63:      /* page up */  
                         return 0x49 | 0x80;  
                 case 0x64:      /* left arrow */  
                         return 0x4b | 0x80;  
                 case 0x66:      /* right arrow */  
                         return 0x4d | 0x80;  
                 case 0x67:      /* end */  
                         return 0x4f | 0x80;  
                 case 0x68:      /* down arrow */  
                         return 0x50 | 0x80;  
                 case 0x69:      /* page down */  
                         return 0x51 | 0x80;  
                 case 0x6a:      /* insert */  
                         return 0x52 | 0x80;  
                 case 0x6b:      /* delete */  
                         return 0x53 | 0x80;  
                 case 0x6c:      /* keypad enter */  
                         return 0x1c | 0x80;  
                 case 0x6d:      /* right ctrl */  
                         return 0x1d | 0x80;  
                 case 0x6f:      /* ctrl - print screen */  
                         return 0x37 | 0x80;  
                 case 0x70:      /* keypad '/' */  
                         return 0x35 | 0x80;  
                 case 0x71:      /* right alt */  
                         return 0x38 | 0x80;  
                 case 0x72:      /* ctrl break */  
                         return 0x46 | 0x80;  
                 case 0x73:      /* left window key */  
                         return 0xff;    /* real scancode is 5b */  
                 case 0x74:      /* right window key */  
                         return 0xff;    /* real scancode is 5c */  
                 case 0x75:      /* menu key */  
                         return 0x5d | 0x80;  
         }  
   
         return 0;  
 }  
   
 static uint16  
 xwin_translate_mouse(unsigned long button)  
 {  
         switch (button)  
         {  
                 case Button1:   /* left */  
                         return MOUSE_FLAG_BUTTON1;  
                 case Button2:   /* middle */  
                         return MOUSE_FLAG_BUTTON3;  
                 case Button3:   /* right */  
                         return MOUSE_FLAG_BUTTON2;  
         }  
   
         return 0;  
 }  
   
 void  
 ui_process_events()  
 {  
         XEvent event;  
         uint8 scancode;  
         uint16 button;  
         uint32 ev_time;  
488    
489          if (display == NULL)          XDefineCursor(display, wnd, current_cursor);
490                  return;  
491            if (!ownbackstore)
492            {
493                    XCopyArea(display, contents, wnd, gc, 0, 0, width, height, 0, 0);
494                    XFreePixmap(display, contents);
495            }
496    }
497    
498    /* Process all events in Xlib queue
499       Returns 0 after user quit, 1 otherwise */
500    static int
501    xwin_process_events(void)
502    {
503            XEvent xevent;
504            KeySym keysym;
505            uint16 button, flags;
506            uint32 ev_time;
507            key_translation tr;
508            char str[256];
509            Status status;
510            unsigned int state;
511            Window wdummy;
512            int dummy;
513    
514          while (XCheckWindowEvent(display, wnd, ~0, &event))          while (XPending(display) > 0)
515          {          {
516                  ev_time = time(NULL);                  XNextEvent(display, &xevent);
517    
518                  switch (event.type)                  if ((IC != NULL) && (XFilterEvent(&xevent, None) == True))
519                  {                  {
520                            DEBUG_KBD(("Filtering event\n"));
521                            continue;
522                    }
523    
524                    flags = 0;
525    
526                    switch (xevent.type)
527                    {
528                            case ClientMessage:
529                                    /* the window manager told us to quit */
530                                    if ((xevent.xclient.message_type == protocol_atom)
531                                        && (xevent.xclient.data.l[0] == kill_atom))
532                                            /* Quit */
533                                            return 0;
534                                    break;
535    
536                          case KeyPress:                          case KeyPress:
537                                  scancode = xwin_translate_key(event.xkey.keycode);                                  if (IC != NULL)
538                                  if (scancode == 0)                                          /* Multi_key compatible version */
539                                    {
540                                            XmbLookupString(IC,
541                                                            (XKeyPressedEvent *) &
542                                                            xevent, str, sizeof(str), &keysym, &status);
543                                            if (!((status == XLookupKeySym) || (status == XLookupBoth)))
544                                            {
545                                                    error("XmbLookupString failed with status 0x%x\n",
546                                                          status);
547                                                    break;
548                                            }
549                                    }
550                                    else
551                                    {
552                                            /* Plain old XLookupString */
553                                            DEBUG_KBD(("\nNo input context, using XLookupString\n"));
554                                            XLookupString((XKeyEvent *) & xevent,
555                                                          str, sizeof(str), &keysym, NULL);
556                                    }
557    
558                                    DEBUG_KBD(("KeyPress for (keysym 0x%lx, %s)\n", keysym,
559                                               get_ksname(keysym)));
560    
561                                    ev_time = time(NULL);
562                                    if (handle_special_keys(keysym, xevent.xkey.state, ev_time, True))
563                                          break;                                          break;
564    
565                                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE, 0,                                  tr = xkeymap_translate_key(keysym,
566                                                 scancode, 0);                                                             xevent.xkey.keycode, xevent.xkey.state);
567    
568                                    if (tr.scancode == 0)
569                                            break;
570    
571                                    ensure_remote_modifiers(ev_time, tr);
572    
573                                    rdp_send_scancode(ev_time, RDP_KEYPRESS, tr.scancode);
574                                  break;                                  break;
575    
576                          case KeyRelease:                          case KeyRelease:
577                                  scancode = xwin_translate_key(event.xkey.keycode);                                  XLookupString((XKeyEvent *) & xevent, str,
578                                  if (scancode == 0)                                                sizeof(str), &keysym, NULL);
579    
580                                    DEBUG_KBD(("\nKeyRelease for (keysym 0x%lx, %s)\n", keysym,
581                                               get_ksname(keysym)));
582    
583                                    ev_time = time(NULL);
584                                    if (handle_special_keys(keysym, xevent.xkey.state, ev_time, False))
585                                          break;                                          break;
586    
587                                  rdp_send_input(ev_time, RDP_INPUT_SCANCODE,                                  tr = xkeymap_translate_key(keysym,
588                                                 KBD_FLAG_DOWN | KBD_FLAG_UP,                                                             xevent.xkey.keycode, xevent.xkey.state);
                                                scancode, 0);  
                                 break;  
589    
590                          case ButtonPress:                                  if (tr.scancode == 0)
                                 button = xwin_translate_mouse(event.xbutton.button);  
                                 if (button == 0)  
591                                          break;                                          break;
592    
593                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_scancode(ev_time, RDP_KEYRELEASE, tr.scancode);
                                                button | MOUSE_FLAG_DOWN,  
                                                event.xbutton.x,  
                                                event.xbutton.y);  
594                                  break;                                  break;
595    
596                            case ButtonPress:
597                                    flags = MOUSE_FLAG_DOWN;
598                                    /* fall through */
599    
600                          case ButtonRelease:                          case ButtonRelease:
601                                  button = xwin_translate_mouse(event.xbutton.button);                                  button = xkeymap_translate_button(xevent.xbutton.button);
602                                  if (button == 0)                                  if (button == 0)
603                                          break;                                          break;
604    
605                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
606                                                 button,                                                 flags | button, xevent.xbutton.x, xevent.xbutton.y);
                                                event.xbutton.x,  
                                                event.xbutton.y);  
607                                  break;                                  break;
608    
609                          case MotionNotify:                          case MotionNotify:
610                                  rdp_send_input(ev_time, RDP_INPUT_MOUSE,                                  rdp_send_input(time(NULL), RDP_INPUT_MOUSE,
611                                                 MOUSE_FLAG_MOVE,                                                 MOUSE_FLAG_MOVE, xevent.xmotion.x, xevent.xmotion.y);
612                                                 event.xmotion.x,                                  break;
613                                                 event.xmotion.y);  
614                            case FocusIn:
615                                    if (xevent.xfocus.mode == NotifyGrab)
616                                            break;
617                                    focused = True;
618                                    XQueryPointer(display, wnd, &wdummy, &wdummy, &dummy, &dummy,
619                                                  &dummy, &dummy, &state);
620                                    reset_modifier_keys(state);
621                                    if (grab_keyboard && mouse_in_wnd)
622                                            XGrabKeyboard(display, wnd, True,
623                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
624                                    break;
625    
626                            case FocusOut:
627                                    if (xevent.xfocus.mode == NotifyUngrab)
628                                            break;
629                                    focused = False;
630                                    if (xevent.xfocus.mode == NotifyWhileGrabbed)
631                                            XUngrabKeyboard(display, CurrentTime);
632                                  break;                                  break;
633    
634                          case EnterNotify:                          case EnterNotify:
635                                  XGrabKeyboard(display, wnd, True, GrabModeAsync,                                  /* we only register for this event when in fullscreen mode */
636                                                GrabModeAsync, CurrentTime);                                  /* or grab_keyboard */
637                                    mouse_in_wnd = True;
638                                    if (fullscreen)
639                                    {
640                                            XSetInputFocus(display, wnd, RevertToPointerRoot,
641                                                           CurrentTime);
642                                            break;
643                                    }
644                                    if (focused)
645                                            XGrabKeyboard(display, wnd, True,
646                                                          GrabModeAsync, GrabModeAsync, CurrentTime);
647                                  break;                                  break;
648    
649                          case LeaveNotify:                          case LeaveNotify:
650                                    /* we only register for this event when grab_keyboard */
651                                    mouse_in_wnd = False;
652                                  XUngrabKeyboard(display, CurrentTime);                                  XUngrabKeyboard(display, CurrentTime);
653                                  break;                                  break;
654    
655                            case Expose:
656                                    XCopyArea(display, backstore, wnd, gc,
657                                              xevent.xexpose.x, xevent.xexpose.y,
658                                              xevent.xexpose.width,
659                                              xevent.xexpose.height,
660                                              xevent.xexpose.x, xevent.xexpose.y);
661                                    break;
662    
663                            case MappingNotify:
664                                    /* Refresh keyboard mapping if it has changed. This is important for
665                                       Xvnc, since it allocates keycodes dynamically */
666                                    if (xevent.xmapping.request == MappingKeyboard
667                                        || xevent.xmapping.request == MappingModifier)
668                                            XRefreshKeyboardMapping(&xevent.xmapping);
669    
670                                    if (xevent.xmapping.request == MappingModifier)
671                                    {
672                                            XFreeModifiermap(mod_map);
673                                            mod_map = XGetModifierMapping(display);
674                                    }
675                                    break;
676    
677                  }                  }
678          }          }
679            /* Keep going */
680            return 1;
681    }
682    
683    /* Returns 0 after user quit, 1 otherwise */
684    int
685    ui_select(int rdp_socket)
686    {
687            int n = (rdp_socket > x_socket) ? rdp_socket + 1 : x_socket + 1;
688            fd_set rfds;
689    
690            FD_ZERO(&rfds);
691    
692            while (True)
693            {
694                    /* Process any events already waiting */
695                    if (!xwin_process_events())
696                            /* User quit */
697                            return 0;
698    
699                    FD_ZERO(&rfds);
700                    FD_SET(rdp_socket, &rfds);
701                    FD_SET(x_socket, &rfds);
702    
703                    switch (select(n, &rfds, NULL, NULL, NULL))
704                    {
705                            case -1:
706                                    error("select: %s\n", strerror(errno));
707    
708                            case 0:
709                                    continue;
710                    }
711    
712                    if (FD_ISSET(rdp_socket, &rfds))
713                            return 1;
714            }
715  }  }
716    
717  void  void
# Line 409  ui_move_pointer(int x, int y) Line 721  ui_move_pointer(int x, int y)
721  }  }
722    
723  HBITMAP  HBITMAP
724  ui_create_bitmap(int width, int height, uint8 *data)  ui_create_bitmap(int width, int height, uint8 * data)
725  {  {
726          XImage *image;          XImage *image;
727          Pixmap bitmap;          Pixmap bitmap;
728          uint8 *tdata;          uint8 *tdata;
729    
730          tdata = (owncolmap ? data : translate(width, height, data));          tdata = (owncolmap ? data : translate_image(width, height, data));
731          bitmap = XCreatePixmap(display, wnd, width, height, depth);          bitmap = XCreatePixmap(display, wnd, width, height, depth);
732          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
733                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
734    
735          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);          XPutImage(display, bitmap, gc, image, 0, 0, 0, 0, width, height);
736    
# Line 429  ui_create_bitmap(int width, int height, Line 741  ui_create_bitmap(int width, int height,
741  }  }
742    
743  void  void
744  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)  
745  {  {
746          XImage *image;          XImage *image;
747          uint8 *tdata;          uint8 *tdata;
748    
749          tdata = (owncolmap ? data : translate(width, height, data));          tdata = (owncolmap ? data : translate_image(width, height, data));
750          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
751                               0, tdata, width, height, 8, 0);                               (char *) tdata, width, height, 8, 0);
752    
753          XPutImage(display, wnd, gc, image, 0, 0, x, y, cx, cy);          if (ownbackstore)
754            {
755                    XPutImage(display, backstore, gc, image, 0, 0, x, y, cx, cy);
756                    XCopyArea(display, backstore, wnd, gc, x, y, cx, cy, x, y);
757            }
758            else
759            {
760                    XPutImage(display, wnd, gc, image, 0, 0, x, y, cx, cy);
761            }
762    
763          XFree(image);          XFree(image);
764          if (!owncolmap)          if (!owncolmap)
# Line 449  ui_paint_bitmap(int x, int y, int cx, in Line 768  ui_paint_bitmap(int x, int y, int cx, in
768  void  void
769  ui_destroy_bitmap(HBITMAP bmp)  ui_destroy_bitmap(HBITMAP bmp)
770  {  {
771          XFreePixmap(display, (Pixmap)bmp);          XFreePixmap(display, (Pixmap) bmp);
772  }  }
773    
774  HGLYPH  HGLYPH
775  ui_create_glyph(int width, int height, uint8 *data)  ui_create_glyph(int width, int height, uint8 * data)
776  {  {
777          XImage *image;          XImage *image;
778          Pixmap bitmap;          Pixmap bitmap;
# Line 465  ui_create_glyph(int width, int height, u Line 784  ui_create_glyph(int width, int height, u
784          bitmap = XCreatePixmap(display, wnd, width, height, 1);          bitmap = XCreatePixmap(display, wnd, width, height, 1);
785          gc = XCreateGC(display, bitmap, 0, NULL);          gc = XCreateGC(display, bitmap, 0, NULL);
786    
787          image = XCreateImage(display, visual, 1, ZPixmap, 0,          image = XCreateImage(display, visual, 1, ZPixmap, 0, (char *) data,
788                               data, width, height, 8, scanline);                               width, height, 8, scanline);
789          image->byte_order = MSBFirst;          image->byte_order = MSBFirst;
790          image->bitmap_bit_order = MSBFirst;          image->bitmap_bit_order = MSBFirst;
791          XInitImage(image);          XInitImage(image);
# Line 475  ui_create_glyph(int width, int height, u Line 794  ui_create_glyph(int width, int height, u
794    
795          XFree(image);          XFree(image);
796          XFreeGC(display, gc);          XFreeGC(display, gc);
797          return (HGLYPH)bitmap;          return (HGLYPH) bitmap;
798  }  }
799    
800  void  void
801  ui_destroy_glyph(HGLYPH glyph)  ui_destroy_glyph(HGLYPH glyph)
802  {  {
803          XFreePixmap(display, (Pixmap)glyph);          XFreePixmap(display, (Pixmap) glyph);
804  }  }
805    
806  HCURSOR  HCURSOR
807  ui_create_cursor(unsigned int x, unsigned int y, int width,  ui_create_cursor(unsigned int x, unsigned int y, int width, int height,
808                   int height, uint8 *andmask, uint8 *xormask)                   uint8 * andmask, uint8 * xormask)
809  {  {
810          HGLYPH maskglyph, cursorglyph;          HGLYPH maskglyph, cursorglyph;
811          XColor bg, fg;          XColor bg, fg;
# Line 543  ui_create_cursor(unsigned int x, unsigne Line 862  ui_create_cursor(unsigned int x, unsigne
862    
863          cursorglyph = ui_create_glyph(width, height, cursor);          cursorglyph = ui_create_glyph(width, height, cursor);
864          maskglyph = ui_create_glyph(width, height, mask);          maskglyph = ui_create_glyph(width, height, mask);
865            
866          xcursor = XCreatePixmapCursor(display, (Pixmap)cursorglyph,          xcursor =
867                                  (Pixmap)maskglyph, &fg, &bg, x, y);                  XCreatePixmapCursor(display, (Pixmap) cursorglyph,
868                                        (Pixmap) maskglyph, &fg, &bg, x, y);
869    
870          ui_destroy_glyph(maskglyph);          ui_destroy_glyph(maskglyph);
871          ui_destroy_glyph(cursorglyph);          ui_destroy_glyph(cursorglyph);
872          xfree(mask);          xfree(mask);
873          xfree(cursor);          xfree(cursor);
874          return (HCURSOR)xcursor;          return (HCURSOR) xcursor;
875  }  }
876    
877  void  void
878  ui_set_cursor(HCURSOR cursor)  ui_set_cursor(HCURSOR cursor)
879  {  {
880          XDefineCursor(display, wnd, (Cursor)cursor);          current_cursor = (Cursor) cursor;
881            XDefineCursor(display, wnd, current_cursor);
882  }  }
883    
884  void  void
885  ui_destroy_cursor(HCURSOR cursor)  ui_destroy_cursor(HCURSOR cursor)
886  {  {
887          XFreeCursor(display, (Cursor)cursor);          XFreeCursor(display, (Cursor) cursor);
888  }  }
889    
890  #define MAKE_XCOLOR(xc,c) \  #define MAKE_XCOLOR(xc,c) \
# Line 572  ui_destroy_cursor(HCURSOR cursor) Line 893  ui_destroy_cursor(HCURSOR cursor)
893                  (xc)->blue  = ((c)->blue  << 8) | (c)->blue; \                  (xc)->blue  = ((c)->blue  << 8) | (c)->blue; \
894                  (xc)->flags = DoRed | DoGreen | DoBlue;                  (xc)->flags = DoRed | DoGreen | DoBlue;
895    
896    
897  HCOLOURMAP  HCOLOURMAP
898  ui_create_colourmap(COLOURMAP *colours)  ui_create_colourmap(COLOURMAP * colours)
899  {  {
900          COLOURENTRY *entry;          COLOURENTRY *entry;
901          int i, ncolours = colours->ncolours;          int i, ncolours = colours->ncolours;
902            if (!owncolmap)
903            {
904                    uint32 *map = xmalloc(sizeof(*colmap) * ncolours);
905                    XColor xentry;
906                    XColor xc_cache[256];
907                    uint32 colour;
908                    int colLookup = 256;
909                    for (i = 0; i < ncolours; i++)
910                    {
911                            entry = &colours->colours[i];
912                            MAKE_XCOLOR(&xentry, entry);
913    
914                            if (XAllocColor(display, xcolmap, &xentry) == 0)
915                            {
916                                    /* Allocation failed, find closest match. */
917                                    int j = 256;
918                                    int nMinDist = 3 * 256 * 256;
919                                    long nDist = nMinDist;
920    
921          if (owncolmap)                                  /* only get the colors once */
922                                    while (colLookup--)
923                                    {
924                                            xc_cache[colLookup].pixel = colLookup;
925                                            xc_cache[colLookup].red = xc_cache[colLookup].green =
926                                                    xc_cache[colLookup].blue = 0;
927                                            xc_cache[colLookup].flags = 0;
928                                            XQueryColor(display,
929                                                        DefaultColormap(display,
930                                                                        DefaultScreen(display)),
931                                                        &xc_cache[colLookup]);
932                                    }
933                                    colLookup = 0;
934    
935                                    /* approximate the pixel */
936                                    while (j--)
937                                    {
938                                            if (xc_cache[j].flags)
939                                            {
940                                                    nDist = ((long) (xc_cache[j].red >> 8) -
941                                                             (long) (xentry.red >> 8)) *
942                                                            ((long) (xc_cache[j].red >> 8) -
943                                                             (long) (xentry.red >> 8)) +
944                                                            ((long) (xc_cache[j].green >> 8) -
945                                                             (long) (xentry.green >> 8)) *
946                                                            ((long) (xc_cache[j].green >> 8) -
947                                                             (long) (xentry.green >> 8)) +
948                                                            ((long) (xc_cache[j].blue >> 8) -
949                                                             (long) (xentry.blue >> 8)) *
950                                                            ((long) (xc_cache[j].blue >> 8) -
951                                                             (long) (xentry.blue >> 8));
952                                            }
953                                            if (nDist < nMinDist)
954                                            {
955                                                    nMinDist = nDist;
956                                                    xentry.pixel = j;
957                                            }
958                                    }
959                            }
960                            colour = xentry.pixel;
961    
962                            /* update our cache */
963                            if (xentry.pixel < 256)
964                            {
965                                    xc_cache[xentry.pixel].red = xentry.red;
966                                    xc_cache[xentry.pixel].green = xentry.green;
967                                    xc_cache[xentry.pixel].blue = xentry.blue;
968    
969                            }
970    
971    
972                            /* byte swap here to make translate_image faster */
973                            map[i] = translate_colour(colour);
974                    }
975                    return map;
976            }
977            else
978          {          {
979                  XColor *xcolours, *xentry;                  XColor *xcolours, *xentry;
980                  Colormap map;                  Colormap map;
# Line 596  ui_create_colourmap(COLOURMAP *colours) Line 992  ui_create_colourmap(COLOURMAP *colours)
992                  XStoreColors(display, map, xcolours, ncolours);                  XStoreColors(display, map, xcolours, ncolours);
993    
994                  xfree(xcolours);                  xfree(xcolours);
995                  return (HCOLOURMAP)map;                  return (HCOLOURMAP) map;
         }  
         else  
         {  
                 uint32 *map = xmalloc(sizeof(*colmap) * ncolours);  
                 XColor xentry;  
   
                 for (i = 0; i < ncolours; i++)  
                 {  
                         entry = &colours->colours[i];  
                         MAKE_XCOLOR(&xentry, entry);  
   
                         if (XAllocColor(display, xcolmap, &xentry) != 0)  
                                 map[i] = xentry.pixel;  
                         else  
                                 map[i] = white;  
                 }  
   
                 return map;  
996          }          }
997  }  }
998    
999  void  void
1000  ui_destroy_colourmap(HCOLOURMAP map)  ui_destroy_colourmap(HCOLOURMAP map)
1001  {  {
1002          if (owncolmap)          if (!owncolmap)
                 XFreeColormap(display, (Colormap)map);  
         else  
1003                  xfree(map);                  xfree(map);
1004            else
1005                    XFreeColormap(display, (Colormap) map);
1006  }  }
1007    
1008  void  void
1009  ui_set_colourmap(HCOLOURMAP map)  ui_set_colourmap(HCOLOURMAP map)
1010  {  {
1011          if (owncolmap)          if (!owncolmap)
                 XSetWindowColormap(display, wnd, (Colormap)map);  
         else  
1012                  colmap = map;                  colmap = map;
1013            else
1014                    XSetWindowColormap(display, wnd, (Colormap) map);
1015  }  }
1016    
1017  void  void
# Line 649  ui_set_clip(int x, int y, int cx, int cy Line 1027  ui_set_clip(int x, int y, int cx, int cy
1027  }  }
1028    
1029  void  void
1030  ui_reset_clip()  ui_reset_clip(void)
1031  {  {
1032          XRectangle rect;          XRectangle rect;
1033    
# Line 661  ui_reset_clip() Line 1039  ui_reset_clip()
1039  }  }
1040    
1041  void  void
1042  ui_bell()  ui_bell(void)
1043  {  {
1044          XBell(display, 0);          XBell(display, 0);
1045  }  }
# Line 671  ui_destblt(uint8 opcode, Line 1049  ui_destblt(uint8 opcode,
1049             /* dest */ int x, int y, int cx, int cy)             /* dest */ int x, int y, int cx, int cy)
1050  {  {
1051          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1052          XFillRectangle(display, wnd, gc, x, y, cx, cy);          FILL_RECTANGLE(x, y, cx, cy);
1053          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1054  }  }
1055    
1056  void  void
1057  ui_patblt(uint8 opcode,  ui_patblt(uint8 opcode,
1058            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
1059            /* brush */ BRUSH *brush, int bgcolour, int fgcolour)            /* brush */ BRUSH * brush, int bgcolour, int fgcolour)
1060  {  {
1061          Pixmap fill;          Pixmap fill;
1062            uint8 i, ipattern[8];
1063    
1064          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1065    
# Line 688  ui_patblt(uint8 opcode, Line 1067  ui_patblt(uint8 opcode,
1067          {          {
1068                  case 0: /* Solid */                  case 0: /* Solid */
1069                          SET_FOREGROUND(fgcolour);                          SET_FOREGROUND(fgcolour);
1070                          XFillRectangle(display, wnd, gc, x, y, cx, cy);                          FILL_RECTANGLE(x, y, cx, cy);
1071                          break;                          break;
1072    
1073                  case 3: /* Pattern */                  case 3: /* Pattern */
1074                          fill = (Pixmap)ui_create_glyph(8, 8, brush->pattern);                          for (i = 0; i != 8; i++)
1075                                    ipattern[7 - i] = brush->pattern[i];
1076                            fill = (Pixmap) ui_create_glyph(8, 8, ipattern);
1077    
1078                          SET_FOREGROUND(bgcolour);                          SET_FOREGROUND(bgcolour);
1079                          SET_BACKGROUND(fgcolour);                          SET_BACKGROUND(fgcolour);
# Line 700  ui_patblt(uint8 opcode, Line 1081  ui_patblt(uint8 opcode,
1081                          XSetStipple(display, gc, fill);                          XSetStipple(display, gc, fill);
1082                          XSetTSOrigin(display, gc, brush->xorigin, brush->yorigin);                          XSetTSOrigin(display, gc, brush->xorigin, brush->yorigin);
1083    
1084                          XFillRectangle(display, wnd, gc, x, y, cx, cy);                          FILL_RECTANGLE(x, y, cx, cy);
1085    
1086                          XSetFillStyle(display, gc, FillSolid);                          XSetFillStyle(display, gc, FillSolid);
1087                          ui_destroy_glyph((HGLYPH)fill);                          XSetTSOrigin(display, gc, 0, 0);
1088                            ui_destroy_glyph((HGLYPH) fill);
1089                          break;                          break;
1090    
1091                  default:                  default:
1092                          NOTIMP("brush %d\n", brush->style);                          unimpl("brush %d\n", brush->style);
1093          }          }
1094    
1095          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
# Line 720  ui_screenblt(uint8 opcode, Line 1102  ui_screenblt(uint8 opcode,
1102  {  {
1103          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1104          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, wnd, wnd, gc, srcx, srcy, cx, cy, x, y);
1105            if (ownbackstore)
1106                    XCopyArea(display, backstore, backstore, gc, srcx, srcy, cx, cy, x, y);
1107          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1108  }  }
1109    
# Line 729  ui_memblt(uint8 opcode, Line 1113  ui_memblt(uint8 opcode,
1113            /* src */ HBITMAP src, int srcx, int srcy)            /* src */ HBITMAP src, int srcx, int srcy)
1114  {  {
1115          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1116          XCopyArea(display, (Pixmap)src, wnd, gc, srcx, srcy, cx, cy, x, y);          XCopyArea(display, (Pixmap) src, wnd, gc, srcx, srcy, cx, cy, x, y);
1117            if (ownbackstore)
1118                    XCopyArea(display, (Pixmap) src, backstore, gc, srcx, srcy, cx, cy, x, y);
1119          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1120  }  }
1121    
# Line 737  void Line 1123  void
1123  ui_triblt(uint8 opcode,  ui_triblt(uint8 opcode,
1124            /* dest */ int x, int y, int cx, int cy,            /* dest */ int x, int y, int cx, int cy,
1125            /* src */ HBITMAP src, int srcx, int srcy,            /* src */ HBITMAP src, int srcx, int srcy,
1126            /* brush */ BRUSH *brush, int bgcolour, int fgcolour)            /* brush */ BRUSH * brush, int bgcolour, int fgcolour)
1127  {  {
1128          /* This is potentially difficult to do in general. Until someone          /* This is potentially difficult to do in general. Until someone
1129             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 746  ui_triblt(uint8 opcode, Line 1132  ui_triblt(uint8 opcode,
1132          {          {
1133                  case 0x69:      /* PDSxxn */                  case 0x69:      /* PDSxxn */
1134                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_XOR, x, y, cx, cy, src, srcx, srcy);
1135                          ui_patblt(ROP2_NXOR, x, y, cx, cy,                          ui_patblt(ROP2_NXOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1136                          break;                          break;
1137    
1138                  case 0xb8:      /* PSDPxax */                  case 0xb8:      /* PSDPxax */
1139                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1140                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_AND, x, y, cx, cy, src, srcx, srcy);
1141                          ui_patblt(ROP2_XOR, x, y, cx, cy,                          ui_patblt(ROP2_XOR, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   brush, bgcolour, fgcolour);  
1142                          break;                          break;
1143    
1144                  case 0xc0:      /* PSa */                  case 0xc0:      /* PSa */
1145                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);
1146                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour,                          ui_patblt(ROP2_AND, x, y, cx, cy, brush, bgcolour, fgcolour);
                                   fgcolour);  
1147                          break;                          break;
1148    
1149                  default:                  default:
1150                          NOTIMP("triblt 0x%x\n", opcode);                          unimpl("triblt 0x%x\n", opcode);
1151                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);                          ui_memblt(ROP2_COPY, x, y, cx, cy, src, srcx, srcy);
1152          }          }
1153  }  }
# Line 773  ui_triblt(uint8 opcode, Line 1155  ui_triblt(uint8 opcode,
1155  void  void
1156  ui_line(uint8 opcode,  ui_line(uint8 opcode,
1157          /* dest */ int startx, int starty, int endx, int endy,          /* dest */ int startx, int starty, int endx, int endy,
1158          /* pen */ PEN *pen)          /* pen */ PEN * pen)
1159  {  {
1160          SET_FUNCTION(opcode);          SET_FUNCTION(opcode);
1161          SET_FOREGROUND(pen->colour);          SET_FOREGROUND(pen->colour);
1162          XDrawLine(display, wnd, gc, startx, starty, endx, endy);          XDrawLine(display, wnd, gc, startx, starty, endx, endy);
1163            if (ownbackstore)
1164                    XDrawLine(display, backstore, gc, startx, starty, endx, endy);
1165          RESET_FUNCTION(opcode);          RESET_FUNCTION(opcode);
1166  }  }
1167    
# Line 787  ui_rect( Line 1171  ui_rect(
1171                 /* brush */ int colour)                 /* brush */ int colour)
1172  {  {
1173          SET_FOREGROUND(colour);          SET_FOREGROUND(colour);
1174          XFillRectangle(display, wnd, gc, x, y, cx, cy);          FILL_RECTANGLE(x, y, cx, cy);
1175  }  }
1176    
1177    /* warning, this function only draws on wnd or backstore, not both */
1178  void  void
1179  ui_draw_glyph(int mixmode,  ui_draw_glyph(int mixmode,
1180                /* dest */ int x, int y, int cx, int cy,                /* dest */ int x, int y, int cx, int cy,
1181                /* src */ HGLYPH glyph, int srcx, int srcy, int bgcolour,                /* src */ HGLYPH glyph, int srcx, int srcy,
1182                int fgcolour)                int bgcolour, int fgcolour)
1183  {  {
1184          SET_FOREGROUND(fgcolour);          SET_FOREGROUND(fgcolour);
1185          SET_BACKGROUND(bgcolour);          SET_BACKGROUND(bgcolour);
1186    
1187          XSetFillStyle(display, gc, (mixmode == MIX_TRANSPARENT)          XSetFillStyle(display, gc,
1188                        ? FillStippled : FillOpaqueStippled);                        (mixmode == MIX_TRANSPARENT) ? FillStippled : FillOpaqueStippled);
1189          XSetStipple(display, gc, (Pixmap)glyph);          XSetStipple(display, gc, (Pixmap) glyph);
1190          XSetTSOrigin(display, gc, x, y);          XSetTSOrigin(display, gc, x, y);
1191    
1192          XFillRectangle(display, wnd, gc, x, y, cx, cy);          FILL_RECTANGLE_FAST(x, y, cx, cy);
1193    
1194          XSetFillStyle(display, gc, FillSolid);          XSetFillStyle(display, gc, FillSolid);
1195  }  }
1196    
1197    #define DO_GLYPH(ttext,idx) \
1198    {\
1199      glyph = cache_get_font (font, ttext[idx]);\
1200      if (!(flags & TEXT2_IMPLICIT_X))\
1201        {\
1202          xyoffset = ttext[++idx];\
1203          if ((xyoffset & 0x80))\
1204            {\
1205              if (flags & TEXT2_VERTICAL) \
1206                y += ttext[idx+1] | (ttext[idx+2] << 8);\
1207              else\
1208                x += ttext[idx+1] | (ttext[idx+2] << 8);\
1209              idx += 2;\
1210            }\
1211          else\
1212            {\
1213              if (flags & TEXT2_VERTICAL) \
1214                y += xyoffset;\
1215              else\
1216                x += xyoffset;\
1217            }\
1218        }\
1219      if (glyph != NULL)\
1220        {\
1221          ui_draw_glyph (mixmode, x + (short) glyph->offset,\
1222                         y + (short) glyph->baseline,\
1223                         glyph->width, glyph->height,\
1224                         glyph->pixmap, 0, 0, bgcolour, fgcolour);\
1225          if (flags & TEXT2_IMPLICIT_X)\
1226            x += glyph->width;\
1227        }\
1228    }
1229    
1230  void  void
1231  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,
1232               int clipx, int clipy, int clipcx, int clipcy,               int clipx, int clipy, int clipcx, int clipcy,
1233               int boxx, int boxy, int boxcx, int boxcy,               int boxx, int boxy, int boxcx, int boxcy, int bgcolour,
1234               int bgcolour, int fgcolour, uint8 *text, uint8 length)               int fgcolour, uint8 * text, uint8 length)
1235  {  {
1236          FONTGLYPH *glyph;          FONTGLYPH *glyph;
1237          int i, offset;          int i, j, xyoffset;
1238            DATABLOB *entry;
1239    
1240          SET_FOREGROUND(bgcolour);          SET_FOREGROUND(bgcolour);
1241    
1242          if (boxcx > 1)          if (boxcx > 1)
1243                  XFillRectangle(display, wnd, gc, boxx, boxy, boxcx, boxcy);          {
1244                    FILL_RECTANGLE_FAST(boxx, boxy, boxcx, boxcy);
1245            }
1246          else if (mixmode == MIX_OPAQUE)          else if (mixmode == MIX_OPAQUE)
1247                  XFillRectangle(display, wnd, gc, clipx, clipy, clipcx, clipcy);          {
1248                    FILL_RECTANGLE_FAST(clipx, clipy, clipcx, clipcy);
1249            }
1250    
1251          /* Paint text, character by character */          /* Paint text, character by character */
1252          for (i = 0; i < length; i++)          for (i = 0; i < length;)
1253          {          {
1254                  glyph = cache_get_font(font, text[i]);                  switch (text[i])
   
                 if (!(flags & TEXT2_IMPLICIT_X))  
1255                  {                  {
1256                          offset = text[++i];                          case 0xff:
1257                          if (offset & 0x80)                                  if (i + 2 < length)
1258                                  offset = ((offset & 0x7f) << 8) | text[++i];                                          cache_put_text(text[i + 1], text, text[i + 2]);
1259                                    else
1260                          if (flags & TEXT2_VERTICAL)                                  {
1261                                  y += offset;                                          error("this shouldn't be happening\n");
1262                          else                                          exit(1);
1263                                  x += offset;                                  }
1264                  }                                  /* this will move pointer from start to first character after FF command */
1265                                    length -= i + 3;
1266                                    text = &(text[i + 3]);
1267                                    i = 0;
1268                                    break;
1269    
1270                  if (glyph != NULL)                          case 0xfe:
1271                  {                                  entry = cache_get_text(text[i + 1]);
1272                          ui_draw_glyph(mixmode, x + (short) glyph->offset,                                  if (entry != NULL)
1273                                        y + (short) glyph->baseline,                                  {
1274                                        glyph->width, glyph->height,                                          if ((((uint8 *) (entry->data))[1] ==
1275                                        glyph->pixmap, 0, 0,                                               0) && (!(flags & TEXT2_IMPLICIT_X)))
1276                                        bgcolour, fgcolour);                                          {
1277                                                    if (flags & TEXT2_VERTICAL)
1278                                                            y += text[i + 2];
1279                                                    else
1280                                                            x += text[i + 2];
1281                                            }
1282                                            if (i + 2 < length)
1283                                                    i += 3;
1284                                            else
1285                                                    i += 2;
1286                                            length -= i;
1287                                            /* this will move pointer from start to first character after FE command */
1288                                            text = &(text[i]);
1289                                            i = 0;
1290                                            for (j = 0; j < entry->size; j++)
1291                                                    DO_GLYPH(((uint8 *) (entry->data)), j);
1292                                    }
1293                                    break;
1294    
1295                          if (flags & TEXT2_IMPLICIT_X)                          default:
1296                                  x += glyph->width;                                  DO_GLYPH(text, i);
1297                                    i++;
1298                                    break;
1299                  }                  }
1300          }          }
1301            if (ownbackstore)
1302            {
1303                    if (boxcx > 1)
1304                            XCopyArea(display, backstore, wnd, gc, boxx,
1305                                      boxy, boxcx, boxcy, boxx, boxy);
1306                    else
1307                            XCopyArea(display, backstore, wnd, gc, clipx,
1308                                      clipy, clipcx, clipcy, clipx, clipy);
1309            }
1310  }  }
1311    
1312  void  void
# Line 862  ui_desktop_save(uint32 offset, int x, in Line 1315  ui_desktop_save(uint32 offset, int x, in
1315          Pixmap pix;          Pixmap pix;
1316          XImage *image;          XImage *image;
1317    
1318          pix = XCreatePixmap(display, wnd, cx, cy, depth);          if (ownbackstore)
1319          XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);          {
1320                    image = XGetImage(display, backstore, x, y, cx, cy, AllPlanes, ZPixmap);
1321          image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes, ZPixmap);          }
1322            else
1323            {
1324                    pix = XCreatePixmap(display, wnd, cx, cy, depth);
1325                    XCopyArea(display, wnd, pix, gc, x, y, cx, cy, 0, 0);
1326                    image = XGetImage(display, pix, 0, 0, cx, cy, AllPlanes, ZPixmap);
1327                    XFreePixmap(display, pix);
1328            }
1329    
1330          offset *= bpp/8;          offset *= bpp / 8;
1331          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, image->data);  
1332    
1333          XDestroyImage(image);          XDestroyImage(image);
         XFreePixmap(display, pix);  
1334  }  }
1335    
1336  void  void
# Line 881  ui_desktop_restore(uint32 offset, int x, Line 1339  ui_desktop_restore(uint32 offset, int x,
1339          XImage *image;          XImage *image;
1340          uint8 *data;          uint8 *data;
1341    
1342          offset *= bpp/8;          offset *= bpp / 8;
1343          data = cache_get_desktop(offset, cx, cy, bpp/8);          data = cache_get_desktop(offset, cx, cy, bpp / 8);
1344          if (data == NULL)          if (data == NULL)
1345                  return;                  return;
1346    
1347          image = XCreateImage(display, visual, depth, ZPixmap,          image = XCreateImage(display, visual, depth, ZPixmap, 0,
1348                               0, data, cx, cy, BitmapPad(display),                               (char *) data, cx, cy, BitmapPad(display), cx * bpp / 8);
1349                               cx * bpp/8);  
1350            if (ownbackstore)
1351            {
1352                    XPutImage(display, backstore, gc, image, 0, 0, x, y, cx, cy);
1353                    XCopyArea(display, backstore, wnd, gc, x, y, cx, cy, x, y);
1354            }
1355            else
1356            {
1357                    XPutImage(display, wnd, gc, image, 0, 0, x, y, cx, cy);
1358            }
1359    
         XPutImage(display, wnd, gc, image, 0, 0, x, y, cx, cy);  
1360          XFree(image);          XFree(image);
1361  }  }
   

Legend:
Removed from v.29  
changed lines
  Added in v.281

  ViewVC Help
Powered by ViewVC 1.1.26