OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "ui/aura/root_window_host_linux.h" | 5 #include "ui/aura/root_window_host_linux.h" |
6 | 6 |
7 #include <X11/Xatom.h> | 7 #include <X11/Xatom.h> |
8 #include <X11/cursorfont.h> | 8 #include <X11/cursorfont.h> |
9 #include <X11/extensions/XInput2.h> | 9 #include <X11/extensions/XInput2.h> |
10 #include <X11/extensions/Xfixes.h> | 10 #include <X11/extensions/Xfixes.h> |
11 #include <X11/extensions/Xrandr.h> | 11 #include <X11/extensions/Xrandr.h> |
12 #include <algorithm> | 12 #include <algorithm> |
13 | 13 |
14 #include "base/message_pump_x.h" | 14 #include "base/message_pump_x.h" |
15 #include "base/stl_util.h" | 15 #include "base/stl_util.h" |
16 #include "base/stringprintf.h" | 16 #include "base/stringprintf.h" |
17 #include "ui/aura/client/user_gesture_client.h" | 17 #include "ui/aura/client/user_gesture_client.h" |
18 #include "ui/aura/dispatcher_linux.h" | 18 #include "ui/aura/dispatcher_linux.h" |
19 #include "ui/aura/env.h" | 19 #include "ui/aura/env.h" |
20 #include "ui/aura/event.h" | 20 #include "ui/aura/event.h" |
21 #include "ui/aura/root_window.h" | 21 #include "ui/aura/root_window.h" |
22 #include "ui/base/cursor/cursor.h" | 22 #include "ui/base/cursor/cursor.h" |
| 23 #include "ui/base/hit_test.h" |
23 #include "ui/base/keycodes/keyboard_codes.h" | 24 #include "ui/base/keycodes/keyboard_codes.h" |
24 #include "ui/base/touch/touch_factory.h" | 25 #include "ui/base/touch/touch_factory.h" |
25 #include "ui/base/view_prop.h" | 26 #include "ui/base/view_prop.h" |
26 #include "ui/base/x/x11_util.h" | 27 #include "ui/base/x/x11_util.h" |
27 #include "ui/compositor/layer.h" | 28 #include "ui/compositor/layer.h" |
28 | 29 |
29 using std::max; | 30 using std::max; |
30 using std::min; | 31 using std::min; |
31 | 32 |
32 namespace aura { | 33 namespace aura { |
33 | 34 |
34 namespace { | 35 namespace { |
35 | 36 |
36 // Standard Linux mouse buttons for going back and forward. | 37 // Standard Linux mouse buttons for going back and forward. |
37 const int kBackMouseButton = 8; | 38 const int kBackMouseButton = 8; |
38 const int kForwardMouseButton = 9; | 39 const int kForwardMouseButton = 9; |
39 | 40 |
40 const char kRootWindowHostLinuxKey[] = "__AURA_ROOT_WINDOW_HOST_LINUX__"; | 41 const char kRootWindowHostLinuxKey[] = "__AURA_ROOT_WINDOW_HOST_LINUX__"; |
41 | 42 |
| 43 // These constants are defined in the Extended Window Manager Hints |
| 44 // standard...and aren't in any header that I can find. |
| 45 const int k_NET_WM_MOVERESIZE_SIZE_TOPLEFT = 0; |
| 46 const int k_NET_WM_MOVERESIZE_SIZE_TOP = 1; |
| 47 const int k_NET_WM_MOVERESIZE_SIZE_TOPRIGHT = 2; |
| 48 const int k_NET_WM_MOVERESIZE_SIZE_RIGHT = 3; |
| 49 const int k_NET_WM_MOVERESIZE_SIZE_BOTTOMRIGHT = 4; |
| 50 const int k_NET_WM_MOVERESIZE_SIZE_BOTTOM = 5; |
| 51 const int k_NET_WM_MOVERESIZE_SIZE_BOTTOMLEFT = 6; |
| 52 const int k_NET_WM_MOVERESIZE_SIZE_LEFT = 7; |
| 53 const int k_NET_WM_MOVERESIZE_MOVE = 8; |
| 54 |
| 55 // This data structure represents additional hints that we send to the window |
| 56 // manager and has a direct lineage back to Motif, which defined this de facto |
| 57 // standard. This struct doesn't seem 64-bit safe though, but it's what GDK |
| 58 // does. |
| 59 typedef struct { |
| 60 unsigned long flags; |
| 61 unsigned long functions; |
| 62 unsigned long decorations; |
| 63 long input_mode; |
| 64 unsigned long status; |
| 65 } MotifWmHints; |
| 66 |
| 67 // The bitflag in |flags| in MotifWmHints that signals that the reader should |
| 68 // pay attention to the value in |decorations|. |
| 69 const unsigned long kHintsDecorations = (1L << 1); |
| 70 |
42 // The events reported for slave devices can have incorrect information for some | 71 // The events reported for slave devices can have incorrect information for some |
43 // fields. This utility function is used to check for such inconsistencies. | 72 // fields. This utility function is used to check for such inconsistencies. |
44 void CheckXEventForConsistency(XEvent* xevent) { | 73 void CheckXEventForConsistency(XEvent* xevent) { |
45 #if defined(USE_XI2_MT) | 74 #if defined(USE_XI2_MT) |
46 static bool expect_master_event = false; | 75 static bool expect_master_event = false; |
47 static XIDeviceEvent slave_event; | 76 static XIDeviceEvent slave_event; |
48 static gfx::Point slave_location; | 77 static gfx::Point slave_location; |
49 static int slave_button; | 78 static int slave_button; |
50 | 79 |
51 // Note: If an event comes from a slave pointer device, then it will be | 80 // Note: If an event comes from a slave pointer device, then it will be |
(...skipping 226 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
278 return true; | 307 return true; |
279 default: | 308 default: |
280 return false; | 309 return false; |
281 } | 310 } |
282 } | 311 } |
283 | 312 |
284 // A list of atoms that we'll intern on host creation to save roundtrips to the | 313 // A list of atoms that we'll intern on host creation to save roundtrips to the |
285 // X11 server. Must be kept in sync with RootWindowHostLinux::AtomList | 314 // X11 server. Must be kept in sync with RootWindowHostLinux::AtomList |
286 const char* kAtomList[] = { | 315 const char* kAtomList[] = { |
287 "WM_DELETE_WINDOW", | 316 "WM_DELETE_WINDOW", |
| 317 "_NET_WM_MOVERESIZE", |
288 "_NET_WM_PING", | 318 "_NET_WM_PING", |
289 "_NET_WM_PID", | 319 "_NET_WM_PID", |
290 "WM_S0" | 320 "WM_S0", |
| 321 "_MOTIF_WM_HINTS" |
291 }; | 322 }; |
292 | 323 |
293 } // namespace | 324 } // namespace |
294 | 325 |
295 RootWindowHostLinux::RootWindowHostLinux(const gfx::Rect& bounds) | 326 RootWindowHostLinux::RootWindowHostLinux(const gfx::Rect& bounds) |
296 : root_window_(NULL), | 327 : root_window_(NULL), |
297 xdisplay_(base::MessagePumpX::GetDefaultXDisplay()), | 328 xdisplay_(base::MessagePumpX::GetDefaultXDisplay()), |
298 xwindow_(0), | 329 xwindow_(0), |
299 x_root_window_(DefaultRootWindow(xdisplay_)), | 330 x_root_window_(DefaultRootWindow(xdisplay_)), |
300 current_cursor_(ui::kCursorNull), | 331 current_cursor_(ui::kCursorNull), |
(...skipping 198 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
499 ui::IsTouchpadEvent(xev) && ui::IsNaturalScrollEnabled(); | 530 ui::IsTouchpadEvent(xev) && ui::IsNaturalScrollEnabled(); |
500 gesture_client->OnUserGesture( | 531 gesture_client->OnUserGesture( |
501 (button == kBackMouseButton && !reverse_direction) || | 532 (button == kBackMouseButton && !reverse_direction) || |
502 (button == kForwardMouseButton && reverse_direction) ? | 533 (button == kForwardMouseButton && reverse_direction) ? |
503 client::UserGestureClient::GESTURE_BACK : | 534 client::UserGestureClient::GESTURE_BACK : |
504 client::UserGestureClient::GESTURE_FORWARD); | 535 client::UserGestureClient::GESTURE_FORWARD); |
505 } | 536 } |
506 break; | 537 break; |
507 } | 538 } |
508 } | 539 } |
| 540 |
509 MouseEvent mouseev(xev); | 541 MouseEvent mouseev(xev); |
510 root_window_->DispatchMouseEvent(&mouseev); | 542 root_window_->DispatchMouseEvent(&mouseev); |
511 break; | 543 break; |
512 } | 544 } |
513 case ui::ET_SCROLL_FLING_START: | 545 case ui::ET_SCROLL_FLING_START: |
514 case ui::ET_SCROLL_FLING_CANCEL: | 546 case ui::ET_SCROLL_FLING_CANCEL: |
515 case ui::ET_SCROLL: { | 547 case ui::ET_SCROLL: { |
516 ScrollEvent scrollev(xev); | 548 ScrollEvent scrollev(xev); |
517 root_window_->DispatchScrollEvent(&scrollev); | 549 root_window_->DispatchScrollEvent(&scrollev); |
518 break; | 550 break; |
(...skipping 260 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
779 root_window_->ConvertPointToNativeScreen(&point); | 811 root_window_->ConvertPointToNativeScreen(&point); |
780 xevent.xmotion.x_root = point.x(); | 812 xevent.xmotion.x_root = point.x(); |
781 xevent.xmotion.y_root = point.y(); | 813 xevent.xmotion.y_root = point.y(); |
782 } | 814 } |
783 default: | 815 default: |
784 break; | 816 break; |
785 } | 817 } |
786 XSendEvent(xdisplay_, xwindow_, False, 0, &xevent); | 818 XSendEvent(xdisplay_, xwindow_, False, 0, &xevent); |
787 } | 819 } |
788 | 820 |
| 821 void RootWindowHostLinux::SetUseHostWindowBorders(bool use_os_border) { |
| 822 MotifWmHints motif_hints; |
| 823 memset(&motif_hints, 0, sizeof(motif_hints)); |
| 824 motif_hints.flags = kHintsDecorations; |
| 825 motif_hints.decorations = use_os_border ? 1 : 0; |
| 826 XChangeProperty(xdisplay_, |
| 827 xwindow_, |
| 828 cached_atoms_[ATOM__MOTIF_WM_HINTS], |
| 829 cached_atoms_[ATOM__MOTIF_WM_HINTS], |
| 830 32, |
| 831 PropModeReplace, |
| 832 reinterpret_cast<unsigned char*>(&motif_hints), |
| 833 sizeof(MotifWmHints)/sizeof(long)); |
| 834 } |
| 835 |
| 836 bool RootWindowHostLinux::DispatchHostWindowDragMovement(int hittest, |
| 837 MouseEvent* event) { |
| 838 int direction = -1; |
| 839 switch (hittest) { |
| 840 case HTBOTTOM: |
| 841 direction = k_NET_WM_MOVERESIZE_SIZE_BOTTOM; |
| 842 break; |
| 843 case HTBOTTOMLEFT: |
| 844 direction = k_NET_WM_MOVERESIZE_SIZE_BOTTOMLEFT; |
| 845 break; |
| 846 case HTBOTTOMRIGHT: |
| 847 direction = k_NET_WM_MOVERESIZE_SIZE_BOTTOMRIGHT; |
| 848 break; |
| 849 case HTCAPTION: |
| 850 direction = k_NET_WM_MOVERESIZE_MOVE; |
| 851 break; |
| 852 case HTLEFT: |
| 853 direction = k_NET_WM_MOVERESIZE_SIZE_LEFT; |
| 854 break; |
| 855 case HTRIGHT: |
| 856 direction = k_NET_WM_MOVERESIZE_SIZE_RIGHT; |
| 857 break; |
| 858 case HTTOP: |
| 859 direction = k_NET_WM_MOVERESIZE_SIZE_TOP; |
| 860 break; |
| 861 case HTTOPLEFT: |
| 862 direction = k_NET_WM_MOVERESIZE_SIZE_TOPLEFT; |
| 863 break; |
| 864 case HTTOPRIGHT: |
| 865 direction = k_NET_WM_MOVERESIZE_SIZE_TOPRIGHT; |
| 866 break; |
| 867 default: |
| 868 return false; |
| 869 } |
| 870 |
| 871 gfx::Point point = event->root_location(); |
| 872 SendWMMoveResize(direction, point.x(), point.y()); |
| 873 return true; |
| 874 } |
| 875 |
789 bool RootWindowHostLinux::IsWindowManagerPresent() { | 876 bool RootWindowHostLinux::IsWindowManagerPresent() { |
790 // Per ICCCM 2.8, "Manager Selections", window managers should take ownership | 877 // Per ICCCM 2.8, "Manager Selections", window managers should take ownership |
791 // of WM_Sn selections (where n is a screen number). | 878 // of WM_Sn selections (where n is a screen number). |
792 return XGetSelectionOwner(xdisplay_, cached_atoms_[ATOM_WM_S0]) != None; | 879 return XGetSelectionOwner(xdisplay_, cached_atoms_[ATOM_WM_S0]) != None; |
793 } | 880 } |
794 | 881 |
795 void RootWindowHostLinux::SetCursorInternal(gfx::NativeCursor cursor) { | 882 void RootWindowHostLinux::SetCursorInternal(gfx::NativeCursor cursor) { |
796 ::Cursor xcursor = | 883 ::Cursor xcursor = |
797 cursor == ui::kCursorNone ? | 884 cursor == ui::kCursorNone ? |
798 invisible_cursor_ : | 885 invisible_cursor_ : |
799 cursor == ui::kCursorCustom ? | 886 cursor == ui::kCursorCustom ? |
800 cursor.platform() : | 887 cursor.platform() : |
801 ui::GetXCursor(CursorShapeFromNative(cursor)); | 888 ui::GetXCursor(CursorShapeFromNative(cursor)); |
802 XDefineCursor(xdisplay_, xwindow_, xcursor); | 889 XDefineCursor(xdisplay_, xwindow_, xcursor); |
803 } | 890 } |
804 | 891 |
| 892 void RootWindowHostLinux::SendWMMoveResize(int direction, |
| 893 int x_root, |
| 894 int y_root) { |
| 895 // We most likely have an implicit grab right here. We need to dump it |
| 896 // because what we're about to do is tell the window manager |
| 897 // that it's now responsible for moving the window around; it immediately |
| 898 // grabs when it receives the event below. |
| 899 XUngrabPointer(xdisplay_, CurrentTime); |
| 900 |
| 901 XEvent event; |
| 902 memset(&event, 0, sizeof(event)); |
| 903 event.xclient.type = ClientMessage; |
| 904 event.xclient.display = xdisplay_; |
| 905 event.xclient.window = xwindow_; |
| 906 event.xclient.message_type = cached_atoms_[ATOM__NET_WM_MOVERESIZE]; |
| 907 event.xclient.format = 32; |
| 908 event.xclient.data.l[0] = x_root; |
| 909 event.xclient.data.l[1] = y_root; |
| 910 event.xclient.data.l[2] = direction; |
| 911 event.xclient.data.l[3] = 0; |
| 912 event.xclient.data.l[4] = 0; |
| 913 |
| 914 XSendEvent(xdisplay_, x_root_window_, False, |
| 915 SubstructureRedirectMask | SubstructureNotifyMask, |
| 916 &event); |
| 917 } |
| 918 |
805 // static | 919 // static |
806 RootWindowHost* RootWindowHost::Create(const gfx::Rect& bounds) { | 920 RootWindowHost* RootWindowHost::Create(const gfx::Rect& bounds) { |
807 return new RootWindowHostLinux(bounds); | 921 return new RootWindowHostLinux(bounds); |
808 } | 922 } |
809 | 923 |
810 // static | 924 // static |
811 RootWindowHost* RootWindowHost::GetForAcceleratedWidget( | 925 RootWindowHost* RootWindowHost::GetForAcceleratedWidget( |
812 gfx::AcceleratedWidget accelerated_widget) { | 926 gfx::AcceleratedWidget accelerated_widget) { |
813 return reinterpret_cast<RootWindowHost*>( | 927 return reinterpret_cast<RootWindowHost*>( |
814 ui::ViewProp::GetValue(accelerated_widget, kRootWindowHostLinuxKey)); | 928 ui::ViewProp::GetValue(accelerated_widget, kRootWindowHostLinuxKey)); |
815 } | 929 } |
816 | 930 |
817 // static | 931 // static |
818 gfx::Size RootWindowHost::GetNativeScreenSize() { | 932 gfx::Size RootWindowHost::GetNativeScreenSize() { |
819 ::Display* xdisplay = base::MessagePumpX::GetDefaultXDisplay(); | 933 ::Display* xdisplay = base::MessagePumpX::GetDefaultXDisplay(); |
820 return gfx::Size(DisplayWidth(xdisplay, 0), DisplayHeight(xdisplay, 0)); | 934 return gfx::Size(DisplayWidth(xdisplay, 0), DisplayHeight(xdisplay, 0)); |
821 } | 935 } |
822 | 936 |
823 } // namespace aura | 937 } // namespace aura |
OLD | NEW |