summaryrefslogtreecommitdiff
path: root/src/fl_dnd_win32.cxx
diff options
context:
space:
mode:
authormaxim nikonov <maxim.nikonov@hqo.co>2026-02-05 15:21:34 +0500
committermaxim nikonov <maxim.nikonov@hqo.co>2026-02-05 15:21:34 +0500
commitdb214d1145e46d527a46d1fc2519548d2c4d23f1 (patch)
treecf0fd9922e4d54f6beb63888f9b28c8e2a787bdf /src/fl_dnd_win32.cxx
parent75fc94d6c71fe686f6dde5b41ad91cba2f6bdd6f (diff)
wip: fork
Diffstat (limited to 'src/fl_dnd_win32.cxx')
-rw-r--r--src/fl_dnd_win32.cxx559
1 files changed, 0 insertions, 559 deletions
diff --git a/src/fl_dnd_win32.cxx b/src/fl_dnd_win32.cxx
deleted file mode 100644
index abaa588cc..000000000
--- a/src/fl_dnd_win32.cxx
+++ /dev/null
@@ -1,559 +0,0 @@
-//
-// Drag & Drop code for the Fast Light Tool Kit (FLTK).
-//
-// Copyright 1998-2023 by Bill Spitzak and others.
-//
-// This library is free software. Distribution and use rights are outlined in
-// the file "COPYING" which should have been included with this file. If this
-// file is missing or damaged, see the license at:
-//
-// https://www.fltk.org/COPYING.php
-//
-// Please see the following page on how to report bugs and issues:
-//
-// https://www.fltk.org/bugs.php
-//
-
-// Note: this file contains platform specific code and will therefore
-// not be processed by doxygen (see Doxyfile.in).
-
-// This file contains Windows-specific code for FLTK which is always linked
-// in. Search other files for "_WIN32" or filenames ending in _win32.cxx
-// for other system-specific code.
-
-#include <FL/Fl.H>
-#include <FL/platform.H>
-#include <FL/Fl_Window.H>
-#include "Fl_Window_Driver.H"
-#include <FL/fl_utf8.h>
-#include "drivers/WinAPI/Fl_WinAPI_Screen_Driver.H"
-#include "flstring.h"
-#include <stdio.h>
-#include <stdlib.h>
-#include <sys/types.h>
-#include <objidl.h>
-#include <time.h>
-#if defined(__CYGWIN__)
-#include <sys/time.h>
-#include <unistd.h>
-#endif
-
-extern char *fl_selection_buffer[2];
-extern int fl_selection_length[2];
-extern int fl_selection_buffer_length[2];
-extern char fl_i_own_selection[2];
-extern char *fl_locale2utf8(const char *s, UINT codepage = 0);
-extern unsigned int fl_codepage;
-
-Fl_Window *fl_dnd_target_window = 0;
-
-#include <ole2.h>
-#include <shellapi.h>
-#include <shlobj.h>
-
-
-/**
- subclass the IDropTarget to receive data from DnD operations
- */
-class FLDropTarget : public IDropTarget
-{
- DWORD m_cRefCount; // for "statistics" only (issue #569)
- DWORD lastEffect;
- int px, py;
-public:
- FLDropTarget() : m_cRefCount(0) { } // initialize
- virtual ~FLDropTarget() { }
- HRESULT STDMETHODCALLTYPE QueryInterface( REFIID riid, LPVOID *ppvObject ) FL_OVERRIDE {
- if (IID_IUnknown==riid || IID_IDropTarget==riid)
- {
- *ppvObject=this;
- ((LPUNKNOWN)*ppvObject)->AddRef();
- return S_OK;
- }
- *ppvObject = NULL;
- return E_NOINTERFACE;
- }
- ULONG STDMETHODCALLTYPE AddRef() FL_OVERRIDE { return ++m_cRefCount; }
- ULONG STDMETHODCALLTYPE Release() FL_OVERRIDE {
- long nTemp;
- nTemp = --m_cRefCount;
- // this is a static object, do not 'delete this' (issue #569)
- return nTemp;
- }
- HRESULT STDMETHODCALLTYPE DragEnter( IDataObject *pDataObj, DWORD /*grfKeyState*/, POINTL pt, DWORD *pdwEffect) FL_OVERRIDE {
- if( !pDataObj ) return E_INVALIDARG;
- // set e_modifiers here from grfKeyState, set e_x and e_root_x
- // check if FLTK handles this drag and return if it can't (i.e. BMP drag without filename)
- POINT ppt;
- Fl::e_x_root = ppt.x = pt.x;
- Fl::e_y_root = ppt.y = pt.y;
- HWND hWnd = WindowFromPoint( ppt );
- Fl_Window *target = fl_find( hWnd );
- if (target) {
- float s = Fl::screen_driver()->scale(Fl_Window_Driver::driver(target)->screen_num());
- Fl::e_x_root = int(Fl::e_x_root / s);
- Fl::e_y_root = int(Fl::e_y_root / s);
- Fl::e_x = Fl::e_x_root-target->x();
- Fl::e_y = Fl::e_y_root-target->y();
- }
- fl_dnd_target_window = target;
- px = pt.x; py = pt.y;
- if (fillCurrentDragData(pDataObj)) {
- // FLTK has no mechanism yet for the different drop effects, so we allow move and copy
- if ( target && Fl::handle( FL_DND_ENTER, target ) )
- *pdwEffect = DROPEFFECT_MOVE|DROPEFFECT_COPY; //|DROPEFFECT_LINK;
- else
- *pdwEffect = DROPEFFECT_NONE;
- } else {
- *pdwEffect = DROPEFFECT_NONE;
- }
- lastEffect = *pdwEffect;
- return S_OK;
- }
- HRESULT STDMETHODCALLTYPE DragOver( DWORD /*grfKeyState*/, POINTL pt, DWORD *pdwEffect) FL_OVERRIDE {
- if ( px==pt.x && py==pt.y )
- {
- *pdwEffect = lastEffect;
- return S_OK;
- }
- if ( !fl_dnd_target_window )
- {
- *pdwEffect = lastEffect = DROPEFFECT_NONE;
- return S_OK;
- }
- // set e_modifiers here from grfKeyState, set e_x and e_root_x
- Fl::e_x_root = pt.x;
- Fl::e_y_root = pt.y;
- if (fl_dnd_target_window) {
- float s = Fl::screen_driver()->scale(Fl_Window_Driver::driver(fl_dnd_target_window)->screen_num());
- Fl::e_x_root = int(Fl::e_x_root /s);
- Fl::e_y_root = int(Fl::e_y_root /s);
- Fl::e_x = Fl::e_x_root-fl_dnd_target_window->x();
- Fl::e_y = Fl::e_y_root-fl_dnd_target_window->y();
- }
- if (fillCurrentDragData(0)) {
- // Fl_Group will change DND_DRAG into DND_ENTER and DND_LEAVE if needed
- if ( Fl::handle( FL_DND_DRAG, fl_dnd_target_window ) )
- *pdwEffect = DROPEFFECT_MOVE|DROPEFFECT_COPY; //|DROPEFFECT_LINK;
- else
- *pdwEffect = DROPEFFECT_NONE;
- } else {
- *pdwEffect = DROPEFFECT_NONE;
- }
- px = pt.x; py = pt.y;
- lastEffect = *pdwEffect;
- // show insert position if dnd'ing in the same window/process (STR #3209)
- Fl::flush();
- return S_OK;
- }
- HRESULT STDMETHODCALLTYPE DragLeave() FL_OVERRIDE {
- if ( fl_dnd_target_window && fillCurrentDragData(0))
- {
- Fl::handle( FL_DND_LEAVE, fl_dnd_target_window );
- fl_dnd_target_window = 0;
- clearCurrentDragData();
- }
- return S_OK;
- }
- HRESULT STDMETHODCALLTYPE Drop( IDataObject *data, DWORD /*grfKeyState*/, POINTL pt, DWORD* /*pdwEffect*/) FL_OVERRIDE {
- if ( !fl_dnd_target_window )
- return S_OK;
- Fl_Window *target = fl_dnd_target_window;
- fl_dnd_target_window = 0;
- Fl::e_x_root = pt.x;
- Fl::e_y_root = pt.y;
- float s = Fl::screen_driver()->scale(Fl_Window_Driver::driver(target)->screen_num());
- Fl::e_x_root = int(Fl::e_x_root / s);
- Fl::e_y_root = int(Fl::e_y_root / s);
- if (target) {
- Fl::e_x = Fl::e_x_root-target->x();
- Fl::e_y = Fl::e_y_root-target->y();
- }
- // tell FLTK that the user released an object on this widget
- if ( !Fl::handle( FL_DND_RELEASE, target ) )
- return S_OK;
-
- Fl_Widget *w = target;
- while (w->parent()) w = w->window();
- HWND hwnd = fl_xid( (Fl_Window*)w );
- if (fillCurrentDragData(data)) {
- int old_event = Fl::e_number;
- char *a, *b;
- a = b = currDragData;
- while (*a) { // strip the CRLF pairs
- if (*a == '\r' && a[1] == '\n') a++;
- else *b++ = *a++;
- }
- *b = 0;
- Fl::e_text = currDragData;
- Fl::e_length = (int) (b - currDragData);
- Fl::belowmouse()->handle(Fl::e_number = FL_PASTE); // e_text will be invalid after this call
- Fl::e_number = old_event;
- SetForegroundWindow( hwnd );
- clearCurrentDragData();
- return S_OK;
- }
- return S_OK;
- }
-private:
-
- static IDataObject *currDragRef;
- static char *currDragData;
- static int currDragSize;
- static char currDragResult;
-
- static void clearCurrentDragData() {
- currDragRef = 0;
- if (currDragData) free(currDragData);
- currDragData = 0;
- currDragSize = 0;
- currDragResult = 0;
- }
- static char fillCurrentDragData(IDataObject *data) {
- // shortcut through this whole procedure if there is no fresh data
- if (!data)
- return currDragResult;
- // shortcut through this whole procedure if this is still the same drag event
- // (* this is safe, because 'currDragRef' is cleared on Leave and Drop events)
- if (data==currDragRef)
- return currDragResult;
-
- // clear currDrag* for a new drag event
- clearCurrentDragData();
-
- currDragRef = data;
- // fill currDrag* with UTF-8 data, if available
- FORMATETC fmt = { 0 };
- STGMEDIUM medium = { 0 };
- fmt.tymed = TYMED_HGLOBAL;
- fmt.dwAspect = DVASPECT_CONTENT;
- fmt.lindex = -1;
- fmt.cfFormat = CF_UNICODETEXT;
- // if it is UNICODE text, return a UTF-8-converted copy of it
- if ( data->GetData( &fmt, &medium )==S_OK )
- {
- void *stuff = GlobalLock( medium.hGlobal );
- unsigned srclen = 0;
- const wchar_t *wstuff = (const wchar_t *)stuff;
- while (*wstuff++) srclen++;
- wstuff = (const wchar_t *)stuff;
- unsigned utf8len = fl_utf8fromwc(NULL, 0, wstuff, srclen);
- currDragSize = utf8len;
- currDragData = (char*)malloc(utf8len + 1);
- fl_utf8fromwc(currDragData, currDragSize+1, wstuff, srclen+1); // include null-byte
- GlobalUnlock( medium.hGlobal );
- ReleaseStgMedium( &medium );
- currDragResult = 1;
- return currDragResult;
- }
- fmt.cfFormat = CF_TEXT;
- // if it is CP1252 text, return a UTF-8-converted copy of it
- if ( data->GetData( &fmt, &medium )==S_OK )
- {
- int len;
- char *p, *q, *last;
- unsigned u;
- void *stuff = GlobalLock( medium.hGlobal );
- currDragData = (char*)malloc(3 * strlen((char*)stuff) + 10);
- p = (char*)stuff;
- last = p + strlen(p);
- q = currDragData;
- while (p < last) {
- u = fl_utf8decode(p, last, &len);
- p += len;
- len = fl_utf8encode(u, q);
- q += len;
- }
- *q = 0;
- currDragSize = (int) (q - currDragData);
- currDragData = (char*)realloc(currDragData, currDragSize + 1);
- GlobalUnlock( medium.hGlobal );
- ReleaseStgMedium( &medium );
- currDragResult = 1;
- return currDragResult;
- }
- // else fill currDrag* with filenames, if possible
- memset(&fmt, 0, sizeof(fmt));
- fmt.tymed = TYMED_HGLOBAL;
- fmt.dwAspect = DVASPECT_CONTENT;
- fmt.lindex = -1;
- fmt.cfFormat = CF_HDROP;
- // if it is a pathname list, send an FL_PASTE with a \n separated list of filepaths
- if ( data->GetData( &fmt, &medium )==S_OK )
- {
- HDROP hdrop = (HDROP)medium.hGlobal;
- int i, n, nn = 0, nf = DragQueryFileW( hdrop, (UINT)-1, 0, 0 );
- for ( i=0; i<nf; i++ ) nn += DragQueryFileW( hdrop, i, 0, 0 );
- nn += nf;
- wchar_t *dst = (wchar_t *)malloc(nn * sizeof(wchar_t));
- wchar_t *bu = dst;
- for ( i=0; i<nf; i++ ) {
- n = DragQueryFileW( hdrop, i, (WCHAR*)dst, nn );
- dst += n;
- if ( i<nf-1 ) {
- *dst++ = L'\n';
- }
- }
- *dst=0;
-
- currDragData = (char*) malloc(nn * 5 + 1);
-// Fl::e_length = fl_unicode2utf(bu, nn, Fl::e_text);
- currDragSize = fl_utf8fromwc(currDragData, (nn*5+1), bu, nn);
- currDragData[currDragSize] = 0;
- free(bu);
-
-// Fl::belowmouse()->handle(FL_DROP);
-// free( Fl::e_text );
- ReleaseStgMedium( &medium );
- currDragResult = 1;
- return currDragResult;
- }
- currDragResult = 0;
- return currDragResult;
- }
-} flDropTarget;
-
-IDropTarget *flIDropTarget = &flDropTarget;
-
-IDataObject *FLDropTarget::currDragRef = 0;
-char *FLDropTarget::currDragData = 0;
-int FLDropTarget::currDragSize = 0;
-char FLDropTarget::currDragResult = 0;
-
-/**
- this class is needed to allow FLTK apps to be a DnD source
- */
-class FLDropSource : public IDropSource
-{
- DWORD m_cRefCount;
-public:
- FLDropSource() { m_cRefCount = 0; }
- virtual ~FLDropSource() { }
- HRESULT STDMETHODCALLTYPE QueryInterface( REFIID riid, LPVOID *ppvObject ) FL_OVERRIDE {
- if (IID_IUnknown==riid || IID_IDropSource==riid)
- {
- *ppvObject=this;
- ((LPUNKNOWN)*ppvObject)->AddRef();
- return S_OK;
- }
- *ppvObject = NULL;
- return E_NOINTERFACE;
- }
- ULONG STDMETHODCALLTYPE AddRef() FL_OVERRIDE { return ++m_cRefCount; }
- ULONG STDMETHODCALLTYPE Release() FL_OVERRIDE {
- long nTemp;
- nTemp = --m_cRefCount;
- if(nTemp==0)
- delete this;
- return nTemp;
- }
- STDMETHODIMP GiveFeedback( DWORD ) FL_OVERRIDE { return DRAGDROP_S_USEDEFAULTCURSORS; }
- STDMETHODIMP QueryContinueDrag( BOOL esc, DWORD keyState ) FL_OVERRIDE {
- if ( esc )
- return DRAGDROP_S_CANCEL;
- if ( !(keyState & (MK_LBUTTON|MK_MBUTTON|MK_RBUTTON)) )
- return DRAGDROP_S_DROP;
- return S_OK;
- }
-};
-class FLEnum : public IEnumFORMATETC
-{
-public:
- int n;
- LONG m_lRefCount;
-
- ULONG __stdcall AddRef(void) FL_OVERRIDE {
- return InterlockedIncrement(&m_lRefCount);
- }
-
- ULONG __stdcall Release(void) FL_OVERRIDE {
- LONG count = InterlockedDecrement(&m_lRefCount);
- if(count == 0) {
- delete this;
- return 0;
- } else {
- return count;
- }
- }
-
-
- HRESULT __stdcall QueryInterface(REFIID iid, void **ppvObject) FL_OVERRIDE {
- if(iid == IID_IEnumFORMATETC || iid == IID_IUnknown) {
- AddRef();
- *ppvObject = this;
- return S_OK;
- } else {
- *ppvObject = 0;
- return E_NOINTERFACE;
- }
- }
-
- HRESULT __stdcall Next(ULONG celt, FORMATETC * rgelt, ULONG *pceltFetched) FL_OVERRIDE {
- if (n > 0) return S_FALSE;
- for (ULONG i = 0; i < celt; i++) {
- n++;
- rgelt->cfFormat = CF_HDROP;
- rgelt->dwAspect = DVASPECT_CONTENT;
- rgelt->lindex = -1;
- rgelt->ptd = NULL;
- rgelt->tymed = TYMED_HGLOBAL;
- }
- if (pceltFetched) *pceltFetched = celt;
- return S_OK;
- }
-
- HRESULT __stdcall Skip(ULONG celt) FL_OVERRIDE {
- n += celt;
- return (n == 0) ? S_OK : S_FALSE;
- }
-
- HRESULT __stdcall Reset(void) FL_OVERRIDE {
- n = 0;
- return S_OK;
- }
-
- HRESULT __stdcall Clone(IEnumFORMATETC **ppenum) FL_OVERRIDE {
- *ppenum = new FLEnum();
- return S_OK;
- }
-
- FLEnum(void) {
- m_lRefCount = 1;
- n = 0;
- }
-
- virtual ~FLEnum(void) {
- n = 0;
- }
-};
-
-
-/**
- This is the actual object that FLTK can drop somewhere.
-
- The implementation is minimal, but it should work with all decent Win32 drop targets
-*/
-class FLDataObject : public IDataObject
-{
- DWORD m_cRefCount;
- //FLEnum *m_EnumF;
-public:
- FLDataObject() { m_cRefCount = 1; }// m_EnumF = new FLEnum();}
- virtual ~FLDataObject() { }
- HRESULT STDMETHODCALLTYPE QueryInterface( REFIID riid, LPVOID *ppvObject ) FL_OVERRIDE {
- if (IID_IUnknown==riid || IID_IDataObject==riid)
- {
- *ppvObject=this;
- ((LPUNKNOWN)*ppvObject)->AddRef();
- return S_OK;
- }
- *ppvObject = NULL;
- return E_NOINTERFACE;
- }
- ULONG STDMETHODCALLTYPE AddRef() FL_OVERRIDE { return ++m_cRefCount; }
- ULONG STDMETHODCALLTYPE Release() FL_OVERRIDE {
- long nTemp;
- nTemp = --m_cRefCount;
- if(nTemp==0)
- delete this;
- return nTemp;
- }
- // GetData currently allows UNICODE text through Global Memory only
- HRESULT STDMETHODCALLTYPE GetData( FORMATETC *pformatetcIn, STGMEDIUM *pmedium ) FL_OVERRIDE {
- if ((pformatetcIn->dwAspect & DVASPECT_CONTENT) &&
- (pformatetcIn->tymed & TYMED_HGLOBAL) &&
- (pformatetcIn->cfFormat == CF_UNICODETEXT))
- {
- int utf16_len = fl_utf8toUtf16(fl_selection_buffer[0], fl_selection_length[0], 0, 0);
- HGLOBAL gh = GlobalAlloc( GHND, utf16_len * 2 + 2 );
- char *pMem = (char*)GlobalLock( gh );
- fl_utf8toUtf16(fl_selection_buffer[0], fl_selection_length[0], (unsigned short*)pMem, utf16_len + 1);
-// HGLOBAL gh = GlobalAlloc( GHND| GMEM_SHARE,
-// (fl_selection_length[0]+4) * sizeof(short)
-// + sizeof(DROPFILES));
-// unsigned char *pMem = (unsigned char*)GlobalLock( gh );
-// if (!pMem) {
-// GlobalFree(gh);
-// return DV_E_FORMATETC;
-// }
-// DROPFILES *df =(DROPFILES*) pMem;
-// int l;
-// df->pFiles = sizeof(DROPFILES);
-// df->pt.x = 0;
-// df->pt.y = 0;
-// df->fNC = FALSE;
-// for (int i = 0; i < fl_selection_length[0]; i++) {
-// if (fl_selection_buffer[0][i] == '\n') {
-// fl_selection_buffer[0][i] = '\0';
-// }
-// }
-//
-// df->fWide = TRUE;
-// l = fl_utf2unicode((unsigned char*)fl_selection_buffer[0],
-// fl_selection_length[0], (wchar_t*)(((char*)pMem)
-// + sizeof(DROPFILES)));
-//
-// pMem[l * sizeof(WCHAR) + sizeof(DROPFILES)] = 0;
-// pMem[l * sizeof(WCHAR) + 1 + sizeof(DROPFILES)] = 0;
-// pMem[l * sizeof(WCHAR) + 2 + sizeof(DROPFILES)] = 0;
-// pMem[l * sizeof(WCHAR) + 3 + sizeof(DROPFILES)] = 0;
- pmedium->tymed = TYMED_HGLOBAL;
- pmedium->hGlobal = gh;
- pmedium->pUnkForRelease = NULL;
- GlobalUnlock( gh );
- return S_OK;
- }
- return DV_E_FORMATETC;
- }
- HRESULT STDMETHODCALLTYPE QueryGetData( FORMATETC *pformatetc ) FL_OVERRIDE
- {
- if ((pformatetc->dwAspect & DVASPECT_CONTENT) &&
- (pformatetc->tymed & TYMED_HGLOBAL) &&
- (pformatetc->cfFormat == CF_UNICODETEXT))
- return S_OK;
- return DV_E_FORMATETC;
- }
-// HRESULT STDMETHODCALLTYPE EnumFormatEtc( DWORD dir, IEnumFORMATETC** ppenumFormatEtc) {
-// *ppenumFormatEtc = m_EnumF;
-// return S_OK;
-// }
-
- // all the following methods are not really needed for a DnD object
- HRESULT STDMETHODCALLTYPE GetDataHere( FORMATETC* /*pformatetcIn*/, STGMEDIUM* /*pmedium*/) FL_OVERRIDE { return E_NOTIMPL; }
- HRESULT STDMETHODCALLTYPE GetCanonicalFormatEtc( FORMATETC* /*in*/, FORMATETC* /*out*/) FL_OVERRIDE { return E_NOTIMPL; }
- HRESULT STDMETHODCALLTYPE SetData( FORMATETC* /*pformatetc*/, STGMEDIUM* /*pmedium*/, BOOL /*fRelease*/) FL_OVERRIDE { return E_NOTIMPL; }
- HRESULT STDMETHODCALLTYPE EnumFormatEtc( DWORD /*dir*/, IEnumFORMATETC** /*ppenumFormatEtc*/) FL_OVERRIDE { return E_NOTIMPL; }
-// HRESULT STDMETHODCALLTYPE EnumFormatEtc( DWORD dir, IEnumFORMATETC** ppenumFormatEtc) FL_OVERRIDE {*ppenumFormatEtc = m_EnumF; return S_OK;}
- HRESULT STDMETHODCALLTYPE DAdvise( FORMATETC* /*pformatetc*/, DWORD /*advf*/,
- IAdviseSink* /*pAdvSink*/, DWORD* /*pdwConnection*/) FL_OVERRIDE { return E_NOTIMPL; }
- HRESULT STDMETHODCALLTYPE DUnadvise( DWORD /*dwConnection*/) FL_OVERRIDE { return E_NOTIMPL; }
- HRESULT STDMETHODCALLTYPE EnumDAdvise( IEnumSTATDATA** /*ppenumAdvise*/) FL_OVERRIDE { return E_NOTIMPL; }
-};
-
-
-int Fl_WinAPI_Screen_Driver::dnd(int unused)
-{
- DWORD dropEffect;
- ReleaseCapture();
-
- FLDataObject *fdo = new FLDataObject;
- fdo->AddRef();
- FLDropSource *fds = new FLDropSource;
- fds->AddRef();
-
- HRESULT ret = DoDragDrop( fdo, fds, DROPEFFECT_MOVE|DROPEFFECT_LINK|DROPEFFECT_COPY, &dropEffect );
-
- fdo->Release();
- fds->Release();
-
- Fl_Widget *w = Fl::pushed();
- if ( w )
- {
- int old_event = Fl::e_number;
- w->handle(Fl::e_number = FL_RELEASE);
- Fl::e_number = old_event;
- Fl::pushed( 0 );
- }
- if ( ret==DRAGDROP_S_DROP ) return 1; // or DD_S_CANCEL
- return 0;
-}