32 |
extern BOOL hide_decorations; |
extern BOOL hide_decorations; |
33 |
extern char title[]; |
extern char title[]; |
34 |
extern int server_bpp; |
extern int server_bpp; |
35 |
|
extern int win_button_size; |
36 |
BOOL enable_compose = False; |
BOOL enable_compose = False; |
37 |
BOOL focused; |
BOOL focused; |
38 |
BOOL mouse_in_wnd; |
BOOL mouse_in_wnd; |
59 |
static BOOL ownbackstore; |
static BOOL ownbackstore; |
60 |
static Pixmap backstore; |
static Pixmap backstore; |
61 |
|
|
62 |
|
/* Moving in single app mode */ |
63 |
|
static BOOL moving_wnd; |
64 |
|
static int move_x_offset = 0; |
65 |
|
static int move_y_offset = 0; |
66 |
|
|
67 |
/* MWM decorations */ |
/* MWM decorations */ |
68 |
#define MWM_HINTS_DECORATIONS (1L << 1) |
#define MWM_HINTS_DECORATIONS (1L << 1) |
69 |
#define PROP_MOTIF_WM_HINTS_ELEMENTS 5 |
#define PROP_MOTIF_WM_HINTS_ELEMENTS 5 |
402 |
uint32 pixel = 0; |
uint32 pixel = 0; |
403 |
while (out < end) |
while (out < end) |
404 |
{ |
{ |
405 |
memcpy(&pixel, data, 3); |
pixel = *(data++); |
406 |
data += 3; |
pixel |= *(data++) << 8; |
407 |
|
pixel |= *(data++) << 16; |
408 |
*(out++) = pixel; |
*(out++) = pixel; |
409 |
} |
} |
410 |
} |
} |
861 |
if (button == 0) |
if (button == 0) |
862 |
break; |
break; |
863 |
|
|
864 |
|
/* If win_button_size is nonzero, enable single app mode */ |
865 |
|
if (xevent.xbutton.y < win_button_size) |
866 |
|
{ |
867 |
|
/* Stop moving window when button is released, regardless of cursor position */ |
868 |
|
if (moving_wnd && (xevent.type == ButtonRelease)) |
869 |
|
moving_wnd = False; |
870 |
|
|
871 |
|
/* Check from right to left: */ |
872 |
|
|
873 |
|
if (xevent.xbutton.x >= width - win_button_size) |
874 |
|
{ |
875 |
|
/* The close button, continue */ |
876 |
|
; |
877 |
|
} |
878 |
|
else if (xevent.xbutton.x >= width - win_button_size * 2) |
879 |
|
{ |
880 |
|
/* The maximize/restore button. Do not send to |
881 |
|
server. It might be a good idea to change the |
882 |
|
cursor or give some other visible indication |
883 |
|
that rdesktop inhibited this click */ |
884 |
|
break; |
885 |
|
} |
886 |
|
else if (xevent.xbutton.x >= width - win_button_size * 3) |
887 |
|
{ |
888 |
|
/* The minimize button. Iconify window. */ |
889 |
|
XIconifyWindow(display, wnd, |
890 |
|
DefaultScreen(display)); |
891 |
|
break; |
892 |
|
} |
893 |
|
else if (xevent.xbutton.x <= win_button_size) |
894 |
|
{ |
895 |
|
/* The system menu. Ignore. */ |
896 |
|
break; |
897 |
|
} |
898 |
|
else |
899 |
|
{ |
900 |
|
/* The title bar. */ |
901 |
|
if ((xevent.type == ButtonPress) && !fullscreen |
902 |
|
&& hide_decorations) |
903 |
|
{ |
904 |
|
moving_wnd = True; |
905 |
|
move_x_offset = xevent.xbutton.x; |
906 |
|
move_y_offset = xevent.xbutton.y; |
907 |
|
} |
908 |
|
break; |
909 |
|
|
910 |
|
} |
911 |
|
} |
912 |
|
|
913 |
rdp_send_input(time(NULL), RDP_INPUT_MOUSE, |
rdp_send_input(time(NULL), RDP_INPUT_MOUSE, |
914 |
flags | button, xevent.xbutton.x, xevent.xbutton.y); |
flags | button, xevent.xbutton.x, xevent.xbutton.y); |
915 |
break; |
break; |
916 |
|
|
917 |
case MotionNotify: |
case MotionNotify: |
918 |
|
if (moving_wnd) |
919 |
|
{ |
920 |
|
XMoveWindow(display, wnd, |
921 |
|
xevent.xmotion.x_root - move_x_offset, |
922 |
|
xevent.xmotion.y_root - move_y_offset); |
923 |
|
break; |
924 |
|
} |
925 |
|
|
926 |
if (fullscreen && !focused) |
if (fullscreen && !focused) |
927 |
XSetInputFocus(display, wnd, RevertToPointerRoot, |
XSetInputFocus(display, wnd, RevertToPointerRoot, |
928 |
CurrentTime); |
CurrentTime); |
1371 |
RESET_FUNCTION(opcode); |
RESET_FUNCTION(opcode); |
1372 |
} |
} |
1373 |
|
|
1374 |
|
static uint8 hatch_patterns[] = { |
1375 |
|
0x00, 0x00, 0x00, 0xff, 0x00, 0x00, 0x00, 0x00, /* 0 - bsHorizontal */ |
1376 |
|
0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08, /* 1 - bsVertical */ |
1377 |
|
0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01, /* 2 - bsFDiagonal */ |
1378 |
|
0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, /* 3 - bsBDiagonal */ |
1379 |
|
0x08, 0x08, 0x08, 0xff, 0x08, 0x08, 0x08, 0x08, /* 4 - bsCross */ |
1380 |
|
0x81, 0x42, 0x24, 0x18, 0x18, 0x24, 0x42, 0x81 /* 5 - bsDiagCross */ |
1381 |
|
}; |
1382 |
|
|
1383 |
void |
void |
1384 |
ui_patblt(uint8 opcode, |
ui_patblt(uint8 opcode, |
1385 |
/* dest */ int x, int y, int cx, int cy, |
/* dest */ int x, int y, int cx, int cy, |
1397 |
FILL_RECTANGLE(x, y, cx, cy); |
FILL_RECTANGLE(x, y, cx, cy); |
1398 |
break; |
break; |
1399 |
|
|
1400 |
|
case 2: /* Hatch */ |
1401 |
|
fill = (Pixmap) ui_create_glyph(8, 8, hatch_patterns + brush->pattern[0] * 8); |
1402 |
|
SET_FOREGROUND(bgcolour); |
1403 |
|
SET_BACKGROUND(fgcolour); |
1404 |
|
XSetFillStyle(display, gc, FillOpaqueStippled); |
1405 |
|
XSetStipple(display, gc, fill); |
1406 |
|
XSetTSOrigin(display, gc, brush->xorigin, brush->yorigin); |
1407 |
|
FILL_RECTANGLE(x, y, cx, cy); |
1408 |
|
XSetFillStyle(display, gc, FillSolid); |
1409 |
|
XSetTSOrigin(display, gc, 0, 0); |
1410 |
|
ui_destroy_glyph((HGLYPH) fill); |
1411 |
|
break; |
1412 |
|
|
1413 |
case 3: /* Pattern */ |
case 3: /* Pattern */ |
1414 |
for (i = 0; i != 8; i++) |
for (i = 0; i != 8; i++) |
1415 |
ipattern[7 - i] = brush->pattern[i]; |
ipattern[7 - i] = brush->pattern[i]; |