1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/dragimag.cpp
3 // Purpose: wxDragImage
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
30 #include "wx/msw/wrapcctl.h" // include <commctrl.h> "properly"
32 #include "wx/window.h"
33 #include "wx/dcclient.h"
34 #include "wx/dcscreen.h"
35 #include "wx/dcmemory.h"
36 #include "wx/settings.h"
43 #include "wx/msw/private.h"
45 #include "wx/msw/dragimag.h"
46 #include "wx/msw/private.h"
48 #ifdef __WXWINCE__ // for SM_CXCURSOR and SM_CYCURSOR
49 #include "wx/msw/wince/missing.h"
52 // Wine doesn't have this yet
53 #ifndef ListView_CreateDragImage
54 #define ListView_CreateDragImage(hwnd, i, lpptUpLeft) \
55 (HIMAGELIST)SNDMSG((hwnd), LVM_CREATEDRAGIMAGE, (WPARAM)(int)(i), (LPARAM)(LPPOINT)(lpptUpLeft))
58 // ----------------------------------------------------------------------------
60 // ----------------------------------------------------------------------------
62 IMPLEMENT_DYNAMIC_CLASS(wxDragImage
, wxObject
)
64 #define GetHimageList() ((HIMAGELIST) m_hImageList)
66 // ============================================================================
68 // ============================================================================
70 // ----------------------------------------------------------------------------
71 // wxDragImage ctors/dtor
72 // ----------------------------------------------------------------------------
74 wxDragImage::wxDragImage()
79 wxDragImage::~wxDragImage()
82 ImageList_Destroy(GetHimageList());
83 #if !wxUSE_SIMPLER_DRAGIMAGE
84 if ( m_hCursorImageList
)
85 ImageList_Destroy((HIMAGELIST
) m_hCursorImageList
);
89 void wxDragImage::Init()
92 #if !wxUSE_SIMPLER_DRAGIMAGE
93 m_hCursorImageList
= 0;
95 m_window
= (wxWindow
*) NULL
;
100 ////////////////////////////////////////////////////////////////////////////
104 ////////////////////////////////////////////////////////////////////////////
106 // Create a drag image from a bitmap and optional cursor
107 bool wxDragImage::Create(const wxBitmap
& image
, const wxCursor
& cursor
)
110 ImageList_Destroy(GetHimageList());
114 UINT flags
= ILC_COLOR
;
116 UINT flags
wxDUMMY_INITIALIZE(0) ;
117 if (image
.GetDepth() <= 4)
119 else if (image
.GetDepth() <= 8)
121 else if (image
.GetDepth() <= 16)
123 else if (image
.GetDepth() <= 24)
129 bool mask
= (image
.GetMask() != 0);
131 // Curiously, even if the image doesn't have a mask,
132 // we still have to use ILC_MASK or the image won't show
137 m_hImageList
= (WXHIMAGELIST
) ImageList_Create(image
.GetWidth(), image
.GetHeight(), flags
, 1, 1);
142 HBITMAP hBitmap1
= (HBITMAP
) image
.GetHBITMAP();
143 index
= ImageList_Add(GetHimageList(), hBitmap1
, 0);
147 HBITMAP hBitmap1
= (HBITMAP
) image
.GetHBITMAP();
148 HBITMAP hBitmap2
= (HBITMAP
) image
.GetMask()->GetMaskBitmap();
149 HBITMAP hbmpMask
= wxInvertMask(hBitmap2
);
151 index
= ImageList_Add(GetHimageList(), hBitmap1
, hbmpMask
);
152 ::DeleteObject(hbmpMask
);
156 wxLogError(_("Couldn't add an image to the image list."));
158 m_cursor
= cursor
; // Can only combine with drag image after calling BeginDrag.
160 return (index
!= -1) ;
163 // Create a drag image from an icon and optional cursor
164 bool wxDragImage::Create(const wxIcon
& image
, const wxCursor
& cursor
)
167 ImageList_Destroy(GetHimageList());
171 UINT flags
= ILC_COLOR
;
173 UINT flags
wxDUMMY_INITIALIZE(0) ;
174 if (image
.GetDepth() <= 4)
176 else if (image
.GetDepth() <= 8)
178 else if (image
.GetDepth() <= 16)
180 else if (image
.GetDepth() <= 24)
188 m_hImageList
= (WXHIMAGELIST
) ImageList_Create(image
.GetWidth(), image
.GetHeight(), flags
, 1, 1);
190 HICON hIcon
= (HICON
) image
.GetHICON();
192 int index
= ImageList_AddIcon(GetHimageList(), hIcon
);
195 wxLogError(_("Couldn't add an image to the image list."));
198 m_cursor
= cursor
; // Can only combine with drag image after calling BeginDrag.
200 return (index
!= -1) ;
203 // Create a drag image from a string and optional cursor
204 bool wxDragImage::Create(const wxString
& str
, const wxCursor
& cursor
)
206 wxFont
font(wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT
));
208 wxCoord w
= 0, h
= 0;
211 dc
.GetTextExtent(str
, & w
, & h
);
212 dc
.SetFont(wxNullFont
);
216 wxBitmap
bitmap((int) w
+2, (int) h
+2);
217 dc2
.SelectObject(bitmap
);
219 dc2
.SetBackground(* wxWHITE_BRUSH
);
221 dc2
.SetBackgroundMode(wxTRANSPARENT
);
222 dc2
.SetTextForeground(* wxLIGHT_GREY
);
223 dc2
.DrawText(str
, 0, 0);
224 dc2
.DrawText(str
, 1, 0);
225 dc2
.DrawText(str
, 2, 0);
226 dc2
.DrawText(str
, 1, 1);
227 dc2
.DrawText(str
, 2, 1);
228 dc2
.DrawText(str
, 1, 2);
229 dc2
.DrawText(str
, 2, 2);
230 dc2
.SetTextForeground(* wxBLACK
);
231 dc2
.DrawText(str
, 1, 1);
233 dc2
.SelectObject(wxNullBitmap
);
236 // Make the bitmap masked
237 wxImage image
= bitmap
.ConvertToImage();
238 image
.SetMaskColour(255, 255, 255);
239 return Create(wxBitmap(image
), cursor
);
246 // Create a drag image for the given tree control item
247 bool wxDragImage::Create(const wxTreeCtrl
& treeCtrl
, wxTreeItemId
& id
)
250 ImageList_Destroy(GetHimageList());
251 m_hImageList
= (WXHIMAGELIST
)
252 TreeView_CreateDragImage(GetHwndOf(&treeCtrl
), (HTREEITEM
) id
.m_pItem
);
253 return m_hImageList
!= 0;
258 // Create a drag image for the given list control item
259 bool wxDragImage::Create(const wxListCtrl
& listCtrl
, long id
)
262 ImageList_Destroy(GetHimageList());
265 m_hImageList
= (WXHIMAGELIST
) ListView_CreateDragImage((HWND
) listCtrl
.GetHWND(), id
, & pt
);
271 bool wxDragImage::BeginDrag(const wxPoint
& hotspot
, wxWindow
* window
, bool fullScreen
, wxRect
* rect
)
273 wxASSERT_MSG( (m_hImageList
!= 0), wxT("Image list must not be null in BeginDrag."));
274 wxASSERT_MSG( (window
!= 0), wxT("Window must not be null in BeginDrag."));
276 m_fullScreen
= fullScreen
;
278 m_boundingRect
= * rect
;
280 bool ret
= (ImageList_BeginDrag(GetHimageList(), 0, hotspot
.x
, hotspot
.y
) != 0);
284 wxFAIL_MSG( _T("BeginDrag failed.") );
291 #if wxUSE_SIMPLER_DRAGIMAGE
292 m_oldCursor
= window
->GetCursor();
293 window
->SetCursor(m_cursor
);
295 if (!m_hCursorImageList
)
298 // Smartphone may not have these metric symbol
302 int cxCursor
= ::GetSystemMetrics(SM_CXCURSOR
);
303 int cyCursor
= ::GetSystemMetrics(SM_CYCURSOR
);
305 m_hCursorImageList
= (WXHIMAGELIST
) ImageList_Create(cxCursor
, cyCursor
, ILC_MASK
, 1, 1);
308 // See if we can find the cursor hotspot
309 wxPoint
curHotSpot(hotspot
);
311 // Although it seems to produce the right position, when the hotspot goeos
312 // negative it has strange effects on the image.
313 // How do we stop the cursor jumping right and below of where it should be?
316 if (::GetIconInfo((HICON
) (HCURSOR
) m_cursor
.GetHCURSOR(), & iconInfo
) != 0)
318 curHotSpot
.x
-= iconInfo
.xHotspot
;
319 curHotSpot
.y
-= iconInfo
.yHotspot
;
323 //msg.Printf("Hotspot = %d, %d", curHotSpot.x, curHotSpot.y);
326 // First add the cursor to the image list
327 HCURSOR hCursor
= (HCURSOR
) m_cursor
.GetHCURSOR();
328 int cursorIndex
= ImageList_AddIcon((HIMAGELIST
) m_hCursorImageList
, (HICON
) hCursor
);
330 wxASSERT_MSG( (cursorIndex
!= -1), wxT("ImageList_AddIcon failed in BeginDrag."));
332 if (cursorIndex
!= -1)
334 ImageList_SetDragCursorImage((HIMAGELIST
) m_hCursorImageList
, cursorIndex
, curHotSpot
.x
, curHotSpot
.y
);
339 #if !wxUSE_SIMPLER_DRAGIMAGE
346 ::SetCapture(GetHwndOf(window
));
351 // Begin drag. hotspot is the location of the drag position relative to the upper-left
352 // corner of the image. This is full screen only. fullScreenRect gives the
353 // position of the window on the screen, to restrict the drag to.
354 bool wxDragImage::BeginDrag(const wxPoint
& hotspot
, wxWindow
* window
, wxWindow
* fullScreenRect
)
358 int x
= fullScreenRect
->GetPosition().x
;
359 int y
= fullScreenRect
->GetPosition().y
;
361 wxSize sz
= fullScreenRect
->GetSize();
363 if (fullScreenRect
->GetParent() && !fullScreenRect
->IsKindOf(CLASSINFO(wxFrame
)))
364 fullScreenRect
->GetParent()->ClientToScreen(& x
, & y
);
366 rect
.x
= x
; rect
.y
= y
;
367 rect
.width
= sz
.x
; rect
.height
= sz
.y
;
369 return BeginDrag(hotspot
, window
, true, & rect
);
373 bool wxDragImage::EndDrag()
375 wxASSERT_MSG( (m_hImageList
!= 0), wxT("Image list must not be null in EndDrag."));
379 if ( !::ReleaseCapture() )
381 wxLogLastError(wxT("ReleaseCapture"));
384 #if wxUSE_SIMPLER_DRAGIMAGE
385 if (m_cursor
.Ok() && m_oldCursor
.Ok())
386 m_window
->SetCursor(m_oldCursor
);
391 m_window
= (wxWindow
*) NULL
;
396 // Move the image: call from OnMouseMove. Pt is in window client coordinates if window
397 // is non-NULL, or in screen coordinates if NULL.
398 bool wxDragImage::Move(const wxPoint
& pt
)
400 wxASSERT_MSG( (m_hImageList
!= 0), wxT("Image list must not be null in Move."));
402 // These are in window, not client coordinates.
403 // So need to convert to client coordinates.
405 if (m_window
&& !m_fullScreen
)
408 rect
.left
= 0; rect
.top
= 0;
409 rect
.right
= 0; rect
.bottom
= 0;
410 DWORD style
= ::GetWindowLong((HWND
) m_window
->GetHWND(), GWL_STYLE
);
412 DWORD exStyle
= ::GetWindowLong((HWND
) m_window
->GetHWND(), GWL_EXSTYLE
);
413 ::AdjustWindowRectEx(& rect
, style
, FALSE
, exStyle
);
415 ::AdjustWindowRect(& rect
, style
, FALSE
);
417 // Subtract the (negative) values, i.e. add a small increment
418 pt2
.x
-= rect
.left
; pt2
.y
-= rect
.top
;
420 else if (m_window
&& m_fullScreen
)
422 pt2
= m_window
->ClientToScreen(pt2
);
425 bool ret
= (ImageList_DragMove( pt2
.x
, pt2
.y
) != 0);
432 bool wxDragImage::Show()
434 wxASSERT_MSG( (m_hImageList
!= 0), wxT("Image list must not be null in Show."));
437 if (m_window
&& !m_fullScreen
)
438 hWnd
= (HWND
) m_window
->GetHWND();
440 bool ret
= (ImageList_DragEnter( hWnd
, m_position
.x
, m_position
.y
) != 0);
445 bool wxDragImage::Hide()
447 wxASSERT_MSG( (m_hImageList
!= 0), wxT("Image list must not be null in Hide."));
450 if (m_window
&& !m_fullScreen
)
451 hWnd
= (HWND
) m_window
->GetHWND();
453 bool ret
= (ImageList_DragLeave( hWnd
) != 0);
458 #endif // wxUSE_DRAGIMAGE