OLD | NEW |
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 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/views/corewm/tooltip_win.h" | 5 #include "ui/views/corewm/tooltip_win.h" |
6 | 6 |
7 #include <winuser.h> | 7 #include <winuser.h> |
8 | 8 |
9 #include "base/debug/stack_trace.h" | 9 #include "base/debug/stack_trace.h" |
10 #include "base/i18n/rtl.h" | 10 #include "base/i18n/rtl.h" |
11 #include "base/logging.h" | 11 #include "base/logging.h" |
12 #include "ui/base/l10n/l10n_util_win.h" | 12 #include "ui/base/l10n/l10n_util_win.h" |
13 #include "ui/gfx/rect.h" | 13 #include "ui/gfx/rect.h" |
14 #include "ui/gfx/screen.h" | 14 #include "ui/gfx/screen.h" |
15 #include "ui/gfx/win/dpi.h" | 15 #include "ui/gfx/win/dpi.h" |
16 | 16 #include "ui/views/corewm/cursor_height_provider_win.h" |
17 | 17 |
18 namespace views { | 18 namespace views { |
19 namespace corewm { | 19 namespace corewm { |
20 | 20 |
21 TooltipWin::TooltipWin(HWND parent) | 21 TooltipWin::TooltipWin(HWND parent) |
22 : parent_hwnd_(parent), | 22 : parent_hwnd_(parent), |
23 tooltip_hwnd_(NULL), | 23 tooltip_hwnd_(NULL), |
24 showing_(false) { | 24 showing_(false) { |
25 memset(&toolinfo_, 0, sizeof(toolinfo_)); | 25 memset(&toolinfo_, 0, sizeof(toolinfo_)); |
26 toolinfo_.cbSize = sizeof(toolinfo_); | 26 toolinfo_.cbSize = sizeof(toolinfo_); |
(...skipping 44 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
71 | 71 |
72 l10n_util::AdjustUIFontForWindow(tooltip_hwnd_); | 72 l10n_util::AdjustUIFontForWindow(tooltip_hwnd_); |
73 | 73 |
74 SendMessage(tooltip_hwnd_, TTM_ADDTOOL, 0, | 74 SendMessage(tooltip_hwnd_, TTM_ADDTOOL, 0, |
75 reinterpret_cast<LPARAM>(&toolinfo_)); | 75 reinterpret_cast<LPARAM>(&toolinfo_)); |
76 return true; | 76 return true; |
77 } | 77 } |
78 | 78 |
79 void TooltipWin::PositionTooltip() { | 79 void TooltipWin::PositionTooltip() { |
80 // This code only runs for non-metro, so GetNativeScreen() is fine. | 80 // This code only runs for non-metro, so GetNativeScreen() is fine. |
81 gfx::Display display( | 81 gfx::Point screen_point = gfx::win::DIPToScreenPoint(location_); |
82 gfx::Screen::GetNativeScreen()->GetDisplayNearestPoint(location_)); | 82 const int cursoroffset = GetCurrentCursorVisibleHeight(); |
| 83 screen_point.Offset(0, cursoroffset); |
83 | 84 |
84 DWORD tooltip_size = SendMessage(tooltip_hwnd_, TTM_GETBUBBLESIZE, 0, | 85 DWORD tooltip_size = SendMessage(tooltip_hwnd_, TTM_GETBUBBLESIZE, 0, |
85 reinterpret_cast<LPARAM>(&toolinfo_)); | 86 reinterpret_cast<LPARAM>(&toolinfo_)); |
86 // 20 accounts for visible cursor size. I tried using SM_CYCURSOR but that's | 87 const gfx::Size size(LOWORD(tooltip_size), HIWORD(tooltip_size)); |
87 // way too big (32 on win7 default). | |
88 // TODO(sky): figure out the right way to determine offset. | |
89 const int initial_y = location_.y(); | |
90 gfx::Rect tooltip_bounds(location_.x(), initial_y + 20, | |
91 LOWORD(tooltip_size), HIWORD(tooltip_size)); | |
92 tooltip_bounds.AdjustToFit(display.work_area()); | |
93 if (tooltip_bounds.y() < initial_y) | |
94 tooltip_bounds.set_y(initial_y - tooltip_bounds.height() - 2); | |
95 | 88 |
96 // Convert the tooltip bounds to pixel coordinates. SetWindowPos works in | 89 const gfx::Display display( |
97 // pixel coordinates. | 90 gfx::Screen::GetNativeScreen()->GetDisplayNearestPoint(screen_point)); |
98 tooltip_bounds = gfx::win::DIPToScreenRect(tooltip_bounds); | 91 |
| 92 gfx::Rect tooltip_bounds(screen_point, size); |
| 93 tooltip_bounds.AdjustToFit(gfx::win::DIPToScreenRect(display.work_area())); |
99 SetWindowPos(tooltip_hwnd_, NULL, tooltip_bounds.x(), tooltip_bounds.y(), 0, | 94 SetWindowPos(tooltip_hwnd_, NULL, tooltip_bounds.x(), tooltip_bounds.y(), 0, |
100 0, SWP_NOSIZE | SWP_NOZORDER | SWP_NOACTIVATE); | 95 0, SWP_NOSIZE | SWP_NOZORDER | SWP_NOACTIVATE); |
101 } | 96 } |
102 | 97 |
103 void TooltipWin::SetText(aura::Window* window, | 98 void TooltipWin::SetText(aura::Window* window, |
104 const base::string16& tooltip_text, | 99 const base::string16& tooltip_text, |
105 const gfx::Point& location) { | 100 const gfx::Point& location) { |
106 if (!EnsureTooltipWindow()) | 101 if (!EnsureTooltipWindow()) |
107 return; | 102 return; |
108 | 103 |
109 // See comment in header for details on why |location_| is needed. | 104 // See comment in header for details on why |location_| is needed. |
110 location_ = location; | 105 location_ = location; |
111 | 106 |
112 // Without this we get a flicker of the tooltip appearing at 0x0. Not sure | 107 // Without this we get a flicker of the tooltip appearing at 0x0. Not sure |
113 // why. | 108 // why. |
114 SetWindowPos(tooltip_hwnd_, NULL, 0, 0, 0, 0, | 109 SetWindowPos(tooltip_hwnd_, NULL, 0, 0, 0, 0, |
115 SWP_HIDEWINDOW | SWP_NOACTIVATE | SWP_NOMOVE | | 110 SWP_HIDEWINDOW | SWP_NOACTIVATE | SWP_NOMOVE | |
116 SWP_NOREPOSITION | SWP_NOSIZE | SWP_NOZORDER); | 111 SWP_NOREPOSITION | SWP_NOSIZE | SWP_NOZORDER); |
117 | 112 |
118 base::string16 adjusted_text(tooltip_text); | 113 base::string16 adjusted_text(tooltip_text); |
119 base::i18n::AdjustStringForLocaleDirection(&adjusted_text); | 114 base::i18n::AdjustStringForLocaleDirection(&adjusted_text); |
120 toolinfo_.lpszText = const_cast<WCHAR*>(adjusted_text.c_str()); | 115 toolinfo_.lpszText = const_cast<WCHAR*>(adjusted_text.c_str()); |
121 SendMessage(tooltip_hwnd_, TTM_SETTOOLINFO, 0, | 116 SendMessage(tooltip_hwnd_, TTM_SETTOOLINFO, 0, |
122 reinterpret_cast<LPARAM>(&toolinfo_)); | 117 reinterpret_cast<LPARAM>(&toolinfo_)); |
123 | 118 |
124 // This code only runs for non-metro, so GetNativeScreen() is fine. | 119 // This code only runs for non-metro, so GetNativeScreen() is fine. |
| 120 const gfx::Point screen_point = gfx::win::DIPToScreenPoint(location_); |
125 gfx::Display display( | 121 gfx::Display display( |
126 gfx::Screen::GetNativeScreen()->GetDisplayNearestPoint(location_)); | 122 gfx::Screen::GetNativeScreen()->GetDisplayNearestPoint(screen_point)); |
127 const gfx::Rect monitor_bounds = display.bounds(); | 123 const gfx::Rect monitor_bounds = display.bounds(); |
128 int max_width = (monitor_bounds.width() + 1) / 2; | 124 int max_width = (monitor_bounds.width() + 1) / 2; |
129 SendMessage(tooltip_hwnd_, TTM_SETMAXTIPWIDTH, 0, max_width); | 125 SendMessage(tooltip_hwnd_, TTM_SETMAXTIPWIDTH, 0, max_width); |
130 } | 126 } |
131 | 127 |
132 void TooltipWin::Show() { | 128 void TooltipWin::Show() { |
133 if (!EnsureTooltipWindow()) | 129 if (!EnsureTooltipWindow()) |
134 return; | 130 return; |
135 | 131 |
136 SendMessage(tooltip_hwnd_, TTM_TRACKACTIVATE, | 132 SendMessage(tooltip_hwnd_, TTM_TRACKACTIVATE, |
137 TRUE, reinterpret_cast<LPARAM>(&toolinfo_)); | 133 TRUE, reinterpret_cast<LPARAM>(&toolinfo_)); |
138 SetWindowPos(tooltip_hwnd_, HWND_TOPMOST, 0, 0, 0, 0, | 134 SetWindowPos(tooltip_hwnd_, HWND_TOPMOST, 0, 0, 0, 0, |
139 SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOOWNERZORDER | SWP_NOSIZE); | 135 SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOOWNERZORDER | SWP_NOSIZE); |
140 } | 136 } |
141 | 137 |
142 void TooltipWin::Hide() { | 138 void TooltipWin::Hide() { |
143 if (!tooltip_hwnd_) | 139 if (!tooltip_hwnd_) |
144 return; | 140 return; |
145 | 141 |
146 SendMessage(tooltip_hwnd_, TTM_TRACKACTIVATE, FALSE, | 142 SendMessage(tooltip_hwnd_, TTM_TRACKACTIVATE, FALSE, |
147 reinterpret_cast<LPARAM>(&toolinfo_)); | 143 reinterpret_cast<LPARAM>(&toolinfo_)); |
148 } | 144 } |
149 | 145 |
150 bool TooltipWin::IsVisible() { | 146 bool TooltipWin::IsVisible() { |
151 return showing_; | 147 return showing_; |
152 } | 148 } |
153 | 149 |
154 } // namespace corewm | 150 } // namespace corewm |
155 } // namespace views | 151 } // namespace views |
OLD | NEW |