1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
12 #pragma implementation "cursor.h"
15 #include "wx/cursor.h"
22 //-----------------------------------------------------------------------------
24 //-----------------------------------------------------------------------------
26 extern void wxapp_install_idle_handler();
29 //-----------------------------------------------------------------------------
31 //-----------------------------------------------------------------------------
33 class wxCursorRefData
: public wxObjectRefData
43 wxCursorRefData::wxCursorRefData()
45 m_cursor
= (GdkCursor
*) NULL
;
48 wxCursorRefData::~wxCursorRefData()
50 if (m_cursor
) gdk_cursor_destroy( m_cursor
);
53 //-----------------------------------------------------------------------------
55 #define M_CURSORDATA ((wxCursorRefData *)m_refData)
57 IMPLEMENT_DYNAMIC_CLASS(wxCursor
,wxObject
)
64 wxCursor::wxCursor( int cursorId
)
66 m_refData
= new wxCursorRefData();
68 GdkCursorType gdk_cur
= GDK_LEFT_PTR
;
71 case wxCURSOR_DEFAULT
: gdk_cur
= GDK_LEFT_PTR
; break;
72 case wxCURSOR_HAND
: gdk_cur
= GDK_HAND1
; break;
73 case wxCURSOR_CROSS
: gdk_cur
= GDK_CROSSHAIR
; break;
74 case wxCURSOR_SIZEWE
: gdk_cur
= GDK_SB_H_DOUBLE_ARROW
; break;
75 case wxCURSOR_SIZENS
: gdk_cur
= GDK_SB_V_DOUBLE_ARROW
; break;
76 case wxCURSOR_ARROWWAIT
:
78 case wxCURSOR_WATCH
: gdk_cur
= GDK_WATCH
; break;
79 case wxCURSOR_SIZING
: gdk_cur
= GDK_SIZING
; break;
80 case wxCURSOR_SPRAYCAN
: gdk_cur
= GDK_SPRAYCAN
; break;
81 case wxCURSOR_IBEAM
: gdk_cur
= GDK_XTERM
; break;
82 case wxCURSOR_PENCIL
: gdk_cur
= GDK_PENCIL
; break;
83 case wxCURSOR_NO_ENTRY
: gdk_cur
= GDK_PIRATE
; break;
84 case wxCURSOR_SIZENWSE
:
85 case wxCURSOR_SIZENESW
: gdk_cur
= GDK_FLEUR
; break;
86 case wxCURSOR_QUESTION_ARROW
: gdk_cur
= GDK_QUESTION_ARROW
; break;
87 case wxCURSOR_PAINT_BRUSH
: gdk_cur
= GDK_SPRAYCAN
; break;
88 case wxCURSOR_MAGNIFIER
: gdk_cur
= GDK_PLUS
; break;
89 case wxCURSOR_CHAR
: gdk_cur
= GDK_XTERM
; break;
90 case wxCURSOR_LEFT_BUTTON
: gdk_cur
= GDK_LEFTBUTTON
; break;
91 case wxCURSOR_MIDDLE_BUTTON
: gdk_cur
= GDK_MIDDLEBUTTON
; break;
92 case wxCURSOR_RIGHT_BUTTON
: gdk_cur
= GDK_RIGHTBUTTON
; break;
93 case wxCURSOR_BULLSEYE
: gdk_cur
= GDK_TARGET
; break;
95 case wxCURSOR_POINT_LEFT
: gdk_cur
= GDK_SB_LEFT_ARROW
; break;
96 case wxCURSOR_POINT_RIGHT
: gdk_cur
= GDK_SB_RIGHT_ARROW
; break;
98 case wxCURSOR_DOUBLE_ARROW: gdk_cur = GDK_DOUBLE_ARROW; break;
99 case wxCURSOR_CROSS_REVERSE: gdk_cur = GDK_CROSS_REVERSE; break;
100 case wxCURSOR_BASED_ARROW_UP: gdk_cur = GDK_BASED_ARROW_UP; break;
101 case wxCURSOR_BASED_ARROW_DOWN: gdk_cur = GDK_BASED_ARROW_DOWN; break;
104 wxFAIL_MSG(wxT("unsupported cursor type"));
105 // will use the standard one
111 M_CURSORDATA
->m_cursor
= gdk_cursor_new( gdk_cur
);
114 extern GtkWidget
*wxGetRootWindow();
116 wxCursor::wxCursor(const char bits
[], int width
, int height
,
117 int hotSpotX
, int hotSpotY
,
118 const char maskBits
[], wxColour
*fg
, wxColour
*bg
)
126 if (hotSpotX
< 0 || hotSpotX
>= width
)
128 if (hotSpotY
< 0 || hotSpotY
>= height
)
131 GdkBitmap
*data
= gdk_bitmap_create_from_data( wxGetRootWindow()->window
, (gchar
*) bits
, width
, height
);
132 GdkBitmap
*mask
= gdk_bitmap_create_from_data( wxGetRootWindow()->window
, (gchar
*) maskBits
, width
, height
);
134 m_refData
= new wxCursorRefData
;
135 M_CURSORDATA
->m_cursor
= gdk_cursor_new_from_pixmap(
136 data
, mask
, fg
->GetColor(), bg
->GetColor(),
137 hotSpotX
, hotSpotY
);
139 gdk_bitmap_unref( data
);
140 gdk_bitmap_unref( mask
);
144 wxCursor::wxCursor( const wxCursor
&cursor
)
150 wxCursor::wxCursor( const wxImage
& image
)
152 unsigned char * rgbBits
= image
.GetData();
153 int w
= image
.GetWidth() ;
154 int h
= image
.GetHeight() ;
155 bool bHasMask
= image
.HasMask() ;
156 int imagebitcount
= (w
*h
)/8;
158 unsigned char r
, g
, b
;
159 unsigned char * bits
= new unsigned char [imagebitcount
];
160 unsigned char * maskBits
= new unsigned char [imagebitcount
];
162 int i
,j
, i8
; unsigned char c
, cMask
;
163 for (i
=0; i
<imagebitcount
; i
++)
171 // possible overflow if we do the summation first ?
172 c
= rgbBits
[(i8
+j
)*3]/3 + rgbBits
[(i8
+j
)*3+1]/3 + rgbBits
[(i8
+j
)*3+2]/3 ;
173 //if average value is > mid grey
175 bits
[i
] = bits
[i
] | cMask
;
181 r
= image
.GetMaskRed() ;
182 g
= image
.GetMaskGreen() ;
183 b
= image
.GetMaskBlue() ;
185 for (i
=0; i
<imagebitcount
; i
++)
193 if (rgbBits
[(i8
+j
)*3] != r
|| rgbBits
[(i8
+j
)*3+1] != g
|| rgbBits
[(i8
+j
)*3+2] != b
)
194 maskBits
[i
] = maskBits
[i
] | cMask
;
201 for (i
=0; i
<imagebitcount
; i
++)
204 //find the most frequent color(s)
205 //it seems a waste of effort to copy the image
206 //but otherwise we need to remove the const modifier ??
207 wxImage tmpImage
= image
.Copy();
212 unsigned long keyMaskColor
= 0;
213 if (bHasMask
) keyMaskColor
= (r
<< 16) | (g
<< 8) | b
;
215 tmpImage
.ComputeHistogram( hTable
);
217 long MostFreqCol
= 0 ; long nMost
= 0;
218 long NextFreqCol
= 0 ; long nNext
= 0;
222 while ((node
= hTable
.Next()) != NULL
)
224 wxHNode
*hnode
= (wxHNode
*) node
->GetData();
225 value
= hnode
->value
;
226 key
= node
->GetKeyInteger() ;
227 if (!bHasMask
|| (key
!= keyMaskColor
) )
244 wxColour fg
= wxColour ( (unsigned char)(MostFreqCol
>> 16),
245 (unsigned char)(MostFreqCol
>> 8),
246 (unsigned char)(MostFreqCol
) ) ;
248 wxColour bg
= wxColour ( (unsigned char)(NextFreqCol
>> 16),
249 (unsigned char)(NextFreqCol
>> 8),
250 (unsigned char)(NextFreqCol
) ) ;
257 if (image
.HasOption(wxCUR_HOTSPOT_X
))
258 hotSpotX
= image
.GetOptionInt(wxCUR_HOTSPOT_X
);
259 if (image
.HasOption(wxCUR_HOTSPOT_Y
))
260 hotSpotY
= image
.GetOptionInt(wxCUR_HOTSPOT_Y
);
262 if (hotSpotX
< 0 || hotSpotX
>= w
)
264 if (hotSpotY
< 0 || hotSpotY
>= h
)
267 GdkBitmap
*data
= gdk_bitmap_create_from_data( wxGetRootWindow()->window
, (gchar
*) bits
,
269 GdkBitmap
*mask
= gdk_bitmap_create_from_data( wxGetRootWindow()->window
, (gchar
*) maskBits
,
272 m_refData
= new wxCursorRefData
;
273 M_CURSORDATA
->m_cursor
= gdk_cursor_new_from_pixmap(
274 data
, mask
, fg
.GetColor(), bg
.GetColor(),
275 hotSpotX
, hotSpotY
);
277 gdk_bitmap_unref( data
);
278 gdk_bitmap_unref( mask
);
285 wxCursor::~wxCursor()
289 wxCursor
& wxCursor::operator = ( const wxCursor
& cursor
)
299 bool wxCursor::operator == ( const wxCursor
& cursor
) const
301 return m_refData
== cursor
.m_refData
;
304 bool wxCursor::operator != ( const wxCursor
& cursor
) const
306 return m_refData
!= cursor
.m_refData
;
309 bool wxCursor::Ok() const
311 return (m_refData
!= NULL
);
314 GdkCursor
*wxCursor::GetCursor() const
316 return M_CURSORDATA
->m_cursor
;
319 //-----------------------------------------------------------------------------
320 // busy cursor routines
321 //-----------------------------------------------------------------------------
323 extern wxCursor g_globalCursor
;
325 static wxCursor gs_savedCursor
;
326 static int gs_busyCount
= 0;
328 const wxCursor
&wxBusyCursor::GetStoredCursor()
330 return gs_savedCursor
;
333 const wxCursor
wxBusyCursor::GetBusyCursor()
335 return wxCursor(wxCURSOR_WATCH
);
338 void wxEndBusyCursor()
340 if (--gs_busyCount
> 0)
343 wxSetCursor( gs_savedCursor
);
344 gs_savedCursor
= wxNullCursor
;
347 wxTheApp
->SendIdleEvents();
350 void wxBeginBusyCursor( wxCursor
*WXUNUSED(cursor
) )
352 if (gs_busyCount
++ > 0)
355 wxASSERT_MSG( !gs_savedCursor
.Ok(),
356 wxT("forgot to call wxEndBusyCursor, will leak memory") );
358 gs_savedCursor
= g_globalCursor
;
360 wxSetCursor( wxCursor(wxCURSOR_WATCH
) );
363 wxTheApp
->SendIdleEvents();
370 return gs_busyCount
> 0;
373 void wxSetCursor( const wxCursor
& cursor
)
376 wxapp_install_idle_handler();
378 g_globalCursor
= cursor
;