]> git.saurik.com Git - wxWidgets.git/blame - src/msw/cursor.cpp
Better fix
[wxWidgets.git] / src / msw / cursor.cpp
CommitLineData
2bda0e17 1/////////////////////////////////////////////////////////////////////////////
eff4ffbf 2// Name: src/msw/cursor.cpp
2bda0e17
KB
3// Purpose: wxCursor class
4// Author: Julian Smart
5// Modified by:
6// Created: 01/02/97
7// RCS-ID: $Id$
eff4ffbf 8// Copyright: (c) 1997-2003 Julian Smart and Vadim Zeitlin
65571936 9// Licence: wxWindows licence
2bda0e17
KB
10/////////////////////////////////////////////////////////////////////////////
11
0d0512bd
VZ
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
2bda0e17
KB
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
0d0512bd 24 #pragma hdrstop
2bda0e17
KB
25#endif
26
c8326d64
WS
27#include "wx/cursor.h"
28
2bda0e17 29#ifndef WX_PRECOMP
0d0512bd
VZ
30 #include "wx/utils.h"
31 #include "wx/app.h"
fef15b42 32 #include "wx/bitmap.h"
0d0512bd 33 #include "wx/icon.h"
ed39ff57 34 #include "wx/settings.h"
51d5ec54 35 #include "wx/intl.h"
155ecd4c 36 #include "wx/image.h"
02761f6c 37 #include "wx/module.h"
2bda0e17
KB
38#endif
39
40#include "wx/msw/private.h"
a6c2e2c7 41#include "wx/msw/missing.h" // IDC_HAND
eff4ffbf 42
eff4ffbf
VZ
43// define functions missing in MicroWin
44#ifdef __WXMICROWIN__
45 static inline void DestroyCursor(HCURSOR) { }
46 static inline void SetCursor(HCURSOR) { }
47#endif // __WXMICROWIN__
48
49// ----------------------------------------------------------------------------
50// private classes
51// ----------------------------------------------------------------------------
52
53class WXDLLEXPORT wxCursorRefData : public wxGDIImageRefData
54{
55public:
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);
60
61 virtual ~wxCursorRefData() { Free(); }
62
63 virtual void Free();
64
65
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();
70
71private:
72 bool m_destroyCursor;
73
74 // standard cursor size, computed on first use
75 static wxSize ms_sizeStd;
76};
77
0d0512bd
VZ
78// ----------------------------------------------------------------------------
79// wxWin macros
80// ----------------------------------------------------------------------------
81
621b3e21 82IMPLEMENT_DYNAMIC_CLASS(wxCursor, wxGDIObject)
bfbd6dc1
VZ
83
84// ----------------------------------------------------------------------------
85// globals
86// ----------------------------------------------------------------------------
87
88// Current cursor, in order to hang on to cursor handle when setting the cursor
89// globally
90static wxCursor *gs_globalCursor = NULL;
91
92// ----------------------------------------------------------------------------
93// private classes
94// ----------------------------------------------------------------------------
95
96class wxCursorModule : public wxModule
97{
98public:
99 virtual bool OnInit()
100 {
101 gs_globalCursor = new wxCursor;
102
02b7b6b0 103 return true;
bfbd6dc1
VZ
104 }
105
106 virtual void OnExit()
107 {
108 delete gs_globalCursor;
109 gs_globalCursor = (wxCursor *)NULL;
110 }
111};
112
113// ============================================================================
114// implementation
115// ============================================================================
2bda0e17 116
0d0512bd
VZ
117// ----------------------------------------------------------------------------
118// wxCursorRefData
119// ----------------------------------------------------------------------------
120
eff4ffbf
VZ
121wxSize wxCursorRefData::ms_sizeStd;
122
123wxCoord wxCursorRefData::GetStandardWidth()
124{
125 if ( !ms_sizeStd.x )
126 ms_sizeStd.x = wxSystemSettings::GetMetric(wxSYS_CURSOR_X);
127
128 return ms_sizeStd.x;
129}
130
131wxCoord wxCursorRefData::GetStandardHeight()
2bda0e17 132{
eff4ffbf
VZ
133 if ( !ms_sizeStd.y )
134 ms_sizeStd.y = wxSystemSettings::GetMetric(wxSYS_CURSOR_Y);
135
136 return ms_sizeStd.y;
137}
138
139wxCursorRefData::wxCursorRefData(HCURSOR hcursor, bool destroy)
140{
141 m_hCursor = (WXHCURSOR)hcursor;
142
143 if ( m_hCursor )
144 {
145 m_width = GetStandardWidth();
146 m_height = GetStandardHeight();
147 }
0d0512bd 148
eff4ffbf 149 m_destroyCursor = destroy;
2bda0e17
KB
150}
151
0d0512bd 152void wxCursorRefData::Free()
2bda0e17 153{
032af30f
VZ
154 if ( m_hCursor )
155 {
7f0586ef 156#ifndef __WXWINCE__
032af30f
VZ
157 if ( m_destroyCursor )
158 ::DestroyCursor((HCURSOR)m_hCursor);
7f0586ef 159#endif
032af30f
VZ
160
161 m_hCursor = 0;
162 }
2bda0e17
KB
163}
164
0d0512bd 165// ----------------------------------------------------------------------------
2bda0e17 166// Cursors
0d0512bd
VZ
167// ----------------------------------------------------------------------------
168
169wxCursor::wxCursor()
2bda0e17
KB
170{
171}
172
461dae94 173#if wxUSE_IMAGE
eff4ffbf 174wxCursor::wxCursor(const wxImage& image)
bff4ec63 175{
eff4ffbf
VZ
176 // image has to be of the standard cursor size, otherwise we won't be able
177 // to create it
178 const int w = wxCursorRefData::GetStandardWidth();
179 const int h = wxCursorRefData::GetStandardHeight();
180
b737ad10
RR
181 int hotSpotX = image.GetOptionInt(wxIMAGE_OPTION_CUR_HOTSPOT_X);
182 int hotSpotY = image.GetOptionInt(wxIMAGE_OPTION_CUR_HOTSPOT_Y);
183 int image_w = image.GetWidth();
184 int image_h = image.GetHeight();
eff4ffbf 185
b737ad10
RR
186 wxASSERT_MSG( hotSpotX >= 0 && hotSpotX < image_w &&
187 hotSpotY >= 0 && hotSpotY < image_h,
7fd328a3 188 _T("invalid cursor hot spot coordinates") );
eff4ffbf 189
b737ad10
RR
190 wxImage imageSized(image); // final image of correct size
191
192 // if image is too small then place it in the center, resize it if too big
193 if ((w > image_w) && (h > image_h))
194 {
195 wxPoint offset((w - image_w)/2, (h - image_h)/2);
196 hotSpotX = hotSpotX + offset.x;
197 hotSpotY = hotSpotY + offset.y;
198
199 imageSized = image.Size(wxSize(w, h), offset);
200 }
201 else if ((w != image_w) || (h != image_h))
202 {
ccbb33c9
JS
203 hotSpotX = int(hotSpotX * double(w) / double(image_w));
204 hotSpotY = int(hotSpotY * double(h) / double(image_h));
b737ad10
RR
205
206 imageSized = image.Scale(w, h);
207 }
208
ccbb33c9 209 HCURSOR hcursor = wxBitmapToHCURSOR( wxBitmap(imageSized),
b737ad10
RR
210 hotSpotX, hotSpotY );
211
eff4ffbf
VZ
212 if ( !hcursor )
213 {
7fd328a3
VZ
214 wxLogWarning(_("Failed to create cursor."));
215 return;
eff4ffbf 216 }
7fd328a3
VZ
217
218 m_refData = new wxCursorRefData(hcursor, true /* delete it later */);
bff4ec63 219}
c7c441cc 220#endif // wxUSE_IMAGE
bff4ec63 221
0d0512bd
VZ
222wxCursor::wxCursor(const char WXUNUSED(bits)[],
223 int WXUNUSED(width),
224 int WXUNUSED(height),
225 int WXUNUSED(hotSpotX), int WXUNUSED(hotSpotY),
226 const char WXUNUSED(maskBits)[])
2bda0e17
KB
227{
228}
229
eff4ffbf 230// MicroWin doesn't have support needed for the other ctors
04ef50df 231#ifdef __WXMICROWIN__
2bda0e17 232
eff4ffbf
VZ
233wxCursor::wxCursor(const wxString& WXUNUSED(filename),
234 long WXUNUSED(kind),
235 int WXUNUSED(hotSpotX),
236 int WXUNUSED(hotSpotY))
237{
238}
239
240wxCursor::wxCursor(int WXUNUSED(cursor_type))
241{
242}
243
244#else // !__WXMICROWIN__
245
246wxCursor::wxCursor(const wxString& filename,
247 long kind,
248 int hotSpotX,
249 int hotSpotY)
250{
251 HCURSOR hcursor;
252 switch ( kind )
0d0512bd 253 {
eff4ffbf 254 case wxBITMAP_TYPE_CUR_RESOURCE:
e0a050e3 255 hcursor = ::LoadCursor(wxGetInstance(), filename.fn_str());
eff4ffbf
VZ
256 break;
257
7f0586ef 258#ifndef __WXWINCE__
eff4ffbf 259 case wxBITMAP_TYPE_CUR:
e0a050e3 260 hcursor = ::LoadCursorFromFile(filename.fn_str());
eff4ffbf 261 break;
7f0586ef 262#endif
eff4ffbf 263
677a9e28 264 case wxBITMAP_TYPE_ICO:
7fd328a3
VZ
265 hcursor = wxBitmapToHCURSOR
266 (
267 wxIcon(filename, wxBITMAP_TYPE_ICO),
268 hotSpotX,
269 hotSpotY
270 );
eff4ffbf
VZ
271 break;
272
273 case wxBITMAP_TYPE_BMP:
7fd328a3
VZ
274 hcursor = wxBitmapToHCURSOR
275 (
276 wxBitmap(filename, wxBITMAP_TYPE_BMP),
277 hotSpotX,
278 hotSpotY
279 );
eff4ffbf
VZ
280 break;
281
282 default:
283 wxFAIL_MSG( _T("unknown cursor resource type") );
284
285 hcursor = NULL;
0d0512bd 286 }
eff4ffbf
VZ
287
288 if ( hcursor )
0d0512bd 289 {
eff4ffbf 290 m_refData = new wxCursorRefData(hcursor, true /* delete it later */);
eff4ffbf 291 }
2bda0e17
KB
292}
293
7da60d7c
WS
294// Cursors by stock number
295wxCursor::wxCursor(int idCursor)
296{
1c6f2414
WS
297 // all wxWidgets standard cursors
298 static const struct StdCursor
299 {
300 // is this a standard Windows cursor?
301 bool isStd;
302
303 // the cursor name or id
304 LPCTSTR name;
305 } stdCursors[] =
306 {
307 { true, NULL }, // wxCURSOR_NONE
308 { true, IDC_ARROW }, // wxCURSOR_ARROW
309 { false, _T("WXCURSOR_RIGHT_ARROW") }, // wxCURSOR_RIGHT_ARROW
310 { false, _T("WXCURSOR_BULLSEYE") }, // wxCURSOR_BULLSEYE
311 { true, IDC_ARROW }, // WXCURSOR_CHAR
c8326d64 312
27e7e859
JS
313 // Displays as an I-beam on XP, so use a cursor file
314// { true, IDC_CROSS }, // WXCURSOR_CROSS
315 { false, _T("WXCURSOR_CROSS") }, // WXCURSOR_CROSS
fe384096
RD
316
317 // See special handling below for wxCURSOR_HAND
318// { false, _T("WXCURSOR_HAND") }, // wxCURSOR_HAND
319 { true, IDC_HAND }, // wxCURSOR_HAND
c8326d64 320
1c6f2414
WS
321 { true, IDC_IBEAM }, // WXCURSOR_IBEAM
322 { true, IDC_ARROW }, // WXCURSOR_LEFT_BUTTON
323 { false, _T("WXCURSOR_MAGNIFIER") }, // wxCURSOR_MAGNIFIER
324 { true, IDC_ARROW }, // WXCURSOR_MIDDLE_BUTTON
325 { true, IDC_NO }, // WXCURSOR_NO_ENTRY
326 { false, _T("WXCURSOR_PBRUSH") }, // wxCURSOR_PAINT_BRUSH
327 { false, _T("WXCURSOR_PENCIL") }, // wxCURSOR_PENCIL
328 { false, _T("WXCURSOR_PLEFT") }, // wxCURSOR_POINT_LEFT
329 { false, _T("WXCURSOR_PRIGHT") }, // wxCURSOR_POINT_RIGHT
330 { true, IDC_HELP }, // WXCURSOR_QUESTION_ARROW
331 { true, IDC_ARROW }, // WXCURSOR_RIGHT_BUTTON
332 { true, IDC_SIZENESW }, // WXCURSOR_SIZENESW
333 { true, IDC_SIZENS }, // WXCURSOR_SIZENS
334 { true, IDC_SIZENWSE }, // WXCURSOR_SIZENWSE
335 { true, IDC_SIZEWE }, // WXCURSOR_SIZEWE
336 { true, IDC_SIZEALL }, // WXCURSOR_SIZING
337 { false, _T("WXCURSOR_PBRUSH") }, // wxCURSOR_SPRAYCAN
338 { true, IDC_WAIT }, // WXCURSOR_WAIT
97000852 339 { true, IDC_WAIT }, // WXCURSOR_WATCH
1c6f2414
WS
340 { false, _T("WXCURSOR_BLANK") }, // wxCURSOR_BLANK
341 { true, IDC_APPSTARTING }, // wxCURSOR_ARROWWAIT
342
343 // no entry for wxCURSOR_MAX
344 };
345
346 wxCOMPILE_TIME_ASSERT( WXSIZEOF(stdCursors) == wxCURSOR_MAX,
347 CursorsIdArrayMismatch );
348
eff4ffbf
VZ
349 wxCHECK_RET( idCursor > 0 && (size_t)idCursor < WXSIZEOF(stdCursors),
350 _T("invalid cursor id in wxCursor() ctor") );
351
352 const StdCursor& stdCursor = stdCursors[idCursor];
fe384096 353 bool deleteLater = !stdCursor.isStd;
eff4ffbf
VZ
354
355 HCURSOR hcursor = ::LoadCursor(stdCursor.isStd ? NULL : wxGetInstance(),
356 stdCursor.name);
357
fe384096
RD
358 // IDC_HAND may not be available on some versions of Windows.
359 if ( !hcursor && idCursor == wxCURSOR_HAND)
360 {
361 hcursor = ::LoadCursor(wxGetInstance(), _T("WXCURSOR_HAND"));
362 deleteLater = true;
363 }
c8326d64 364
eff4ffbf 365 if ( !hcursor )
2bda0e17 366 {
eff4ffbf 367 wxLogLastError(_T("LoadCursor"));
2bda0e17 368 }
eff4ffbf 369 else
15dadf31 370 {
fe384096 371 m_refData = new wxCursorRefData(hcursor, deleteLater);
15dadf31 372 }
2bda0e17
KB
373}
374
eff4ffbf
VZ
375#endif // __WXMICROWIN__/!__WXMICROWIN__
376
0d0512bd 377wxCursor::~wxCursor()
2bda0e17 378{
2bda0e17
KB
379}
380
eff4ffbf
VZ
381// ----------------------------------------------------------------------------
382// other wxCursor functions
383// ----------------------------------------------------------------------------
384
eff4ffbf
VZ
385wxGDIImageRefData *wxCursor::CreateData() const
386{
387 return new wxCursorRefData;
388}
389
0d0512bd 390// ----------------------------------------------------------------------------
2bda0e17 391// Global cursor setting
0d0512bd
VZ
392// ----------------------------------------------------------------------------
393
bfbd6dc1 394const wxCursor *wxGetGlobalCursor()
2bda0e17 395{
bfbd6dc1
VZ
396 return gs_globalCursor;
397}
2bda0e17 398
bfbd6dc1
VZ
399void wxSetCursor(const wxCursor& cursor)
400{
401 if ( cursor.Ok() )
6bf57206 402 {
bfbd6dc1 403 ::SetCursor(GetHcursorOf(cursor));
6bf57206 404
bfbd6dc1
VZ
405 if ( gs_globalCursor )
406 *gs_globalCursor = cursor;
6bf57206 407 }
2bda0e17 408}