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
3 // found in the LICENSE file.
4
5 #include "ui/gfx/win/hwnd_util.h"
6
7 #include "base/i18n/rtl.h"
8 #include "base/strings/string_util.h"
9 #include "base/win/metro.h"
10 #include "base/win/win_util.h"
11 #include "ui/gfx/point.h"
12 #include "ui/gfx/rect.h"
13 #include "ui/gfx/size.h"
14
15 namespace gfx {
16
17 namespace {
18
19 // Adjust the window to fit.
AdjustWindowToFit(HWND hwnd,const RECT & bounds,bool fit_to_monitor)20 void AdjustWindowToFit(HWND hwnd, const RECT& bounds, bool fit_to_monitor) {
21 if (fit_to_monitor) {
22 // Get the monitor.
23 HMONITOR hmon = MonitorFromRect(&bounds, MONITOR_DEFAULTTONEAREST);
24 if (hmon) {
25 MONITORINFO mi;
26 mi.cbSize = sizeof(mi);
27 GetMonitorInfo(hmon, &mi);
28 Rect window_rect(bounds);
29 Rect monitor_rect(mi.rcWork);
30 Rect new_window_rect = window_rect;
31 new_window_rect.AdjustToFit(monitor_rect);
32 if (new_window_rect != window_rect) {
33 // Window doesn't fit on monitor, move and possibly resize.
34 SetWindowPos(hwnd, 0, new_window_rect.x(), new_window_rect.y(),
35 new_window_rect.width(), new_window_rect.height(),
36 SWP_NOACTIVATE | SWP_NOZORDER);
37 return;
38 }
39 // Else fall through.
40 } else {
41 NOTREACHED() << "Unable to find default monitor";
42 // Fall through.
43 }
44 } // Else fall through.
45
46 // The window is not being fit to monitor, or the window fits on the monitor
47 // as is, or we have no monitor info; reset the bounds.
48 ::SetWindowPos(hwnd, 0, bounds.left, bounds.top,
49 bounds.right - bounds.left, bounds.bottom - bounds.top,
50 SWP_NOACTIVATE | SWP_NOZORDER);
51 }
52
53 // Turn off optimizations for these functions so they show up in crash reports.
54 MSVC_DISABLE_OPTIMIZE();
55
CrashOutOfMemory()56 void CrashOutOfMemory() {
57 PLOG(FATAL);
58 }
59
CrashAccessDenied()60 void CrashAccessDenied() {
61 PLOG(FATAL);
62 }
63
64 // Crash isn't one of the ones we commonly see.
CrashOther()65 void CrashOther() {
66 PLOG(FATAL);
67 }
68
69 MSVC_ENABLE_OPTIMIZE();
70
71 } // namespace
72
GetClassName(HWND window)73 base::string16 GetClassName(HWND window) {
74 // GetClassNameW will return a truncated result (properly null terminated) if
75 // the given buffer is not large enough. So, it is not possible to determine
76 // that we got the entire class name if the result is exactly equal to the
77 // size of the buffer minus one.
78 DWORD buffer_size = MAX_PATH;
79 while (true) {
80 std::wstring output;
81 DWORD size_ret =
82 GetClassNameW(window, WriteInto(&output, buffer_size), buffer_size);
83 if (size_ret == 0)
84 break;
85 if (size_ret < (buffer_size - 1)) {
86 output.resize(size_ret);
87 return output;
88 }
89 buffer_size *= 2;
90 }
91 return std::wstring(); // error
92 }
93
94 #pragma warning(push)
95 #pragma warning(disable:4312 4244)
96
SetWindowProc(HWND hwnd,WNDPROC proc)97 WNDPROC SetWindowProc(HWND hwnd, WNDPROC proc) {
98 // The reason we don't return the SetwindowLongPtr() value is that it returns
99 // the orignal window procedure and not the current one. I don't know if it is
100 // a bug or an intended feature.
101 WNDPROC oldwindow_proc =
102 reinterpret_cast<WNDPROC>(GetWindowLongPtr(hwnd, GWLP_WNDPROC));
103 SetWindowLongPtr(hwnd, GWLP_WNDPROC, reinterpret_cast<LONG_PTR>(proc));
104 return oldwindow_proc;
105 }
106
SetWindowUserData(HWND hwnd,void * user_data)107 void* SetWindowUserData(HWND hwnd, void* user_data) {
108 return
109 reinterpret_cast<void*>(SetWindowLongPtr(hwnd, GWLP_USERDATA,
110 reinterpret_cast<LONG_PTR>(user_data)));
111 }
112
GetWindowUserData(HWND hwnd)113 void* GetWindowUserData(HWND hwnd) {
114 DWORD process_id = 0;
115 DWORD thread_id = GetWindowThreadProcessId(hwnd, &process_id);
116 // A window outside the current process needs to be ignored.
117 if (process_id != ::GetCurrentProcessId())
118 return NULL;
119 return reinterpret_cast<void*>(GetWindowLongPtr(hwnd, GWLP_USERDATA));
120 }
121
122 #pragma warning(pop)
123
DoesWindowBelongToActiveWindow(HWND window)124 bool DoesWindowBelongToActiveWindow(HWND window) {
125 DCHECK(window);
126 HWND top_window = ::GetAncestor(window, GA_ROOT);
127 if (!top_window)
128 return false;
129
130 HWND active_top_window = ::GetAncestor(::GetForegroundWindow(), GA_ROOT);
131 return (top_window == active_top_window);
132 }
133
CenterAndSizeWindow(HWND parent,HWND window,const Size & pref)134 void CenterAndSizeWindow(HWND parent,
135 HWND window,
136 const Size& pref) {
137 DCHECK(window && pref.width() > 0 && pref.height() > 0);
138
139 // Calculate the ideal bounds.
140 RECT window_bounds;
141 RECT center_bounds = {0};
142 if (parent) {
143 // If there is a parent, center over the parents bounds.
144 ::GetWindowRect(parent, ¢er_bounds);
145 }
146
147 if (::IsRectEmpty(¢er_bounds)) {
148 // No parent or no parent rect. Center over the monitor the window is on.
149 HMONITOR monitor = MonitorFromWindow(window, MONITOR_DEFAULTTONEAREST);
150 if (monitor) {
151 MONITORINFO mi = {0};
152 mi.cbSize = sizeof(mi);
153 GetMonitorInfo(monitor, &mi);
154 center_bounds = mi.rcWork;
155 } else {
156 NOTREACHED() << "Unable to get default monitor";
157 }
158 }
159
160 window_bounds.left = center_bounds.left;
161 if (pref.width() < (center_bounds.right - center_bounds.left)) {
162 window_bounds.left +=
163 (center_bounds.right - center_bounds.left - pref.width()) / 2;
164 }
165 window_bounds.right = window_bounds.left + pref.width();
166
167 window_bounds.top = center_bounds.top;
168 if (pref.height() < (center_bounds.bottom - center_bounds.top)) {
169 window_bounds.top +=
170 (center_bounds.bottom - center_bounds.top - pref.height()) / 2;
171 }
172 window_bounds.bottom = window_bounds.top + pref.height();
173
174 // If we're centering a child window, we are positioning in client
175 // coordinates, and as such we need to offset the target rectangle by the
176 // position of the parent window.
177 if (::GetWindowLong(window, GWL_STYLE) & WS_CHILD) {
178 DCHECK(parent && ::GetParent(window) == parent);
179 POINT topleft = { window_bounds.left, window_bounds.top };
180 ::MapWindowPoints(HWND_DESKTOP, parent, &topleft, 1);
181 window_bounds.left = topleft.x;
182 window_bounds.top = topleft.y;
183 window_bounds.right = window_bounds.left + pref.width();
184 window_bounds.bottom = window_bounds.top + pref.height();
185 }
186
187 AdjustWindowToFit(window, window_bounds, !parent);
188 }
189
CheckWindowCreated(HWND hwnd)190 void CheckWindowCreated(HWND hwnd) {
191 if (!hwnd) {
192 switch (GetLastError()) {
193 case ERROR_NOT_ENOUGH_MEMORY:
194 CrashOutOfMemory();
195 break;
196 case ERROR_ACCESS_DENIED:
197 CrashAccessDenied();
198 break;
199 default:
200 CrashOther();
201 break;
202 }
203 PLOG(FATAL);
204 }
205 }
206
ShowSystemMenu(HWND window)207 void ShowSystemMenu(HWND window) {
208 RECT rect;
209 GetWindowRect(window, &rect);
210 Point point = Point(base::i18n::IsRTL() ? rect.right : rect.left, rect.top);
211 static const int kSystemMenuOffset = 10;
212 point.Offset(base::i18n::IsRTL() ? -kSystemMenuOffset : kSystemMenuOffset,
213 kSystemMenuOffset);
214 ShowSystemMenuAtPoint(window, point);
215 }
216
ShowSystemMenuAtPoint(HWND window,const Point & point)217 void ShowSystemMenuAtPoint(HWND window, const Point& point) {
218 // In the Metro process, we never want to show the system menu.
219 if (base::win::IsMetroProcess())
220 return;
221 UINT flags = TPM_LEFTBUTTON | TPM_RIGHTBUTTON | TPM_RETURNCMD;
222 if (base::i18n::IsRTL())
223 flags |= TPM_RIGHTALIGN;
224 HMENU menu = GetSystemMenu(window, FALSE);
225 const int command =
226 TrackPopupMenu(menu, flags, point.x(), point.y(), 0, window, NULL);
227 if (command)
228 SendMessage(window, WM_SYSCOMMAND, command, 0);
229 }
230
231 extern "C" {
232 typedef HWND (*RootWindow)();
233 }
234
GetWindowToParentTo(bool get_real_hwnd)235 HWND GetWindowToParentTo(bool get_real_hwnd) {
236 HMODULE metro = base::win::GetMetroModule();
237 if (!metro)
238 return get_real_hwnd ? ::GetDesktopWindow() : HWND_DESKTOP;
239 // In windows 8 metro-mode the root window is not the desktop.
240 RootWindow root_window =
241 reinterpret_cast<RootWindow>(::GetProcAddress(metro, "GetRootWindow"));
242 return root_window();
243 }
244
245 } // namespace gfx
246