]> git.saurik.com Git - wxWidgets.git/blob - src/mgl/cursor.cpp
*do* generate the LEAVE events even when the mouse is captured
[wxWidgets.git] / src / mgl / cursor.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: cursor.cpp
3 // Purpose:
4 // Author: Vaclav Slavik
5 // Id: $Id$
6 // Copyright: (c) 2001-2002 SciTech Software, Inc. (www.scitechsoft.com)
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
9
10
11 #ifdef __GNUG__
12 #pragma implementation "cursor.h"
13 #endif
14
15 // For compilers that support precompilation, includes "wx.h".
16 #include "wx/wxprec.h"
17
18 #ifdef __BORLANDC__
19 #pragma hdrstop
20 #endif
21
22 #include "wx/cursor.h"
23 #include "wx/module.h"
24 #include "wx/utils.h"
25 #include "wx/log.h"
26 #include "wx/intl.h"
27 #include "wx/hashmap.h"
28
29 #include "wx/mgl/private.h"
30
31
32 //-----------------------------------------------------------------------------
33 // wxCursor
34 //-----------------------------------------------------------------------------
35
36 class wxCursorRefData: public wxObjectRefData
37 {
38 public:
39
40 wxCursorRefData();
41 ~wxCursorRefData();
42
43 MGLCursor *m_cursor;
44 };
45
46 wxCursorRefData::wxCursorRefData()
47 {
48 m_cursor = (MGLCursor*) NULL;
49 }
50
51 wxCursorRefData::~wxCursorRefData()
52 {
53 delete m_cursor;
54 }
55
56 #define M_CURSORDATA ((wxCursorRefData *)m_refData)
57
58 //-----------------------------------------------------------------------------
59
60 WX_DECLARE_HASH_MAP(int, wxCursor, wxIntegerHash, wxIntegerEqual, wxCursorsHash)
61
62 static wxCursorsHash *gs_cursorsHash = NULL;
63
64 IMPLEMENT_DYNAMIC_CLASS(wxCursor,wxObject)
65
66 wxCursor::wxCursor()
67 {
68 }
69
70 wxCursor::wxCursor(int cursorId)
71 {
72 if ( !gs_cursorsHash )
73 gs_cursorsHash = new wxCursorsHash;
74
75 if ( gs_cursorsHash->find(cursorId) != gs_cursorsHash->end() )
76 {
77 wxLogTrace(_T("mglcursor"), _T("cursor id %i fetched from cache"), cursorId);
78 *this = (*gs_cursorsHash)[cursorId];
79 return;
80 }
81
82 const char *cursorname = NULL;
83 m_refData = new wxCursorRefData();
84
85 switch (cursorId)
86 {
87 case wxCURSOR_ARROW: cursorname = "arrow.cur"; break;
88 case wxCURSOR_BULLSEYE: cursorname = "bullseye.cur"; break;
89 case wxCURSOR_CHAR: cursorname = "char.cur"; break;
90 case wxCURSOR_CROSS: cursorname = "cross.cur"; break;
91 case wxCURSOR_HAND: cursorname = "hand.cur"; break;
92 case wxCURSOR_IBEAM: cursorname = "ibeam.cur"; break;
93 case wxCURSOR_LEFT_BUTTON: cursorname = "leftbtn.cur"; break;
94 case wxCURSOR_MAGNIFIER: cursorname = "magnif1.cur"; break;
95 case wxCURSOR_MIDDLE_BUTTON: cursorname = "midbtn.cur"; break;
96 case wxCURSOR_NO_ENTRY: cursorname = "noentry.cur"; break;
97 case wxCURSOR_PAINT_BRUSH: cursorname = "pbrush.cur"; break;
98 case wxCURSOR_PENCIL: cursorname = "pencil.cur"; break;
99 case wxCURSOR_POINT_LEFT: cursorname = "pntleft.cur"; break;
100 case wxCURSOR_POINT_RIGHT: cursorname = "pntright.cur"; break;
101 case wxCURSOR_QUESTION_ARROW: cursorname = "query.cur"; break;
102 case wxCURSOR_RIGHT_BUTTON: cursorname = "rightbtn.cur"; break;
103 case wxCURSOR_SIZENESW: cursorname = "sizenesw.cur"; break;
104 case wxCURSOR_SIZENS: cursorname = "sizens.cur"; break;
105 case wxCURSOR_SIZENWSE: cursorname = "sizenwse.cur"; break;
106 case wxCURSOR_SIZEWE: cursorname = "sizewe.cur"; break;
107 case wxCURSOR_SIZING: cursorname = "size.cur"; break;
108 case wxCURSOR_SPRAYCAN: cursorname = "spraycan.cur"; break;
109 case wxCURSOR_WAIT: cursorname = "wait.cur"; break;
110 case wxCURSOR_WATCH: cursorname = "clock.cur"; break;
111 case wxCURSOR_BLANK: cursorname = "blank.cur"; break;
112
113 case wxCURSOR_NONE:
114 *this = wxNullCursor;
115 return;
116 break;
117
118 default:
119 wxFAIL_MSG(wxT("unsupported cursor type"));
120 break;
121 }
122
123 M_CURSORDATA->m_cursor = new MGLCursor(cursorname);
124
125 // if we cannot load arrow cursor, use MGL's default arrow cursor:
126 if ( !M_CURSORDATA->m_cursor->valid() && cursorId == wxCURSOR_ARROW )
127 {
128 delete M_CURSORDATA->m_cursor;
129 M_CURSORDATA->m_cursor = new MGLCursor(MGL_DEF_CURSOR);
130 }
131
132 if ( !M_CURSORDATA->m_cursor->valid() )
133 {
134 wxLogError(_("Couldn't create cursor."));
135 UnRef();
136 }
137 else
138 {
139 (*gs_cursorsHash)[cursorId] = *this;
140 wxLogTrace(_T("mglcursor"), _T("cursor id %i added to cache (%s)"),
141 cursorId, cursorname);
142 }
143 }
144
145 wxCursor::wxCursor(const char WXUNUSED(bits)[],
146 int WXUNUSED(width),
147 int WXUNUSED(height),
148 int WXUNUSED(hotSpotX), int WXUNUSED(hotSpotY),
149 const char WXUNUSED(maskBits)[],
150 wxColour * WXUNUSED(fg), wxColour * WXUNUSED(bg) )
151 {
152 //FIXME_MGL
153 }
154
155 wxCursor::wxCursor(const wxString& cursor_file,
156 long flags,
157 int hotSpotX, int hotSpotY)
158 {
159 if ( flags == wxBITMAP_TYPE_CUR || flags == wxBITMAP_TYPE_CUR_RESOURCE )
160 {
161 m_refData = new wxCursorRefData();
162 M_CURSORDATA->m_cursor = new MGLCursor(cursor_file.mb_str());
163 if ( !M_CURSORDATA->m_cursor->valid() )
164 {
165 wxLogError(_("Couldn't create cursor."));
166 UnRef();
167 }
168 }
169 else
170 {
171 wxLogError(wxT("Cannot load cursor resource of this type."));
172 }
173 }
174
175 wxCursor::wxCursor(const wxCursor &cursor)
176 {
177 Ref(cursor);
178 }
179
180 wxCursor::~wxCursor()
181 {
182 // wxObject unrefs data
183 }
184
185 wxCursor& wxCursor::operator = (const wxCursor& cursor)
186 {
187 if ( *this == cursor )
188 return (*this);
189 Ref(cursor);
190 return *this;
191 }
192
193 bool wxCursor::operator == (const wxCursor& cursor) const
194 {
195 return (m_refData == cursor.m_refData);
196 }
197
198 bool wxCursor::operator != (const wxCursor& cursor) const
199 {
200 return (m_refData != cursor.m_refData);
201 }
202
203 bool wxCursor::Ok() const
204 {
205 return (m_refData != NULL);
206 }
207
208 MGLCursor *wxCursor::GetMGLCursor() const
209 {
210 return M_CURSORDATA->m_cursor;
211 }
212
213
214
215 // ----------------------------------------------------------------------------
216 // Global cursor setting
217 // ----------------------------------------------------------------------------
218
219 static wxCursor gs_globalCursor = wxNullCursor;
220
221 void wxSetCursor(const wxCursor& cursor)
222 {
223 if ( cursor.Ok() )
224 {
225 if ( g_winMng )
226 MGL_wmSetGlobalCursor(g_winMng, *cursor.GetMGLCursor());
227 gs_globalCursor = cursor;
228 }
229 else
230 {
231 if ( g_winMng )
232 MGL_wmSetGlobalCursor(g_winMng, NULL);
233 gs_globalCursor = wxNullCursor;
234 }
235 }
236
237
238
239 //-----------------------------------------------------------------------------
240 // busy cursor routines
241 //-----------------------------------------------------------------------------
242
243 static wxCursor gs_savedCursor = wxNullCursor;
244 static int gs_busyCount = 0;
245
246 const wxCursor &wxBusyCursor::GetStoredCursor()
247 {
248 return gs_savedCursor;
249 }
250
251 const wxCursor wxBusyCursor::GetBusyCursor()
252 {
253 return gs_globalCursor;
254 }
255
256 void wxEndBusyCursor()
257 {
258 if ( --gs_busyCount > 0 ) return;
259
260 wxSetCursor(gs_savedCursor);
261 gs_savedCursor = wxNullCursor;
262 }
263
264 void wxBeginBusyCursor(wxCursor *cursor)
265 {
266 if ( gs_busyCount++ > 0 ) return;
267
268 wxASSERT_MSG( !gs_savedCursor.Ok(),
269 wxT("forgot to call wxEndBusyCursor, will leak memory") );
270
271 gs_savedCursor = gs_globalCursor;
272 if ( cursor->Ok() )
273 wxSetCursor(*cursor);
274 else
275 wxSetCursor(wxCursor(wxCURSOR_WAIT));
276 }
277
278 bool wxIsBusy()
279 {
280 return (gs_busyCount > 0);
281 }
282
283
284
285 //-----------------------------------------------------------------------------
286 // module - clean up code
287 //-----------------------------------------------------------------------------
288
289 class wxCursorModule : public wxModule
290 {
291 public:
292 virtual bool OnInit() { return TRUE; }
293
294 virtual void OnExit()
295 {
296 wxDELETE(gs_cursorsHash);
297 }
298
299 private:
300 DECLARE_DYNAMIC_CLASS(wxCursorModule)
301 };
302
303 IMPLEMENT_DYNAMIC_CLASS(wxCursorModule, wxModule)