]> git.saurik.com Git - wxWidgets.git/blame - src/msw/ole/dropsrc.cpp
we do want arrows even in a read only text ctrl
[wxWidgets.git] / src / msw / ole / dropsrc.cpp
CommitLineData
269a5a34
VZ
1///////////////////////////////////////////////////////////////////////////////
2// Name: msw/ole/dropsrc.cpp
3// Purpose: implementation of wxIDropSource and wxDropSource
4// Author: Vadim Zeitlin
3f480da3 5// Modified by:
269a5a34
VZ
6// Created: 10.05.98
7// RCS-ID: $Id$
8// Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9// Licence: wxWindows license
10///////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20#ifdef __GNUG__
21 #pragma implementation "dropsrc.h"
22#endif
23
24// For compilers that support precompilation, includes "wx.h".
25#include "wx/wxprec.h"
26
27#if defined(__BORLANDC__)
28 #pragma hdrstop
29#endif
7dee726c
RS
30#ifndef WX_PRECOMP
31#include "wx/window.h"
32#endif
269a5a34 33
3f480da3 34#include "wx/setup.h"
269a5a34 35
21709999 36#if wxUSE_OLE && wxUSE_DRAG_AND_DROP
269a5a34 37
3f480da3 38#include "wx/log.h"
9e2896e5 39#include "wx/dnd.h"
269a5a34 40
17b74d79 41#include <windows.h>
9e2896e5 42
b64f0a5f 43#if wxUSE_NORLANDER_HEADERS
9e2896e5 44 #include <ole2.h>
7dee726c 45#endif
17b74d79 46
269a5a34 47#ifndef __WIN32__
9e2896e5
VZ
48 #include <ole2.h>
49 #include <olestd.h>
269a5a34
VZ
50#endif
51
17b74d79
JS
52#include <oleauto.h>
53
3f480da3 54#include "wx/msw/ole/oleutils.h"
17b74d79 55
269a5a34
VZ
56// ----------------------------------------------------------------------------
57// wxIDropSource implementation of IDropSource interface
58// ----------------------------------------------------------------------------
59
60class wxIDropSource : public IDropSource
61{
62public:
63 wxIDropSource(wxDropSource *pDropSource);
64
65 DECLARE_IUNKNOWN_METHODS;
66
67 // IDropSource
68 STDMETHODIMP QueryContinueDrag(BOOL fEscapePressed, DWORD grfKeyState);
69 STDMETHODIMP GiveFeedback(DWORD dwEffect);
70
71private:
72 DWORD m_grfInitKeyState; // button which started the d&d operation
73 wxDropSource *m_pDropSource; // pointer to C++ class we belong to
74};
75
76// ============================================================================
77// Implementation
78// ============================================================================
79
80// ----------------------------------------------------------------------------
81// wxIDropSource implementation
82// ----------------------------------------------------------------------------
83BEGIN_IID_TABLE(wxIDropSource)
84 ADD_IID(Unknown)
85 ADD_IID(DropSource)
86END_IID_TABLE;
87
88IMPLEMENT_IUNKNOWN_METHODS(wxIDropSource)
89
90wxIDropSource::wxIDropSource(wxDropSource *pDropSource)
91{
92 wxASSERT( pDropSource != NULL );
93
94 m_pDropSource = pDropSource;
95 m_grfInitKeyState = 0;
269a5a34
VZ
96}
97
98// Name : wxIDropSource::QueryContinueDrag
99// Purpose : decide if drag operation must be continued or not
100// Returns : HRESULT: S_OK if we should continue
101// DRAGDROP_S_DROP to drop right now
102// DRAGDROP_S_CANCEL to cancel everything
103// Params : [in] BOOL fEscapePressed <Esc> pressed since last call?
104// [in] DWORD grfKeyState mask containing state of kbd keys
105// Notes : as there is no reasonably simple portable way to implement this
106// function, we currently don't give the possibility to override the
107// default behaviour implemented here
108STDMETHODIMP wxIDropSource::QueryContinueDrag(BOOL fEscapePressed,
109 DWORD grfKeyState)
110{
111 if ( fEscapePressed )
112 return DRAGDROP_S_CANCEL;
113
114 // initialize ourself with the drag begin button
115 if ( m_grfInitKeyState == 0 ) {
116 m_grfInitKeyState = grfKeyState & (MK_LBUTTON | MK_RBUTTON | MK_MBUTTON);
117 }
118
119 if ( !(grfKeyState & m_grfInitKeyState) ) {
120 // button which started d&d released, go!
121 return DRAGDROP_S_DROP;
122 }
123
124 return S_OK;
125}
126
127// Name : wxIDropSource::GiveFeedback
128// Purpose : give UI feedback according to current state of operation
3f480da3 129// Returns : STDMETHODIMP
269a5a34
VZ
130// Params : [in] DWORD dwEffect - what would happen if we dropped now
131// Notes : default implementation is ok in more than 99% of cases
132STDMETHODIMP wxIDropSource::GiveFeedback(DWORD dwEffect)
133{
46ccb510 134 wxDragResult effect;
269a5a34 135 if ( dwEffect & DROPEFFECT_COPY )
46ccb510 136 effect = wxDragCopy;
269a5a34 137 else if ( dwEffect & DROPEFFECT_MOVE )
46ccb510 138 effect = wxDragMove;
269a5a34 139 else
46ccb510 140 effect = wxDragNone;
269a5a34 141
2d93e133 142 if ( m_pDropSource->GiveFeedback(effect) )
269a5a34
VZ
143 return S_OK;
144
145 return DRAGDROP_S_USEDEFAULTCURSORS;
146}
147
148// ----------------------------------------------------------------------------
149// wxDropSource implementation
150// ----------------------------------------------------------------------------
151
152// ctors
153
154// common part of all ctors
155void wxDropSource::Init()
156{
2d93e133
VZ
157 m_pIDropSource = new wxIDropSource(this);
158 m_pIDropSource->AddRef();
269a5a34
VZ
159}
160
8e193f38 161wxDropSource::wxDropSource(wxWindow* WXUNUSED(win),
2d93e133
VZ
162 const wxCursor &cursorCopy,
163 const wxCursor &cursorMove,
164 const wxCursor &cursorStop)
165 : wxDropSourceBase(cursorCopy, cursorMove, cursorStop)
269a5a34 166{
2d93e133 167 Init();
269a5a34
VZ
168}
169
8e193f38
VZ
170wxDropSource::wxDropSource(wxDataObject& data,
171 wxWindow* WXUNUSED(win),
2d93e133
VZ
172 const wxCursor &cursorCopy,
173 const wxCursor &cursorMove,
174 const wxCursor &cursorStop)
175 : wxDropSourceBase(cursorCopy, cursorMove, cursorStop)
269a5a34 176{
2d93e133
VZ
177 Init();
178 SetData(data);
269a5a34
VZ
179}
180
269a5a34
VZ
181wxDropSource::~wxDropSource()
182{
2d93e133 183 m_pIDropSource->Release();
269a5a34
VZ
184}
185
186// Name : DoDragDrop
187// Purpose : start drag and drop operation
46ccb510 188// Returns : wxDragResult - the code of performed operation
2245b2b2 189// Params : [in] int flags: specifies if moving is allowe (or only copying)
269a5a34 190// Notes : you must call SetData() before if you had used def ctor
2245b2b2 191wxDragResult wxDropSource::DoDragDrop(int flags)
269a5a34 192{
9e2896e5 193 wxCHECK_MSG( m_data != NULL, wxDragNone, wxT("No data in wxDropSource!") );
269a5a34
VZ
194
195 DWORD dwEffect;
9e2896e5 196 HRESULT hr = ::DoDragDrop(m_data->GetInterface(),
3f480da3 197 m_pIDropSource,
2245b2b2
VZ
198 (flags & wxDrag_AllowMove)
199 ? DROPEFFECT_COPY | DROPEFFECT_MOVE
200 : DROPEFFECT_COPY,
269a5a34
VZ
201 &dwEffect);
202
203 if ( hr == DRAGDROP_S_CANCEL ) {
46ccb510 204 return wxDragCancel;
269a5a34
VZ
205 }
206 else if ( hr == DRAGDROP_S_DROP ) {
207 if ( dwEffect & DROPEFFECT_COPY ) {
46ccb510 208 return wxDragCopy;
269a5a34
VZ
209 }
210 else if ( dwEffect & DROPEFFECT_MOVE ) {
211 // consistency check: normally, we shouldn't get "move" at all
2245b2b2
VZ
212 // here if we don't allow it, but in practice it does happen quite often
213 return (flags & wxDrag_AllowMove) ? wxDragMove : wxDragCopy;
269a5a34
VZ
214 }
215 else {
216 // not copy or move
46ccb510 217 return wxDragNone;
269a5a34
VZ
218 }
219 }
220 else {
221 if ( FAILED(hr) ) {
161f4f73 222 wxLogApiError(wxT("DoDragDrop"), hr);
223d09f6 223 wxLogError(wxT("Drag & drop operation failed."));
269a5a34
VZ
224 }
225 else {
2d93e133
VZ
226 wxLogDebug(wxT("Unexpected success return code %08lx from DoDragDrop."),
227 hr);
269a5a34
VZ
228 }
229
46ccb510 230 return wxDragError;
269a5a34
VZ
231 }
232}
233
234// Name : wxDropSource::GiveFeedback
235// Purpose : visually inform the user about d&d operation state
236// Returns : bool: true if we do all ourselves or false for default feedback
237// Params : [in] DragResult effect - what would happen if we dropped now
269a5a34 238// Notes : here we just leave this stuff for default implementation
2d93e133 239bool wxDropSource::GiveFeedback(wxDragResult effect)
269a5a34 240{
2d93e133
VZ
241 const wxCursor& cursor = GetCursor(effect);
242 if ( cursor.Ok() )
243 {
244 ::SetCursor((HCURSOR)cursor.GetHCURSOR());
245
246 return TRUE;
247 }
248 else
249 {
250 return FALSE;
251 }
269a5a34
VZ
252}
253
46ccb510 254#endif //USE_DRAG_AND_DROP