VirtualBox

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

Last change on this file since 58611 was 58336, checked in by vboxsync, 9 years ago

DnD/VBoxTray: Not needed anymore.

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