diff --git a/video/out/w32_common.c b/video/out/w32_common.c index 287bb24618..18acaa4423 100644 --- a/video/out/w32_common.c +++ b/video/out/w32_common.c @@ -37,6 +37,7 @@ #include "win_state.h" #include "w32_common.h" #include "win32/displayconfig.h" +#include "win32/droptarget.h" #include "osdep/io.h" #include "osdep/threads.h" #include "osdep/w32_keyboard.h" @@ -148,203 +149,6 @@ struct vo_w32_state { HANDLE avrt_handle; }; -typedef struct tagDropTarget { - IDropTarget iface; - atomic_int refCnt; - DWORD lastEffect; - IDataObject* dataObj; - struct vo_w32_state *w32; -} DropTarget; - -static FORMATETC fmtetc_file = { CF_HDROP, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; -static FORMATETC fmtetc_url = { 0, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; - -static void DropTarget_Destroy(DropTarget* This) -{ - if (This->dataObj != NULL) { - This->dataObj->lpVtbl->Release(This->dataObj); - This->dataObj->lpVtbl = NULL; - } - - talloc_free(This); -} - -static HRESULT STDMETHODCALLTYPE DropTarget_QueryInterface(IDropTarget* This, - REFIID riid, - void** ppvObject) -{ - if (!IsEqualGUID(riid, &IID_IUnknown) || - !IsEqualGUID(riid, &IID_IDataObject)) { - *ppvObject = NULL; - return E_NOINTERFACE; - } - - *ppvObject = This; - This->lpVtbl->AddRef(This); - return S_OK; -} - -static ULONG STDMETHODCALLTYPE DropTarget_AddRef(IDropTarget* This) -{ - DropTarget* t = (DropTarget*)This; - return atomic_fetch_add(&t->refCnt, 1) + 1; -} - -static ULONG STDMETHODCALLTYPE DropTarget_Release(IDropTarget* This) -{ - DropTarget* t = (DropTarget*)This; - ULONG cRef = atomic_fetch_add(&t->refCnt, -1) - 1; - - if (cRef == 0) { - DropTarget_Destroy(t); - } - - return cRef; -} - -static HRESULT STDMETHODCALLTYPE DropTarget_DragEnter(IDropTarget* This, - IDataObject* pDataObj, - DWORD grfKeyState, - POINTL pt, - DWORD* pdwEffect) -{ - DropTarget* t = (DropTarget*)This; - - pDataObj->lpVtbl->AddRef(pDataObj); - if (pDataObj->lpVtbl->QueryGetData(pDataObj, &fmtetc_file) != S_OK && - pDataObj->lpVtbl->QueryGetData(pDataObj, &fmtetc_url) != S_OK) { - - *pdwEffect = DROPEFFECT_NONE; - } - - if (t->dataObj != NULL) { - t->dataObj->lpVtbl->Release(t->dataObj); - } - - t->dataObj = pDataObj; - t->lastEffect = *pdwEffect; - return S_OK; -} - -static HRESULT STDMETHODCALLTYPE DropTarget_DragOver(IDropTarget* This, - DWORD grfKeyState, - POINTL pt, - DWORD* pdwEffect) -{ - DropTarget* t = (DropTarget*)This; - - *pdwEffect = t->lastEffect; - return S_OK; -} - -static HRESULT STDMETHODCALLTYPE DropTarget_DragLeave(IDropTarget* This) -{ - DropTarget* t = (DropTarget*)This; - - if (t->dataObj != NULL) { - t->dataObj->lpVtbl->Release(t->dataObj); - t->dataObj = NULL; - } - - return S_OK; -} - -static HRESULT STDMETHODCALLTYPE DropTarget_Drop(IDropTarget* This, - IDataObject* pDataObj, - DWORD grfKeyState, POINTL pt, - DWORD* pdwEffect) -{ - DropTarget* t = (DropTarget*)This; - - STGMEDIUM medium; - - if (t->dataObj != NULL) { - t->dataObj->lpVtbl->Release(t->dataObj); - t->dataObj = NULL; - } - - enum mp_dnd_action action = (grfKeyState & MK_SHIFT) ? DND_APPEND : DND_REPLACE; - - pDataObj->lpVtbl->AddRef(pDataObj); - - if (pDataObj->lpVtbl->GetData(pDataObj, &fmtetc_file, &medium) == S_OK) { - if (GlobalLock(medium.hGlobal) != NULL) { - HDROP hDrop = (HDROP)medium.hGlobal; - - UINT numFiles = DragQueryFileW(hDrop, 0xFFFFFFFF, NULL, 0); - char** files = talloc_zero_array(NULL, char*, numFiles); - - UINT nrecvd_files = 0; - for (UINT i = 0; i < numFiles; i++) { - UINT len = DragQueryFileW(hDrop, i, NULL, 0); - wchar_t* buf = talloc_array(NULL, wchar_t, len + 1); - - if (DragQueryFileW(hDrop, i, buf, len + 1) == len) { - char* fname = mp_to_utf8(files, buf); - files[nrecvd_files++] = fname; - - MP_VERBOSE(t->w32, "received dropped file: %s\n", - fname); - } else { - MP_ERR(t->w32, "error getting dropped file name\n"); - } - - talloc_free(buf); - } - - GlobalUnlock(medium.hGlobal); - mp_event_drop_files(t->w32->input_ctx, nrecvd_files, files, - action); - - talloc_free(files); - } - - ReleaseStgMedium(&medium); - } else if (pDataObj->lpVtbl->GetData(pDataObj, - &fmtetc_url, &medium) == S_OK) { - // get the URL encoded in US-ASCII - wchar_t* wurl = GlobalLock(medium.hGlobal); - if (wurl != NULL) { - char *url = mp_to_utf8(NULL, wurl); - if (mp_event_drop_mime_data(t->w32->input_ctx, "text/uri-list", - bstr0(url), action) > 0) { - MP_VERBOSE(t->w32, "received dropped URL: %s\n", url); - } else { - MP_ERR(t->w32, "error getting dropped URL\n"); - } - - talloc_free(url); - GlobalUnlock(medium.hGlobal); - } - - ReleaseStgMedium(&medium); - } - else { - t->lastEffect = DROPEFFECT_NONE; - } - - pDataObj->lpVtbl->Release(pDataObj); - *pdwEffect = t->lastEffect; - return S_OK; -} - - -static void DropTarget_Init(DropTarget* This, struct vo_w32_state *w32) -{ - IDropTargetVtbl* vtbl = talloc(This, IDropTargetVtbl); - *vtbl = (IDropTargetVtbl){ - DropTarget_QueryInterface, DropTarget_AddRef, DropTarget_Release, - DropTarget_DragEnter, DropTarget_DragOver, DropTarget_DragLeave, - DropTarget_Drop - }; - - This->iface.lpVtbl = vtbl; - atomic_store(&This->refCnt, 0); - This->lastEffect = 0; - This->dataObj = NULL; - This->w32 = w32; -} - static void add_window_borders(HWND hwnd, RECT *rc) { AdjustWindowRect(rc, GetWindowLongPtrW(hwnd, GWL_STYLE), 0); @@ -1536,10 +1340,8 @@ static void *gui_thread(void *ptr) if (SUCCEEDED(OleInitialize(NULL))) { ole_ok = true; - fmtetc_url.cfFormat = (CLIPFORMAT)RegisterClipboardFormat(TEXT("UniformResourceLocatorW")); - DropTarget* dropTarget = talloc(NULL, DropTarget); - DropTarget_Init(dropTarget, w32); - RegisterDragDrop(w32->window, &dropTarget->iface); + IDropTarget *dt = mp_w32_droptarget_create(w32->log, w32->input_ctx); + RegisterDragDrop(w32->window, dt); // ITaskbarList2 has the MarkFullscreenWindow method, which is used to // make sure the taskbar is hidden when mpv goes fullscreen diff --git a/video/out/win32/droptarget.c b/video/out/win32/droptarget.c new file mode 100644 index 0000000000..8da484439f --- /dev/null +++ b/video/out/win32/droptarget.c @@ -0,0 +1,233 @@ +/* + * This file is part of mpv. + * + * mpv is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2.1 of the License, or (at your option) any later version. + * + * mpv is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with mpv. If not, see . + */ + +#include +#include +#include + +#include "common/msg.h" +#include "common/common.h" +#include "input/input.h" +#include "input/event.h" +#include "osdep/atomic.h" +#include "osdep/io.h" +#include "mpv_talloc.h" + +#include "droptarget.h" + +typedef struct tagDropTarget { + IDropTarget iface; + atomic_int refCnt; + DWORD lastEffect; + IDataObject* dataObj; + struct mp_log *log; + struct input_ctx *input_ctx; +} DropTarget; + +static FORMATETC fmtetc_file = { CF_HDROP, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; +static FORMATETC fmtetc_url = { 0, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; + +static void DropTarget_Destroy(DropTarget* This) +{ + if (This->dataObj != NULL) { + This->dataObj->lpVtbl->Release(This->dataObj); + This->dataObj->lpVtbl = NULL; + } + + talloc_free(This); +} + +static HRESULT STDMETHODCALLTYPE DropTarget_QueryInterface(IDropTarget* This, + REFIID riid, + void** ppvObject) +{ + if (!IsEqualGUID(riid, &IID_IUnknown) || + !IsEqualGUID(riid, &IID_IDataObject)) { + *ppvObject = NULL; + return E_NOINTERFACE; + } + + *ppvObject = This; + This->lpVtbl->AddRef(This); + return S_OK; +} + +static ULONG STDMETHODCALLTYPE DropTarget_AddRef(IDropTarget* This) +{ + DropTarget* t = (DropTarget*)This; + return atomic_fetch_add(&t->refCnt, 1) + 1; +} + +static ULONG STDMETHODCALLTYPE DropTarget_Release(IDropTarget* This) +{ + DropTarget* t = (DropTarget*)This; + ULONG cRef = atomic_fetch_add(&t->refCnt, -1) - 1; + + if (cRef == 0) { + DropTarget_Destroy(t); + } + + return cRef; +} + +static HRESULT STDMETHODCALLTYPE DropTarget_DragEnter(IDropTarget* This, + IDataObject* pDataObj, + DWORD grfKeyState, + POINTL pt, + DWORD* pdwEffect) +{ + DropTarget* t = (DropTarget*)This; + + pDataObj->lpVtbl->AddRef(pDataObj); + if (pDataObj->lpVtbl->QueryGetData(pDataObj, &fmtetc_file) != S_OK && + pDataObj->lpVtbl->QueryGetData(pDataObj, &fmtetc_url) != S_OK) { + + *pdwEffect = DROPEFFECT_NONE; + } + + if (t->dataObj != NULL) { + t->dataObj->lpVtbl->Release(t->dataObj); + } + + t->dataObj = pDataObj; + t->lastEffect = *pdwEffect; + return S_OK; +} + +static HRESULT STDMETHODCALLTYPE DropTarget_DragOver(IDropTarget* This, + DWORD grfKeyState, + POINTL pt, + DWORD* pdwEffect) +{ + DropTarget* t = (DropTarget*)This; + + *pdwEffect = t->lastEffect; + return S_OK; +} + +static HRESULT STDMETHODCALLTYPE DropTarget_DragLeave(IDropTarget* This) +{ + DropTarget* t = (DropTarget*)This; + + if (t->dataObj != NULL) { + t->dataObj->lpVtbl->Release(t->dataObj); + t->dataObj = NULL; + } + + return S_OK; +} + +static HRESULT STDMETHODCALLTYPE DropTarget_Drop(IDropTarget* This, + IDataObject* pDataObj, + DWORD grfKeyState, POINTL pt, + DWORD* pdwEffect) +{ + DropTarget* t = (DropTarget*)This; + + STGMEDIUM medium; + + if (t->dataObj != NULL) { + t->dataObj->lpVtbl->Release(t->dataObj); + t->dataObj = NULL; + } + + enum mp_dnd_action action = (grfKeyState & MK_SHIFT) ? DND_APPEND : DND_REPLACE; + + pDataObj->lpVtbl->AddRef(pDataObj); + + if (pDataObj->lpVtbl->GetData(pDataObj, &fmtetc_file, &medium) == S_OK) { + if (GlobalLock(medium.hGlobal) != NULL) { + HDROP hDrop = (HDROP)medium.hGlobal; + + UINT numFiles = DragQueryFileW(hDrop, 0xFFFFFFFF, NULL, 0); + char** files = talloc_zero_array(NULL, char*, numFiles); + + UINT nrecvd_files = 0; + for (UINT i = 0; i < numFiles; i++) { + UINT len = DragQueryFileW(hDrop, i, NULL, 0); + wchar_t* buf = talloc_array(NULL, wchar_t, len + 1); + + if (DragQueryFileW(hDrop, i, buf, len + 1) == len) { + char* fname = mp_to_utf8(files, buf); + files[nrecvd_files++] = fname; + + MP_VERBOSE(t, "received dropped file: %s\n", fname); + } else { + MP_ERR(t, "error getting dropped file name\n"); + } + + talloc_free(buf); + } + + GlobalUnlock(medium.hGlobal); + mp_event_drop_files(t->input_ctx, nrecvd_files, files, + action); + + talloc_free(files); + } + + ReleaseStgMedium(&medium); + } else if (pDataObj->lpVtbl->GetData(pDataObj, + &fmtetc_url, &medium) == S_OK) { + // get the URL encoded in US-ASCII + wchar_t* wurl = GlobalLock(medium.hGlobal); + if (wurl != NULL) { + char *url = mp_to_utf8(NULL, wurl); + if (mp_event_drop_mime_data(t->input_ctx, "text/uri-list", + bstr0(url), action) > 0) { + MP_VERBOSE(t, "received dropped URL: %s\n", url); + } else { + MP_ERR(t, "error getting dropped URL\n"); + } + + talloc_free(url); + GlobalUnlock(medium.hGlobal); + } + + ReleaseStgMedium(&medium); + } + else { + t->lastEffect = DROPEFFECT_NONE; + } + + pDataObj->lpVtbl->Release(pDataObj); + *pdwEffect = t->lastEffect; + return S_OK; +} + +IDropTarget *mp_w32_droptarget_create(struct mp_log *log, + struct input_ctx *input_ctx) +{ + DropTarget* dropTarget = talloc(NULL, DropTarget); + IDropTargetVtbl* vtbl = talloc(dropTarget, IDropTargetVtbl); + *vtbl = (IDropTargetVtbl){ + DropTarget_QueryInterface, DropTarget_AddRef, DropTarget_Release, + DropTarget_DragEnter, DropTarget_DragOver, DropTarget_DragLeave, + DropTarget_Drop + }; + + fmtetc_url.cfFormat = (CLIPFORMAT)RegisterClipboardFormat(TEXT("UniformResourceLocatorW")); + + dropTarget->iface.lpVtbl = vtbl; + atomic_store(&dropTarget->refCnt, 0); + dropTarget->lastEffect = 0; + dropTarget->dataObj = NULL; + dropTarget->log = mp_log_new(dropTarget, log, "droptarget"); + dropTarget->input_ctx = input_ctx; + + return &dropTarget->iface; +} diff --git a/video/out/win32/droptarget.h b/video/out/win32/droptarget.h new file mode 100644 index 0000000000..53eb5ac998 --- /dev/null +++ b/video/out/win32/droptarget.h @@ -0,0 +1,33 @@ +/* + * This file is part of mpv. + * + * mpv is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2.1 of the License, or (at your option) any later version. + * + * mpv is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with mpv. If not, see . + */ + +#ifndef MP_WIN32_DROPTARGET_H_ +#define MP_WIN32_DROPTARGET_H_ + +#include +#include +#include + +#include "input/input.h" +#include "common/msg.h" +#include "common/common.h" + +// Create a IDropTarget implementation that sends dropped files to input_ctx +IDropTarget *mp_w32_droptarget_create(struct mp_log *log, + struct input_ctx *input_ctx); + +#endif diff --git a/wscript_build.py b/wscript_build.py index 4f1d3440c0..7d112782a5 100644 --- a/wscript_build.py +++ b/wscript_build.py @@ -398,6 +398,7 @@ def build(ctx): ( "video/out/vo_xv.c", "xv" ), ( "video/out/w32_common.c", "win32" ), ( "video/out/win32/displayconfig.c", "win32" ), + ( "video/out/win32/droptarget.c", "win32" ), ( "video/out/win32/exclusive_hack.c", "gl-win32" ), ( "video/out/wayland_common.c", "wayland" ), ( "video/out/wayland/buffer.c", "wayland" ),