1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/cursor.cpp
3 // Purpose: wxCursor class
4 // Author: Julian Smart
8 // Copyright: (c) 1997-2003 Julian Smart and Vadim Zeitlin
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
27 #include "wx/cursor.h"
32 #include "wx/bitmap.h"
34 #include "wx/settings.h"
37 #include "wx/module.h"
40 #include "wx/msw/private.h"
41 #include "wx/msw/missing.h" // IDC_HAND
43 // define functions missing in MicroWin
45 static inline void DestroyCursor(HCURSOR
) { }
46 static inline void SetCursor(HCURSOR
) { }
47 #endif // __WXMICROWIN__
49 // ----------------------------------------------------------------------------
51 // ----------------------------------------------------------------------------
53 class WXDLLEXPORT wxCursorRefData
: public wxGDIImageRefData
56 // the second parameter is used to tell us to delete the cursor when we're
57 // done with it (normally we shouldn't call DestroyCursor() this is why it
58 // doesn't happen by default)
59 wxCursorRefData(HCURSOR hcursor
= 0, bool takeOwnership
= false);
61 virtual ~wxCursorRefData() { Free(); }
66 // return the size of the standard cursor: notice that the system only
67 // supports the cursors of this size
68 static wxCoord
GetStandardWidth();
69 static wxCoord
GetStandardHeight();
74 // standard cursor size, computed on first use
75 static wxSize ms_sizeStd
;
78 // ----------------------------------------------------------------------------
80 // ----------------------------------------------------------------------------
82 IMPLEMENT_DYNAMIC_CLASS(wxCursor
, wxGDIObject
)
84 // ----------------------------------------------------------------------------
86 // ----------------------------------------------------------------------------
88 // Current cursor, in order to hang on to cursor handle when setting the cursor
90 static wxCursor
*gs_globalCursor
= NULL
;
92 // ----------------------------------------------------------------------------
94 // ----------------------------------------------------------------------------
96 class wxCursorModule
: public wxModule
101 gs_globalCursor
= new wxCursor
;
106 virtual void OnExit()
108 wxDELETE(gs_globalCursor
);
112 // ============================================================================
114 // ============================================================================
116 // ----------------------------------------------------------------------------
118 // ----------------------------------------------------------------------------
120 wxSize
wxCursorRefData::ms_sizeStd
;
122 wxCoord
wxCursorRefData::GetStandardWidth()
125 ms_sizeStd
.x
= wxSystemSettings::GetMetric(wxSYS_CURSOR_X
);
130 wxCoord
wxCursorRefData::GetStandardHeight()
133 ms_sizeStd
.y
= wxSystemSettings::GetMetric(wxSYS_CURSOR_Y
);
138 wxCursorRefData::wxCursorRefData(HCURSOR hcursor
, bool destroy
)
140 m_hCursor
= (WXHCURSOR
)hcursor
;
144 m_width
= GetStandardWidth();
145 m_height
= GetStandardHeight();
148 m_destroyCursor
= destroy
;
151 void wxCursorRefData::Free()
156 if ( m_destroyCursor
)
157 ::DestroyCursor((HCURSOR
)m_hCursor
);
164 // ----------------------------------------------------------------------------
166 // ----------------------------------------------------------------------------
173 wxCursor::wxCursor(const wxImage
& image
)
175 // image has to be of the standard cursor size, otherwise we won't be able
177 const int w
= wxCursorRefData::GetStandardWidth();
178 const int h
= wxCursorRefData::GetStandardHeight();
180 int hotSpotX
= image
.GetOptionInt(wxIMAGE_OPTION_CUR_HOTSPOT_X
);
181 int hotSpotY
= image
.GetOptionInt(wxIMAGE_OPTION_CUR_HOTSPOT_Y
);
182 int image_w
= image
.GetWidth();
183 int image_h
= image
.GetHeight();
185 wxASSERT_MSG( hotSpotX
>= 0 && hotSpotX
< image_w
&&
186 hotSpotY
>= 0 && hotSpotY
< image_h
,
187 wxT("invalid cursor hot spot coordinates") );
189 wxImage
imageSized(image
); // final image of correct size
191 // if image is too small then place it in the center, resize it if too big
192 if ((w
> image_w
) && (h
> image_h
))
194 wxPoint
offset((w
- image_w
)/2, (h
- image_h
)/2);
195 hotSpotX
= hotSpotX
+ offset
.x
;
196 hotSpotY
= hotSpotY
+ offset
.y
;
198 imageSized
= image
.Size(wxSize(w
, h
), offset
);
200 else if ((w
!= image_w
) || (h
!= image_h
))
202 hotSpotX
= int(hotSpotX
* double(w
) / double(image_w
));
203 hotSpotY
= int(hotSpotY
* double(h
) / double(image_h
));
205 imageSized
= image
.Scale(w
, h
);
208 HCURSOR hcursor
= wxBitmapToHCURSOR( wxBitmap(imageSized
),
209 hotSpotX
, hotSpotY
);
213 wxLogWarning(_("Failed to create cursor."));
217 m_refData
= new wxCursorRefData(hcursor
, true /* delete it later */);
219 #endif // wxUSE_IMAGE
221 // MicroWin doesn't have support needed for the other ctors
222 #ifdef __WXMICROWIN__
224 wxCursor::InitFromStock(wxStockCursor
WXUNUSED(cursor_type
))
228 #else // !__WXMICROWIN__
230 wxCursor::wxCursor(const wxString
& filename
,
238 case wxBITMAP_TYPE_CUR_RESOURCE
:
239 hcursor
= ::LoadCursor(wxGetInstance(), filename
.fn_str());
243 case wxBITMAP_TYPE_CUR
:
244 hcursor
= ::LoadCursorFromFile(filename
.fn_str());
248 case wxBITMAP_TYPE_ICO
:
249 hcursor
= wxBitmapToHCURSOR
251 wxIcon(filename
, wxBITMAP_TYPE_ICO
),
257 case wxBITMAP_TYPE_BMP
:
258 hcursor
= wxBitmapToHCURSOR
260 wxBitmap(filename
, wxBITMAP_TYPE_BMP
),
267 wxLogError( wxT("unknown cursor resource type '%d'"), kind
);
274 m_refData
= new wxCursorRefData(hcursor
, true /* delete it later */);
278 // Cursors by stock number
279 void wxCursor::InitFromStock(wxStockCursor idCursor
)
281 // all wxWidgets standard cursors
282 static const struct StdCursor
284 // is this a standard Windows cursor?
287 // the cursor name or id
291 { true, NULL
}, // wxCURSOR_NONE
292 { true, IDC_ARROW
}, // wxCURSOR_ARROW
293 { false, wxT("WXCURSOR_RIGHT_ARROW") }, // wxCURSOR_RIGHT_ARROW
294 { false, wxT("WXCURSOR_BULLSEYE") }, // wxCURSOR_BULLSEYE
295 { true, IDC_ARROW
}, // WXCURSOR_CHAR
297 // Displays as an I-beam on XP, so use a cursor file
298 // { true, IDC_CROSS }, // WXCURSOR_CROSS
299 { false, wxT("WXCURSOR_CROSS") }, // WXCURSOR_CROSS
301 // See special handling below for wxCURSOR_HAND
302 // { false, wxT("WXCURSOR_HAND") }, // wxCURSOR_HAND
303 { true, IDC_HAND
}, // wxCURSOR_HAND
305 { true, IDC_IBEAM
}, // WXCURSOR_IBEAM
306 { true, IDC_ARROW
}, // WXCURSOR_LEFT_BUTTON
307 { false, wxT("WXCURSOR_MAGNIFIER") }, // wxCURSOR_MAGNIFIER
308 { true, IDC_ARROW
}, // WXCURSOR_MIDDLE_BUTTON
309 { true, IDC_NO
}, // WXCURSOR_NO_ENTRY
310 { false, wxT("WXCURSOR_PBRUSH") }, // wxCURSOR_PAINT_BRUSH
311 { false, wxT("WXCURSOR_PENCIL") }, // wxCURSOR_PENCIL
312 { false, wxT("WXCURSOR_PLEFT") }, // wxCURSOR_POINT_LEFT
313 { false, wxT("WXCURSOR_PRIGHT") }, // wxCURSOR_POINT_RIGHT
314 { true, IDC_HELP
}, // WXCURSOR_QUESTION_ARROW
315 { true, IDC_ARROW
}, // WXCURSOR_RIGHT_BUTTON
316 { true, IDC_SIZENESW
}, // WXCURSOR_SIZENESW
317 { true, IDC_SIZENS
}, // WXCURSOR_SIZENS
318 { true, IDC_SIZENWSE
}, // WXCURSOR_SIZENWSE
319 { true, IDC_SIZEWE
}, // WXCURSOR_SIZEWE
320 { true, IDC_SIZEALL
}, // WXCURSOR_SIZING
321 { false, wxT("WXCURSOR_PBRUSH") }, // wxCURSOR_SPRAYCAN
322 { true, IDC_WAIT
}, // WXCURSOR_WAIT
323 { true, IDC_WAIT
}, // WXCURSOR_WATCH
324 { false, wxT("WXCURSOR_BLANK") }, // wxCURSOR_BLANK
325 { true, IDC_APPSTARTING
}, // wxCURSOR_ARROWWAIT
327 // no entry for wxCURSOR_MAX
330 wxCOMPILE_TIME_ASSERT( WXSIZEOF(stdCursors
) == wxCURSOR_MAX
,
331 CursorsIdArrayMismatch
);
333 wxCHECK_RET( idCursor
> 0 && (size_t)idCursor
< WXSIZEOF(stdCursors
),
334 wxT("invalid cursor id in wxCursor() ctor") );
336 const StdCursor
& stdCursor
= stdCursors
[idCursor
];
337 bool deleteLater
= !stdCursor
.isStd
;
339 HCURSOR hcursor
= ::LoadCursor(stdCursor
.isStd
? NULL
: wxGetInstance(),
342 // IDC_HAND may not be available on some versions of Windows.
343 if ( !hcursor
&& idCursor
== wxCURSOR_HAND
)
345 hcursor
= ::LoadCursor(wxGetInstance(), wxT("WXCURSOR_HAND"));
351 if ( !stdCursor
.isStd
)
353 // it may be not obvious to the programmer why did loading fail,
354 // try to help by pointing to the by far the most probable reason
355 wxFAIL_MSG(wxT("Loading a cursor defined by wxWidgets failed, ")
356 wxT("did you include include/wx/msw/wx.rc file from ")
357 wxT("your resource file?"));
360 wxLogLastError(wxT("LoadCursor"));
364 m_refData
= new wxCursorRefData(hcursor
, deleteLater
);
368 #endif // __WXMICROWIN__/!__WXMICROWIN__
370 wxCursor::~wxCursor()
374 // ----------------------------------------------------------------------------
375 // other wxCursor functions
376 // ----------------------------------------------------------------------------
378 wxGDIImageRefData
*wxCursor::CreateData() const
380 return new wxCursorRefData
;
383 // ----------------------------------------------------------------------------
384 // Global cursor setting
385 // ----------------------------------------------------------------------------
387 const wxCursor
*wxGetGlobalCursor()
389 return gs_globalCursor
;
392 void wxSetCursor(const wxCursor
& cursor
)
396 ::SetCursor(GetHcursorOf(cursor
));
398 if ( gs_globalCursor
)
399 *gs_globalCursor
= cursor
;