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
.t_str()); 
 243         case wxBITMAP_TYPE_CUR
: 
 244             hcursor 
= ::LoadCursorFromFile(filename
.t_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
;