VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/VBoxTray/VBoxDnDDropTarget.cpp@ 58306

Last change on this file since 58306 was 57297, checked in by vboxsync, 9 years ago

DnD/VBoxTray: Fixed memory leak.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 21.3 KB
Line 
1/* $Id: VBoxDnDDropTarget.cpp 57297 2015-08-12 14:31:28Z vboxsync $ */
2/** @file
3 * VBoxDnDTarget.cpp - IDropTarget implementation.
4 */
5
6/*
7 * Copyright (C) 2014-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17#include <windows.h>
18#include <new> /* For bad_alloc. */
19#include <Shlobj.h> /* For DROPFILES and friends. */
20
21#include "VBoxTray.h"
22#include "VBoxHelpers.h"
23#include "VBoxDnD.h"
24
25#include "VBox/GuestHost/DragAndDrop.h"
26#include "VBox/HostServices/DragAndDropSvc.h"
27
28#ifdef LOG_GROUP
29# undef LOG_GROUP
30#endif
31#define LOG_GROUP LOG_GROUP_GUEST_DND
32#include <VBox/log.h>
33
34
35
36VBoxDnDDropTarget::VBoxDnDDropTarget(VBoxDnDWnd *pParent)
37 : mRefCount(1),
38 mpWndParent(pParent),
39 mdwCurEffect(0),
40 mpvData(NULL),
41 mcbData(0),
42 hEventDrop(NIL_RTSEMEVENT)
43{
44 int rc = VbglR3DnDConnect(&mDnDCtx);
45 if (RT_SUCCESS(rc))
46 rc = RTSemEventCreate(&hEventDrop);
47
48 LogFlowFunc(("clientID=%RU32, rc=%Rrc\n", mDnDCtx.uClientID, rc));
49}
50
51VBoxDnDDropTarget::~VBoxDnDDropTarget(void)
52{
53 reset();
54
55 int rc2 = VbglR3DnDDisconnect(&mDnDCtx);
56 AssertRC(rc2);
57 rc2 = RTSemEventDestroy(hEventDrop);
58 AssertRC(rc2);
59
60 LogFlowFunc(("rc=%Rrc, mRefCount=%RI32\n", rc2, mRefCount));
61}
62
63/*
64 * IUnknown methods.
65 */
66
67STDMETHODIMP_(ULONG) VBoxDnDDropTarget::AddRef(void)
68{
69 return InterlockedIncrement(&mRefCount);
70}
71
72STDMETHODIMP_(ULONG) VBoxDnDDropTarget::Release(void)
73{
74 LONG lCount = InterlockedDecrement(&mRefCount);
75 if (lCount == 0)
76 {
77 delete this;
78 return 0;
79 }
80
81 return lCount;
82}
83
84STDMETHODIMP VBoxDnDDropTarget::QueryInterface(REFIID iid, void **ppvObject)
85{
86 AssertPtrReturn(ppvObject, E_INVALIDARG);
87
88 if ( iid == IID_IDropSource
89 || iid == IID_IUnknown)
90 {
91 AddRef();
92 *ppvObject = this;
93 return S_OK;
94 }
95
96 *ppvObject = 0;
97 return E_NOINTERFACE;
98}
99
100/*
101 * IDropTarget methods.
102 */
103
104STDMETHODIMP VBoxDnDDropTarget::DragEnter(IDataObject *pDataObject, DWORD grfKeyState,
105 POINTL pt, DWORD *pdwEffect)
106{
107 AssertPtrReturn(pDataObject, E_INVALIDARG);
108 AssertPtrReturn(pdwEffect, E_INVALIDARG);
109
110 LogFlowFunc(("pDataObject=0x%p, grfKeyState=0x%x, x=%ld, y=%ld, dwEffect=%RU32\n",
111 pDataObject, grfKeyState, pt.x, pt.y, *pdwEffect));
112
113 reset();
114
115 /** @todo At the moment we only support one DnD format at a time. */
116
117 /* Try different formats. CF_HDROP is the most common one, so start
118 * with this. */
119 FORMATETC fmtEtc = { CF_HDROP, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
120 HRESULT hr = pDataObject->QueryGetData(&fmtEtc);
121 if (hr == S_OK)
122 {
123 mFormats = "text/uri-list";
124 }
125 else
126 {
127 LogFlowFunc(("CF_HDROP not wanted, hr=%Rhrc\n", hr));
128
129 /* So we couldn't retrieve the data in CF_HDROP format; try with
130 * CF_UNICODETEXT + CF_TEXT formats now. Rest stays the same. */
131 fmtEtc.cfFormat = CF_UNICODETEXT;
132 hr = pDataObject->QueryGetData(&fmtEtc);
133 if (hr == S_OK)
134 {
135 mFormats = "text/plain;charset=utf-8";
136 }
137 else
138 {
139 LogFlowFunc(("CF_UNICODETEXT not wanted, hr=%Rhrc\n", hr));
140
141 fmtEtc.cfFormat = CF_TEXT;
142 hr = pDataObject->QueryGetData(&fmtEtc);
143 if (hr == S_OK)
144 {
145 mFormats = "text/plain;charset=utf-8";
146 }
147 else
148 {
149 LogFlowFunc(("CF_TEXT not wanted, hr=%Rhrc\n", hr));
150 fmtEtc.cfFormat = 0; /* Mark it to not supported. */
151 }
152 }
153 }
154
155 /* Did we find a format that we support? */
156 if (fmtEtc.cfFormat)
157 {
158 LogFlowFunc(("Found supported format %RI16 (%s)\n",
159 fmtEtc.cfFormat, VBoxDnDDataObject::ClipboardFormatToString(fmtEtc.cfFormat)));
160
161 /* Make a copy of the FORMATETC structure so that we later can
162 * use this for comparrison and stuff. */
163 /** Note: The DVTARGETDEVICE member only is a shallow copy for now! */
164 memcpy(&mFormatEtc, &fmtEtc, sizeof(FORMATETC));
165
166 /* Which drop effect we're going to use? */
167 /* Note: pt is not used since we don't need to differentiate within our
168 * proxy window. */
169 *pdwEffect = VBoxDnDDropTarget::GetDropEffect(grfKeyState, *pdwEffect);
170 }
171 else
172 {
173 /* No or incompatible data -- so no drop effect required. */
174 *pdwEffect = DROPEFFECT_NONE;
175
176 switch (hr)
177 {
178 case ERROR_INVALID_FUNCTION:
179 {
180 LogRel(("DnD: Drag and drop format is not supported by VBoxTray\n"));
181
182 /* Enumerate supported source formats. This shouldn't happen too often
183 * on day to day use, but still keep it in here. */
184 IEnumFORMATETC *pEnumFormats;
185 HRESULT hr2 = pDataObject->EnumFormatEtc(DATADIR_GET, &pEnumFormats);
186 if (SUCCEEDED(hr2))
187 {
188 LogRel(("DnD: The following formats were offered to us:\n"));
189
190 FORMATETC curFormatEtc;
191 while (pEnumFormats->Next(1, &curFormatEtc,
192 NULL /* pceltFetched */) == S_OK)
193 {
194 WCHAR wszCfName[128]; /* 128 chars should be enough, rest will be truncated. */
195 hr2 = GetClipboardFormatNameW(curFormatEtc.cfFormat, wszCfName,
196 sizeof(wszCfName) / sizeof(WCHAR));
197 LogRel(("\tcfFormat=%RI16 (%s), tyMed=%RI32, dwAspect=%RI32, strCustomName=%ls, hr=%Rhrc\n",
198 curFormatEtc.cfFormat,
199 VBoxDnDDataObject::ClipboardFormatToString(curFormatEtc.cfFormat),
200 curFormatEtc.tymed,
201 curFormatEtc.dwAspect,
202 wszCfName, hr2));
203 }
204
205 pEnumFormats->Release();
206 }
207
208 break;
209 }
210
211 default:
212 break;
213 }
214 }
215
216 LogFlowFunc(("Returning cfFormat=%RI16, pdwEffect=%ld, hr=%Rhrc\n",
217 fmtEtc.cfFormat, *pdwEffect, hr));
218 return hr;
219}
220
221STDMETHODIMP VBoxDnDDropTarget::DragOver(DWORD grfKeyState, POINTL pt, DWORD *pdwEffect)
222{
223 AssertPtrReturn(pdwEffect, E_INVALIDARG);
224
225#ifdef DEBUG_andy
226 LogFlowFunc(("cfFormat=%RI16, grfKeyState=0x%x, x=%ld, y=%ld\n",
227 mFormatEtc.cfFormat, grfKeyState, pt.x, pt.y));
228#endif
229
230 if (mFormatEtc.cfFormat)
231 {
232 /* Note: pt is not used since we don't need to differentiate within our
233 * proxy window. */
234 *pdwEffect = VBoxDnDDropTarget::GetDropEffect(grfKeyState, *pdwEffect);
235 }
236 else
237 {
238 *pdwEffect = DROPEFFECT_NONE;
239 }
240
241#ifdef DEBUG_andy
242 LogFlowFunc(("Returning *pdwEffect=%ld\n", *pdwEffect));
243#endif
244 return S_OK;
245}
246
247STDMETHODIMP VBoxDnDDropTarget::DragLeave(void)
248{
249#ifdef DEBUG_andy
250 LogFlowFunc(("cfFormat=%RI16\n", mFormatEtc.cfFormat));
251#endif
252
253 if (mpWndParent)
254 mpWndParent->hide();
255
256 return S_OK;
257}
258
259STDMETHODIMP VBoxDnDDropTarget::Drop(IDataObject *pDataObject,
260 DWORD grfKeyState, POINTL pt, DWORD *pdwEffect)
261{
262 AssertPtrReturn(pDataObject, E_INVALIDARG);
263 AssertPtrReturn(pdwEffect, E_INVALIDARG);
264
265#ifdef DEBUG
266 LogFlowFunc(("mFormatEtc.cfFormat=%RI16 (%s), pDataObject=0x%p, grfKeyState=0x%x, x=%ld, y=%ld\n",
267 mFormatEtc.cfFormat, VBoxDnDDataObject::ClipboardFormatToString(mFormatEtc.cfFormat),
268 pDataObject, grfKeyState, pt.x, pt.y));
269#endif
270 HRESULT hr = S_OK;
271
272 if (mFormatEtc.cfFormat) /* Did we get a supported format yet? */
273 {
274 /* Make sure the data object's data format is still the same
275 * as we got it in DragEnter(). */
276 hr = pDataObject->QueryGetData(&mFormatEtc);
277 AssertMsg(SUCCEEDED(hr),
278 ("Data format changed between DragEnter() and Drop(), cfFormat=%RI16 (%s), hr=%Rhrc\n",
279 mFormatEtc.cfFormat, VBoxDnDDataObject::ClipboardFormatToString(mFormatEtc.cfFormat),
280 hr));
281 }
282
283 int rc = VINF_SUCCESS;
284
285 if (SUCCEEDED(hr))
286 {
287 STGMEDIUM stgMed;
288 hr = pDataObject->GetData(&mFormatEtc, &stgMed);
289 if (SUCCEEDED(hr))
290 {
291 /*
292 * First stage: Prepare the access to the storage medium.
293 * For now we only support HGLOBAL stuff.
294 */
295 PVOID pvData = NULL; /** @todo Put this in an own union? */
296
297 switch (mFormatEtc.tymed)
298 {
299 case TYMED_HGLOBAL:
300 pvData = GlobalLock(stgMed.hGlobal);
301 if (!pvData)
302 {
303 LogFlowFunc(("Locking HGLOBAL storage failed with %Rrc\n",
304 RTErrConvertFromWin32(GetLastError())));
305 rc = VERR_INVALID_HANDLE;
306 hr = E_INVALIDARG; /* Set special hr for OLE. */
307 }
308 break;
309
310 default:
311 AssertMsgFailed(("Storage medium type %RI32 supported\n",
312 mFormatEtc.tymed));
313 rc = VERR_NOT_SUPPORTED;
314 hr = DV_E_TYMED; /* Set special hr for OLE. */
315 break;
316 }
317
318 if (RT_SUCCESS(rc))
319 {
320 /* Second stage: Do the actual copying of the data object's data,
321 based on the storage medium type. */
322 switch (mFormatEtc.cfFormat)
323 {
324 case CF_UNICODETEXT:
325 {
326 AssertPtr(pvData);
327 size_t cbSize = GlobalSize(pvData);
328 LogFlowFunc(("CF_UNICODETEXT 0x%p got %zu bytes\n", pvData, cbSize));
329 if (cbSize)
330 {
331 char *pszText = NULL;
332 rc = RTUtf16ToUtf8((PCRTUTF16)pvData, &pszText);
333 if (RT_SUCCESS(rc))
334 {
335 mpvData = (void *)pszText;
336 mcbData = strlen(pszText) + 1; /* Include termination. */
337
338 /* Note: Don't free data of pszText, mpvData now owns it. */
339 }
340 }
341
342 break;
343 }
344
345 case CF_TEXT:
346 {
347 AssertPtr(pvData);
348 size_t cbSize = GlobalSize(pvData);
349 LogFlowFunc(("CF_TEXT 0x%p got %zu bytes\n", pvData, cbSize));
350 if (cbSize)
351 {
352 char *pszText = NULL;
353 rc = RTStrCurrentCPToUtf8(&pszText, (char *)pvData);
354 if (RT_SUCCESS(rc))
355 {
356 mpvData = (void *)pszText;
357 mcbData = strlen(pszText) + 1; /* Include termination. */
358
359 /* Note: Don't free data of pszText, mpvData now owns it. */
360 }
361 }
362
363 break;
364 }
365
366 case CF_HDROP:
367 {
368 AssertPtr(pvData);
369
370 /* Convert to a string list, separated by \r\n. */
371 DROPFILES *pDropFiles = (DROPFILES *)pvData;
372 AssertPtr(pDropFiles);
373 bool fUnicode = RT_BOOL(pDropFiles->fWide);
374
375 /* Get the offset of the file list. */
376 Assert(pDropFiles->pFiles >= sizeof(DROPFILES));
377 /* Note: This is *not* pDropFiles->pFiles! DragQueryFile only
378 * will work with the plain storage medium pointer! */
379 HDROP hDrop = (HDROP)(pvData);
380
381 /* First, get the file count. */
382 /** @todo Does this work on Windows 2000 / NT4? */
383 char *pszFiles = NULL;
384 uint32_t cchFiles = 0;
385 UINT cFiles = DragQueryFile(hDrop, UINT32_MAX /* iFile */,
386 NULL /* lpszFile */, 0 /* cchFile */);
387 LogFlowFunc(("CF_HDROP got %RU16 file(s)\n", cFiles));
388
389 for (UINT i = 0; i < cFiles; i++)
390 {
391 UINT cch = DragQueryFile(hDrop, i /* File index */,
392 NULL /* Query size first */,
393 0 /* cchFile */);
394 Assert(cch);
395
396 if (RT_FAILURE(rc))
397 break;
398
399 char *pszFile = NULL; /* UTF-8 version. */
400 UINT cchFile = 0;
401 if (fUnicode)
402 {
403 /* Allocate enough space (including terminator). */
404 WCHAR *pwszFile = (WCHAR *)RTMemAlloc((cch + 1) * sizeof(WCHAR));
405 if (pwszFile)
406 {
407 cchFile = DragQueryFileW(hDrop, i /* File index */,
408 pwszFile, cch + 1 /* Include terminator */);
409 AssertMsg(cchFile == cch, ("cchCopied (%RU16) does not match cchFile (%RU16)\n",
410 cchFile, cch));
411 rc = RTUtf16ToUtf8(pwszFile, &pszFile);
412 AssertRC(rc);
413
414 RTMemFree(pwszFile);
415 }
416 else
417 rc = VERR_NO_MEMORY;
418 }
419 else /* ANSI */
420 {
421 /* Allocate enough space (including terminator). */
422 pszFile = (char *)RTMemAlloc((cch + 1) * sizeof(char));
423 if (pszFile)
424 {
425 cchFile = DragQueryFileA(hDrop, i /* File index */,
426 pszFile, cchFile + 1 /* Include terminator */);
427 AssertMsg(cchFile == cch, ("cchCopied (%RU16) does not match cchFile (%RU16)\n",
428 cchFile, cch));
429 }
430 else
431 rc = VERR_NO_MEMORY;
432 }
433
434 if (RT_SUCCESS(rc))
435 {
436 LogFlowFunc(("\tFile: %s (cchFile=%RU32)\n",
437 pszFile, cchFile));
438
439 rc = RTStrAAppendExN(&pszFiles, 1 /* cPairs */,
440 pszFile, cchFile);
441 if (RT_SUCCESS(rc))
442 cchFiles += cchFile;
443 }
444
445 if (pszFile)
446 RTStrFree(pszFile);
447
448 if (RT_FAILURE(rc))
449 break;
450
451 /* Add separation between filenames.
452 * Note: Also do this for the last element of the list. */
453 rc = RTStrAAppendExN(&pszFiles, 1 /* cPairs */,
454 "\r\n", 2 /* Bytes */);
455 if (RT_SUCCESS(rc))
456 cchFiles += 2; /* Include \r\n */
457 }
458
459 if (RT_SUCCESS(rc))
460 {
461 cchFiles += 1; /* Add string termination. */
462 uint32_t cbFiles = cchFiles * sizeof(char);
463
464 LogFlowFunc(("cFiles=%u, cchFiles=%RU32, cbFiles=%RU32, pszFiles=0x%p\n",
465 cFiles, cchFiles, cbFiles, pszFiles));
466
467 /* Translate the list into URI elements. */
468 DnDURIList lstURI;
469 rc = lstURI.AppendNativePathsFromList(pszFiles, cbFiles,
470 DNDURILIST_FLAGS_ABSOLUTE_PATHS);
471 if (RT_SUCCESS(rc))
472 {
473 RTCString strRoot = lstURI.RootToString();
474 size_t cbRoot = strRoot.length() + 1; /* Include termination */
475
476 mpvData = RTMemAlloc(cbRoot);
477 if (mpvData)
478 {
479 memcpy(mpvData, strRoot.c_str(), cbRoot);
480 mcbData = cbRoot;
481 }
482 else
483 rc = VERR_NO_MEMORY;
484 }
485 }
486
487 LogFlowFunc(("Building CF_HDROP list rc=%Rrc, pszFiles=0x%p, cFiles=%RU16, cchFiles=%RU32\n",
488 rc, pszFiles, cFiles, cchFiles));
489
490 if (pszFiles)
491 RTStrFree(pszFiles);
492 break;
493 }
494
495 default:
496 /* Note: Should not happen due to the checks done in DragEnter(). */
497 AssertMsgFailed(("Format of type %RI16 (%s) not supported\n",
498 mFormatEtc.cfFormat,
499 VBoxDnDDataObject::ClipboardFormatToString(mFormatEtc.cfFormat)));
500 hr = DV_E_CLIPFORMAT; /* Set special hr for OLE. */
501 break;
502 }
503
504 /*
505 * Third stage: Unlock + release access to the storage medium again.
506 */
507 switch (mFormatEtc.tymed)
508 {
509 case TYMED_HGLOBAL:
510 GlobalUnlock(stgMed.hGlobal);
511 break;
512
513 default:
514 AssertMsgFailed(("Really should not happen -- see init stage!\n"));
515 break;
516 }
517 }
518
519 /* Release storage medium again. */
520 ReleaseStgMedium(&stgMed);
521
522 /* Signal waiters. */
523 mDroppedRc = rc;
524 RTSemEventSignal(hEventDrop);
525 }
526 }
527
528 if (RT_SUCCESS(rc))
529 {
530 /* Note: pt is not used since we don't need to differentiate within our
531 * proxy window. */
532 *pdwEffect = VBoxDnDDropTarget::GetDropEffect(grfKeyState, *pdwEffect);
533 }
534 else
535 *pdwEffect = DROPEFFECT_NONE;
536
537 if (mpWndParent)
538 mpWndParent->hide();
539
540 LogFlowFunc(("Returning with hr=%Rhrc (%Rrc), mFormatEtc.cfFormat=%RI16 (%s), *pdwEffect=%RI32\n",
541 hr, rc, mFormatEtc.cfFormat, VBoxDnDDataObject::ClipboardFormatToString(mFormatEtc.cfFormat),
542 *pdwEffect));
543
544 return hr;
545}
546
547/* static */
548DWORD VBoxDnDDropTarget::GetDropEffect(DWORD grfKeyState, DWORD dwAllowedEffects)
549{
550 DWORD dwEffect = DROPEFFECT_NONE;
551
552 if(grfKeyState & MK_CONTROL)
553 dwEffect = dwAllowedEffects & DROPEFFECT_COPY;
554 else if(grfKeyState & MK_SHIFT)
555 dwEffect = dwAllowedEffects & DROPEFFECT_MOVE;
556
557 /* If there still was no drop effect assigned, check for the handed-in
558 * allowed effects and assign one of them.
559 *
560 * Note: A move action has precendence over a copy action! */
561 if (dwEffect == DROPEFFECT_NONE)
562 {
563 if (dwAllowedEffects & DROPEFFECT_COPY)
564 dwEffect = DROPEFFECT_COPY;
565 if (dwAllowedEffects & DROPEFFECT_MOVE)
566 dwEffect = DROPEFFECT_MOVE;
567 }
568
569#ifdef DEBUG_andy
570 LogFlowFunc(("grfKeyState=0x%x, dwAllowedEffects=0x%x, dwEffect=0x%x\n",
571 grfKeyState, dwAllowedEffects, dwEffect));
572#endif
573 return dwEffect;
574}
575
576void VBoxDnDDropTarget::reset(void)
577{
578 LogFlowFuncEnter();
579
580 if (mpvData)
581 {
582 RTMemFree(mpvData);
583 mpvData = NULL;
584 }
585
586 mcbData = 0;
587 RT_ZERO(mFormatEtc);
588 mFormats = "";
589}
590
591RTCString VBoxDnDDropTarget::Formats(void)
592{
593 return mFormats;
594}
595
596int VBoxDnDDropTarget::WaitForDrop(RTMSINTERVAL msTimeout)
597{
598 LogFlowFunc(("msTimeout=%RU32\n", msTimeout));
599
600 int rc = RTSemEventWait(hEventDrop, msTimeout);
601 if (RT_SUCCESS(rc))
602 rc = mDroppedRc;
603
604 LogFlowFuncLeaveRC(rc);
605 return rc;
606}
607
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