VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/VBoxTray/VBoxHelpers.cpp@ 55365

Last change on this file since 55365 was 48070, checked in by vboxsync, 11 years ago

wddm,vboxtray: forward-port autoresize, multimon, and seamless fixes from 4.2 r87071, r87353, r87356, r87528, r87568, r87581, r87584, r87608, r87673, r87678, r87708, r87629, r87529; additional fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 10.5 KB
Line 
1/** @file
2 * helpers - Guest Additions Service helper functions
3 */
4
5/*
6 * Copyright (C) 2006-2011 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 */
16
17#include <malloc.h>
18#include <windows.h>
19
20#include <iprt/string.h>
21#include <VBox/Log.h>
22#include <VBox/VBoxGuestLib.h>
23
24#include <VBoxGuestInternal.h>
25
26#include "VBoxHelpers.h"
27#include "resource.h"
28
29
30int hlpReportStatus(VBoxGuestFacilityStatus statusCurrent)
31{
32 int rc = VbglR3ReportAdditionsStatus(VBoxGuestFacilityType_VBoxTrayClient,
33 statusCurrent,
34 0 /* Flags */);
35 if (RT_FAILURE(rc))
36 Log(("VBoxTray: Could not report VBoxTray status \"%ld\", rc=%Rrc\n", statusCurrent, rc));
37 return rc;
38}
39
40/**
41 * Attempt to force Windows to reload the cursor image by attaching to the
42 * thread of the window currently under the mouse, hiding the cursor and
43 * showing it again. This could fail to work in any number of ways (no
44 * window under the cursor, the cursor has moved to a different window while
45 * we are processing), but we just accept this, as the cursor will be reloaded
46 * at some point anyway.
47 */
48void hlpReloadCursor(void)
49{
50 POINT mousePos;
51 HWND hWin;
52 DWORD hThread, hCurrentThread;
53
54 GetCursorPos(&mousePos);
55 hWin = WindowFromPoint(mousePos);
56 if (hWin)
57 {
58 hThread = GetWindowThreadProcessId(hWin, NULL);
59 hCurrentThread = GetCurrentThreadId();
60 if (hCurrentThread != hThread)
61 AttachThreadInput(hCurrentThread, hThread, TRUE);
62 }
63 ShowCursor(false);
64 ShowCursor(true);
65 if (hWin && (hCurrentThread != hThread))
66 AttachThreadInput(hCurrentThread, hThread, FALSE);
67}
68
69static unsigned hlpNextAdjacentRectXP(RECTL *paRects, unsigned nRects, unsigned uRect)
70{
71 unsigned i;
72 for (i = 0; i < nRects; i++)
73 {
74 if (paRects[uRect].right == paRects[i].left)
75 return i;
76 }
77 return ~0;
78}
79
80static unsigned hlpNextAdjacentRectXN(RECTL *paRects, unsigned nRects, unsigned uRect)
81{
82 unsigned i;
83 for (i = 0; i < nRects; i++)
84 {
85 if (paRects[uRect].left == paRects[i].right)
86 return i;
87 }
88 return ~0;
89}
90
91static unsigned hlpNextAdjacentRectYP(RECTL *paRects, unsigned nRects, unsigned uRect)
92{
93 unsigned i;
94 for (i = 0; i < nRects; i++)
95 {
96 if (paRects[uRect].bottom == paRects[i].top)
97 return i;
98 }
99 return ~0;
100}
101
102static unsigned hlpNextAdjacentRectYN(RECTL *paRects, unsigned nRects, unsigned uRect)
103{
104 unsigned i;
105 for (i = 0; i < nRects; i++)
106 {
107 if (paRects[uRect].top == paRects[i].bottom)
108 return i;
109 }
110 return ~0;
111}
112
113void hlpResizeRect(RECTL *paRects, unsigned nRects, unsigned uPrimary,
114 unsigned uResized, int iNewWidth, int iNewHeight,
115 int iNewPosX, int iNewPosY)
116{
117 DDCLOG(("nRects %d, iPrimary %d, iResized %d, NewWidth %d, NewHeight %d\n", nRects, uPrimary, uResized, iNewWidth, iNewHeight));
118
119 RECTL *paNewRects = (RECTL *)alloca (sizeof (RECTL) * nRects);
120 memcpy (paNewRects, paRects, sizeof (RECTL) * nRects);
121 paNewRects[uResized].right += iNewWidth - (paNewRects[uResized].right - paNewRects[uResized].left);
122 paNewRects[uResized].bottom += iNewHeight - (paNewRects[uResized].bottom - paNewRects[uResized].top);
123 paNewRects[uResized].right += iNewPosX - paNewRects[uResized].left;
124 paNewRects[uResized].bottom += iNewPosY - paNewRects[uResized].top;
125 paNewRects[uResized].left = iNewPosX;
126 paNewRects[uResized].top = iNewPosY;
127
128 /* Verify all pairs of originally adjacent rectangles for all 4 directions.
129 * If the pair has a "good" delta (that is the first rectangle intersects the second)
130 * at a direction and the second rectangle is not primary one (which can not be moved),
131 * move the second rectangle to make it adjacent to the first one.
132 */
133
134 /* X positive. */
135 unsigned iRect;
136 for (iRect = 0; iRect < nRects; iRect++)
137 {
138 /* Find the next adjacent original rect in x positive direction. */
139 unsigned iNextRect = hlpNextAdjacentRectXP(paRects, nRects, iRect);
140 DDCLOG(("next %d -> %d\n", iRect, iNextRect));
141
142 if (iNextRect == ~0 || iNextRect == uPrimary)
143 {
144 continue;
145 }
146
147 /* Check whether there is an X intersection between these adjacent rects in the new rectangles
148 * and fix the intersection if delta is "good".
149 */
150 int delta = paNewRects[iRect].right - paNewRects[iNextRect].left;
151
152 if (delta != 0)
153 {
154 DDCLOG(("XP intersection right %d left %d, diff %d\n",
155 paNewRects[iRect].right, paNewRects[iNextRect].left,
156 delta));
157
158 paNewRects[iNextRect].left += delta;
159 paNewRects[iNextRect].right += delta;
160 }
161 }
162
163 /* X negative. */
164 for (iRect = 0; iRect < nRects; iRect++)
165 {
166 /* Find the next adjacent original rect in x negative direction. */
167 unsigned iNextRect = hlpNextAdjacentRectXN(paRects, nRects, iRect);
168 DDCLOG(("next %d -> %d\n", iRect, iNextRect));
169
170 if (iNextRect == ~0 || iNextRect == uPrimary)
171 {
172 continue;
173 }
174
175 /* Check whether there is an X intersection between these adjacent rects in the new rectangles
176 * and fix the intersection if delta is "good".
177 */
178 int delta = paNewRects[iRect].left - paNewRects[iNextRect].right;
179
180 if (delta != 0)
181 {
182 DDCLOG(("XN intersection left %d right %d, diff %d\n",
183 paNewRects[iRect].left, paNewRects[iNextRect].right,
184 delta));
185
186 paNewRects[iNextRect].left += delta;
187 paNewRects[iNextRect].right += delta;
188 }
189 }
190
191 /* Y positive (in the computer sense, top->down). */
192 for (iRect = 0; iRect < nRects; iRect++)
193 {
194 /* Find the next adjacent original rect in y positive direction. */
195 unsigned iNextRect = hlpNextAdjacentRectYP(paRects, nRects, iRect);
196 DDCLOG(("next %d -> %d\n", iRect, iNextRect));
197
198 if (iNextRect == ~0 || iNextRect == uPrimary)
199 {
200 continue;
201 }
202
203 /* Check whether there is an Y intersection between these adjacent rects in the new rectangles
204 * and fix the intersection if delta is "good".
205 */
206 int delta = paNewRects[iRect].bottom - paNewRects[iNextRect].top;
207
208 if (delta != 0)
209 {
210 DDCLOG(("YP intersection bottom %d top %d, diff %d\n",
211 paNewRects[iRect].bottom, paNewRects[iNextRect].top,
212 delta));
213
214 paNewRects[iNextRect].top += delta;
215 paNewRects[iNextRect].bottom += delta;
216 }
217 }
218
219 /* Y negative (in the computer sense, down->top). */
220 for (iRect = 0; iRect < nRects; iRect++)
221 {
222 /* Find the next adjacent original rect in x negative direction. */
223 unsigned iNextRect = hlpNextAdjacentRectYN(paRects, nRects, iRect);
224 DDCLOG(("next %d -> %d\n", iRect, iNextRect));
225
226 if (iNextRect == ~0 || iNextRect == uPrimary)
227 {
228 continue;
229 }
230
231 /* Check whether there is an Y intersection between these adjacent rects in the new rectangles
232 * and fix the intersection if delta is "good".
233 */
234 int delta = paNewRects[iRect].top - paNewRects[iNextRect].bottom;
235
236 if (delta != 0)
237 {
238 DDCLOG(("YN intersection top %d bottom %d, diff %d\n",
239 paNewRects[iRect].top, paNewRects[iNextRect].bottom,
240 delta));
241
242 paNewRects[iNextRect].top += delta;
243 paNewRects[iNextRect].bottom += delta;
244 }
245 }
246
247 memcpy (paRects, paNewRects, sizeof (RECTL) * nRects);
248 return;
249}
250
251int hlpShowBalloonTip(HINSTANCE hInst, HWND hWnd, UINT uID,
252 const char *pszMsg, const char *pszTitle,
253 UINT uTimeout, DWORD dwInfoFlags)
254{
255 NOTIFYICONDATA niData;
256 ZeroMemory(&niData, sizeof(NOTIFYICONDATA));
257 niData.cbSize = sizeof(NOTIFYICONDATA);
258 niData.uFlags = NIF_INFO; /* Display a balloon notification. */
259 niData.hWnd = hWnd;
260 niData.uID = uID;
261 /* If not timeout set, set it to 5sec. */
262 if (uTimeout == 0)
263 uTimeout = 5000;
264 niData.uTimeout = uTimeout;
265 /* If no info flag (info, warning, error) set,
266 * set it to info by default. */
267 if (dwInfoFlags == 0)
268 dwInfoFlags = NIIF_INFO;
269 niData.dwInfoFlags = dwInfoFlags;
270
271 /* Do we want to have */
272
273 /* Get running OS version. */
274 OSVERSIONINFO osInfo;
275 osInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
276 if (FALSE == GetVersionEx(&osInfo))
277 return RTErrConvertFromWin32(GetLastError());
278
279 /* Is the current OS supported (at least WinXP) for displaying
280 * our own icon and do we actually *want* to display our own stuff? */
281 if ( osInfo.dwMajorVersion >= 5
282 && (dwInfoFlags & NIIF_INFO))
283 {
284 /* Load (or retrieve handle of) the app's icon. */
285 HICON hIcon = LoadIcon(hInst, MAKEINTRESOURCE(IDI_VIRTUALBOX));
286 if (hIcon)
287 niData.dwInfoFlags = NIIF_USER; /* Use an own notification icon. */
288
289 if ( osInfo.dwMajorVersion == 5
290 && osInfo.dwMinorVersion == 1) /* WinXP. */
291 {
292 /* Use an own icon instead of the default one. */
293 niData.hIcon = hIcon;
294 }
295 else if (osInfo.dwMajorVersion == 6) /* Vista and up. */
296 {
297 /* Use an own icon instead of the default one. */
298 niData.dwInfoFlags |= NIIF_LARGE_ICON; /* Use a large icon if available! */
299 niData.hIcon = hIcon;
300 niData.hBalloonIcon = hIcon;
301 }
302 }
303 else
304 {
305 /* This might be a warning, error message or a to old OS. Use the
306 * standard icons provided by Windows (if any). */
307 }
308
309 strcpy(niData.szInfo, pszMsg ? pszMsg : "-");
310 strcpy(niData.szInfoTitle, pszTitle ? pszTitle : "Information");
311
312 if (!Shell_NotifyIcon(NIM_MODIFY, &niData))
313 {
314 DWORD dwErr = GetLastError();
315 return RTErrConvertFromWin32(dwErr);
316 }
317 return VINF_SUCCESS;
318}
319
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette