]> git.saurik.com Git - wxWidgets.git/blame - src/generic/statusbr.cpp
Fixed a rare wxAuiFloatingFrame dtor crash on MSW using a registration mechanism...
[wxWidgets.git] / src / generic / statusbr.cpp
CommitLineData
c801d85f 1/////////////////////////////////////////////////////////////////////////////
a71d815b 2// Name: src/generic/statusbr.cpp
ed791986 3// Purpose: wxStatusBarGeneric class implementation
c801d85f
KB
4// Author: Julian Smart
5// Modified by:
6// Created: 01/02/97
7// RCS-ID: $Id$
6aa89a22 8// Copyright: (c) Julian Smart
65571936 9// Licence: wxWindows licence
c801d85f
KB
10/////////////////////////////////////////////////////////////////////////////
11
c801d85f
KB
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
14
15#ifdef __BORLANDC__
76b49cf4 16 #pragma hdrstop
c801d85f
KB
17#endif
18
1e6feb95 19#if wxUSE_STATUSBAR
ed791986 20
3304646d
WS
21#include "wx/statusbr.h"
22
c801d85f 23#ifndef WX_PRECOMP
76b49cf4
WS
24 #include "wx/settings.h"
25 #include "wx/dcclient.h"
a0125cfc 26 #include "wx/toplevel.h"
c801d85f
KB
27#endif
28
2b5f62a0 29#ifdef __WXGTK20__
53b6d7a2 30 #include <gtk/gtk.h>
2b5f62a0
VZ
31#endif
32
84f68c21
VZ
33// we only have to do it here when we use wxStatusBarGeneric in addition to the
34// standard wxStatusBar class, if wxStatusBarGeneric is the same as
35// wxStatusBar, then the corresponding IMPLEMENT_DYNAMIC_CLASS is already in
36// common/statbar.cpp
37#if defined(__WXMAC__) || \
38 (defined(wxUSE_NATIVE_STATUSBAR) && wxUSE_NATIVE_STATUSBAR)
b64a8473 39 #include "wx/generic/statusbr.h"
c801d85f 40
b64a8473
VZ
41 IMPLEMENT_DYNAMIC_CLASS(wxStatusBarGeneric, wxWindow)
42#endif // wxUSE_NATIVE_STATUSBAR
ed791986 43
ed791986
VZ
44BEGIN_EVENT_TABLE(wxStatusBarGeneric, wxWindow)
45 EVT_PAINT(wxStatusBarGeneric::OnPaint)
2b5f62a0
VZ
46 EVT_LEFT_DOWN(wxStatusBarGeneric::OnLeftDown)
47 EVT_RIGHT_DOWN(wxStatusBarGeneric::OnRightDown)
ed791986 48 EVT_SYS_COLOUR_CHANGED(wxStatusBarGeneric::OnSysColourChanged)
c801d85f 49END_EVENT_TABLE()
c801d85f
KB
50
51// Default status border dimensions
52#define wxTHICK_LINE_BORDER 2
c801d85f 53
390015c0 54void wxStatusBarGeneric::Init()
c801d85f 55{
48f7ffbe
WS
56 m_borderX = wxTHICK_LINE_BORDER;
57 m_borderY = wxTHICK_LINE_BORDER;
c801d85f
KB
58}
59
ed791986 60wxStatusBarGeneric::~wxStatusBarGeneric()
c801d85f 61{
c801d85f
KB
62}
63
ed791986 64bool wxStatusBarGeneric::Create(wxWindow *parent,
76880b87
RL
65 wxWindowID id,
66 long style,
67 const wxString& name)
c801d85f 68{
53b6d7a2 69 style |= wxTAB_TRAVERSAL | wxFULL_REPAINT_ON_RESIZE;
48f7ffbe
WS
70 if ( !wxWindow::Create(parent, id,
71 wxDefaultPosition, wxDefaultSize,
53b6d7a2 72 style, name) )
48f7ffbe 73 return false;
c801d85f 74
48f7ffbe
WS
75 // The status bar should have a themed background
76 SetThemeEnabled( true );
77
78 InitColours();
f90566f5 79
1ca21594 80#ifdef __WXPM__
48f7ffbe 81 SetFont(*wxSMALL_FONT);
1ca21594 82#endif
c801d85f 83
7422d7c4
VZ
84 wxCoord y;
85 {
86 // Set the height according to the font and the border size
87 wxClientDC dc(this);
88 dc.SetFont(GetFont());
89
90 dc.GetTextExtent(_T("X"), NULL, &y );
91 }
48f7ffbe 92 int height = (int)( (11*y)/10 + 2*GetBorderY());
71e03035 93
48f7ffbe 94 SetSize(wxDefaultCoord, wxDefaultCoord, wxDefaultCoord, height);
71e03035 95
48f7ffbe 96 SetFieldsCount(1);
8aa2cea1 97
48f7ffbe 98 return true;
c801d85f
KB
99}
100
595a9493
RD
101
102wxSize wxStatusBarGeneric::DoGetBestSize() const
103{
104 int width, height;
105
106 // best width is the width of the parent
107 GetParent()->GetClientSize(&width, NULL);
108
109 // best height is as calculated above in Create
110 wxClientDC dc((wxWindow*)this);
111 dc.SetFont(GetFont());
112 wxCoord y;
113 dc.GetTextExtent(_T("X"), NULL, &y );
114 height = (int)( (11*y)/10 + 2*GetBorderY());
115
116 return wxSize(width, height);
ca65c044 117}
595a9493 118
cc56206f 119void wxStatusBarGeneric::SetFieldsCount(int number, const int *widths)
c801d85f 120{
390015c0 121 wxASSERT_MSG( number >= 0, _T("negative number of fields in wxStatusBar?") );
76880b87 122
893f25f2
JS
123 int i;
124 for(i = m_nFields; i < number; ++i)
76880b87
RL
125 m_statusStrings.Add( wxEmptyString );
126
893f25f2 127 for (i = m_nFields - 1; i >= number; --i)
390015c0 128 m_statusStrings.RemoveAt(i);
76880b87 129
f9a961e1
DS
130 // forget the old cached pixel widths
131 m_widthsAbs.Empty();
132
d64b1c76
DS
133 wxStatusBarBase::SetFieldsCount(number, widths);
134
76880b87
RL
135 wxASSERT_MSG( m_nFields == (int)m_statusStrings.GetCount(),
136 _T("This really should never happen, can we do away with m_nFields here?") );
c801d85f
KB
137}
138
ed791986 139void wxStatusBarGeneric::SetStatusText(const wxString& text, int number)
c801d85f 140{
633d67bb
VZ
141 wxCHECK_RET( (number >= 0) && (number < m_nFields),
142 _T("invalid status bar field index") );
c801d85f 143
87ff599d
JS
144 wxString oldText = m_statusStrings[number];
145 if (oldText != text)
146 {
147 m_statusStrings[number] = text;
c801d85f 148
87ff599d
JS
149 wxRect rect;
150 GetFieldRect(number, rect);
e715f4e7 151
e5e0c727
VZ
152 Refresh(true, &rect);
153
154 // it's common to show some text in the status bar before starting a
155 // relatively lengthy operation, ensure that the text is shown to the
156 // user immediately and not after the lengthy operation end
157 Update();
87ff599d 158 }
c801d85f
KB
159}
160
ed791986 161wxString wxStatusBarGeneric::GetStatusText(int n) const
c801d85f 162{
ed791986
VZ
163 wxCHECK_MSG( (n >= 0) && (n < m_nFields), wxEmptyString,
164 _T("invalid status bar field index") );
165
c801d85f
KB
166 return m_statusStrings[n];
167}
168
ed791986 169void wxStatusBarGeneric::SetStatusWidths(int n, const int widths_field[])
c801d85f 170{
633d67bb
VZ
171 // only set status widths, when n == number of statuswindows
172 wxCHECK_RET( n == m_nFields, _T("status bar field count mismatch") );
173
174 // delete the old widths in any case - this function may be used to reset
175 // the widths to the default (all equal)
1f361cdd
MB
176 // MBN: this is incompatible with at least wxMSW and wxMAC and not
177 // documented, but let's keep it for now
178 ReinitWidths();
633d67bb 179
390015c0
VZ
180 // forget the old cached pixel widths
181 m_widthsAbs.Empty();
182
633d67bb
VZ
183 if ( !widths_field )
184 {
185 // not an error, see the comment above
f8a4fa4c 186 Refresh();
633d67bb
VZ
187 return;
188 }
189
1f361cdd 190 wxStatusBarBase::SetStatusWidths(n, widths_field);
c801d85f
KB
191}
192
7422d7c4
VZ
193bool wxStatusBarGeneric::ShowsSizeGrip() const
194{
195 if ( !HasFlag(wxST_SIZEGRIP) )
196 return false;
197
198 wxTopLevelWindow * const
199 tlw = wxDynamicCast(wxGetTopLevelParent(GetParent()), wxTopLevelWindow);
200 return tlw && !tlw->IsMaximized() && tlw->HasFlag(wxRESIZE_BORDER);
201}
202
ed791986 203void wxStatusBarGeneric::OnPaint(wxPaintEvent& WXUNUSED(event) )
c801d85f 204{
2b5f62a0 205 wxPaintDC dc(this);
c801d85f 206
2b5f62a0
VZ
207#ifdef __WXGTK20__
208 // Draw grip first
7422d7c4 209 if ( ShowsSizeGrip() )
2b5f62a0
VZ
210 {
211 int width, height;
212 GetClientSize(&width, &height);
7422d7c4 213
fc2d4209
RR
214 if (GetLayoutDirection() == wxLayout_RightToLeft)
215 {
216 gtk_paint_resize_grip( m_widget->style,
08f53168 217 GTKGetDrawingWindow(),
fc2d4209
RR
218 (GtkStateType) GTK_WIDGET_STATE (m_widget),
219 NULL,
220 m_widget,
221 "statusbar",
222 GDK_WINDOW_EDGE_SOUTH_WEST,
223 2, 2, height-2, height-4 );
224 }
225 else
226 {
227 gtk_paint_resize_grip( m_widget->style,
08f53168 228 GTKGetDrawingWindow(),
2b5f62a0
VZ
229 (GtkStateType) GTK_WIDGET_STATE (m_widget),
230 NULL,
231 m_widget,
232 "statusbar",
233 GDK_WINDOW_EDGE_SOUTH_EAST,
fc2d4209
RR
234 width-height-2, 2, height-2, height-4 );
235 }
2b5f62a0 236 }
7422d7c4 237#endif // __WXGTK20__
5b3ed311 238
48f7ffbe 239 if (GetFont().Ok())
2b5f62a0 240 dc.SetFont(GetFont());
8aa2cea1 241
48f7ffbe 242 dc.SetBackgroundMode(wxTRANSPARENT);
c801d85f 243
48f7ffbe 244 for (int i = 0; i < m_nFields; i ++)
2b5f62a0 245 DrawField(dc, i);
c801d85f
KB
246}
247
ed791986 248void wxStatusBarGeneric::DrawFieldText(wxDC& dc, int i)
c801d85f 249{
48f7ffbe 250 int leftMargin = 2;
c801d85f 251
48f7ffbe
WS
252 wxRect rect;
253 GetFieldRect(i, rect);
c801d85f 254
48f7ffbe 255 wxString text(GetStatusText(i));
c801d85f 256
01399dcd 257 wxCoord x = 0, y = 0;
c801d85f 258
48f7ffbe 259 dc.GetTextExtent(text, &x, &y);
c801d85f 260
48f7ffbe
WS
261 int xpos = rect.x + leftMargin;
262 int ypos = (int) (((rect.height - y) / 2 ) + rect.y + 0.5) ;
ed791986 263
7c74e7fe 264#if defined( __WXGTK__ ) || defined(__WXMAC__)
48f7ffbe
WS
265 xpos++;
266 ypos++;
92976ab6 267#endif
c801d85f 268
48f7ffbe 269 dc.SetClippingRegion(rect.x, rect.y, rect.width, rect.height);
c801d85f 270
48f7ffbe 271 dc.DrawText(text, xpos, ypos);
c801d85f 272
48f7ffbe 273 dc.DestroyClippingRegion();
c801d85f
KB
274}
275
ed791986 276void wxStatusBarGeneric::DrawField(wxDC& dc, int i)
c801d85f 277{
2b5f62a0
VZ
278 wxRect rect;
279 GetFieldRect(i, rect);
c801d85f 280
c2919ab3
VZ
281 int style = wxSB_NORMAL;
282 if (m_statusStyles)
283 style = m_statusStyles[i];
c801d85f 284
c2919ab3
VZ
285 if (style != wxSB_FLAT)
286 {
287 // Draw border
288 // For wxSB_NORMAL:
289 // Have grey background, plus 3-d border -
290 // One black rectangle.
291 // Inside this, left and top sides - dark grey. Bottom and right -
292 // white.
293 // Reverse it for wxSB_RAISED
294
295 dc.SetPen((style == wxSB_RAISED) ? m_mediumShadowPen : m_hilightPen);
296
297 #ifndef __WXPM__
298
299 // Right and bottom lines
300 dc.DrawLine(rect.x + rect.width, rect.y,
301 rect.x + rect.width, rect.y + rect.height);
302 dc.DrawLine(rect.x + rect.width, rect.y + rect.height,
303 rect.x, rect.y + rect.height);
304
305 dc.SetPen((style == wxSB_RAISED) ? m_hilightPen : m_mediumShadowPen);
306
307 // Left and top lines
308 dc.DrawLine(rect.x, rect.y + rect.height,
309 rect.x, rect.y);
310 dc.DrawLine(rect.x, rect.y,
311 rect.x + rect.width, rect.y);
312 #else
313
314 dc.DrawLine(rect.x + rect.width, rect.height + 2,
315 rect.x, rect.height + 2);
316 dc.DrawLine(rect.x + rect.width, rect.y,
317 rect.x + rect.width, rect.y + rect.height);
318
319 dc.SetPen((style == wxSB_RAISED) ? m_hilightPen : m_mediumShadowPen);
320 dc.DrawLine(rect.x, rect.y,
321 rect.x + rect.width, rect.y);
322 dc.DrawLine(rect.x, rect.y + rect.height,
323 rect.x, rect.y);
b34590eb
DW
324
325#endif
c2919ab3 326 }
c801d85f 327
f51f94ea 328 DrawFieldText(dc, i);
c801d85f
KB
329}
330
331 // Get the position and size of the field's internal bounding rectangle
ed791986 332bool wxStatusBarGeneric::GetFieldRect(int n, wxRect& rect) const
c801d85f 333{
ca65c044 334 wxCHECK_MSG( (n >= 0) && (n < m_nFields), false,
390015c0 335 _T("invalid status bar field index") );
c801d85f 336
390015c0
VZ
337 // FIXME: workarounds for OS/2 bugs have nothing to do here (VZ)
338 int width, height;
28be2e8a 339#ifdef __WXPM__
390015c0 340 GetSize(&width, &height);
28be2e8a 341#else
390015c0 342 GetClientSize(&width, &height);
28be2e8a 343#endif
c801d85f 344
2b5f62a0
VZ
345 // we cache m_widthsAbs between calls and recompute it if client
346 // width has changed (or when it is initially empty)
347 if ( m_widthsAbs.IsEmpty() || (m_lastClientWidth != width) )
c801d85f 348 {
390015c0
VZ
349 wxConstCast(this, wxStatusBarGeneric)->
350 m_widthsAbs = CalculateAbsWidths(width);
2b5f62a0
VZ
351 // remember last width for which we have recomputed the widths in pixels
352 wxConstCast(this, wxStatusBarGeneric)->
353 m_lastClientWidth = width;
c801d85f 354 }
c801d85f 355
390015c0
VZ
356 rect.x = 0;
357 for ( int i = 0; i < n; i++ )
358 {
359 rect.x += m_widthsAbs[i];
c801d85f 360 }
c801d85f 361
390015c0
VZ
362 rect.x += m_borderX;
363 rect.y = m_borderY;
c801d85f 364
390015c0
VZ
365 rect.width = m_widthsAbs[n] - 2*m_borderX;
366 rect.height = height - 2*m_borderY;
c801d85f 367
ca65c044 368 return true;
c801d85f
KB
369}
370
371// Initialize colours
ed791986 372void wxStatusBarGeneric::InitColours()
c801d85f
KB
373{
374 // Shadow colours
a71d815b 375#if defined(__WXMSW__) || defined(__WXMAC__)
a756f210 376 wxColour mediumShadowColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DSHADOW));
c801d85f
KB
377 m_mediumShadowPen = wxPen(mediumShadowColour, 1, wxSOLID);
378
a756f210 379 wxColour hilightColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DHILIGHT));
c801d85f 380 m_hilightPen = wxPen(hilightColour, 1, wxSOLID);
b34590eb 381#elif defined(__WXPM__)
e715f4e7 382 m_mediumShadowPen = wxPen(wxColour(127, 127, 127), 1, wxSOLID);
b0fa2187 383 m_hilightPen = *wxWHITE_PEN;
dd7d1435 384
b0fa2187
PC
385 SetBackgroundColour(*wxLIGHT_GREY);
386 SetForegroundColour(*wxBLACK);
c801d85f 387#else
b0fa2187
PC
388 m_mediumShadowPen = *wxGREY_PEN;
389 m_hilightPen = *wxWHITE_PEN;
c801d85f 390#endif
c801d85f
KB
391}
392
393// Responds to colour changes, and passes event on to children.
ed791986 394void wxStatusBarGeneric::OnSysColourChanged(wxSysColourChangedEvent& event)
c801d85f
KB
395{
396 InitColours();
c801d85f
KB
397
398 // Propagate the event to the non-top-level children
399 wxWindow::OnSysColourChanged(event);
400}
401
ed791986
VZ
402void wxStatusBarGeneric::SetMinHeight(int height)
403{
404 // check that this min height is not less than minimal height for the
405 // current font
406 wxClientDC dc(this);
407 wxCoord y;
2b5f62a0 408 dc.GetTextExtent( wxT("X"), NULL, &y );
ed791986
VZ
409
410 if ( height > (11*y)/10 )
411 {
422d0ff0 412 SetSize(wxDefaultCoord, wxDefaultCoord, wxDefaultCoord, height + 2*m_borderY);
ed791986
VZ
413 }
414}
415
2b5f62a0 416void wxStatusBarGeneric::OnLeftDown(wxMouseEvent& event)
390015c0 417{
2b5f62a0
VZ
418#ifdef __WXGTK20__
419 int width, height;
420 GetClientSize(&width, &height);
8aa2cea1 421
7422d7c4 422 if ( ShowsSizeGrip() && (event.GetX() > width-height) )
2b5f62a0
VZ
423 {
424 GtkWidget *ancestor = gtk_widget_get_toplevel( m_widget );
425
426 if (!GTK_IS_WINDOW (ancestor))
427 return;
390015c0 428
08f53168 429 GdkWindow *source = GTKGetDrawingWindow();
2b5f62a0
VZ
430
431 int org_x = 0;
432 int org_y = 0;
433 gdk_window_get_origin( source, &org_x, &org_y );
434
fc2d4209
RR
435 if (GetLayoutDirection() == wxLayout_RightToLeft)
436 {
437 gtk_window_begin_resize_drag (GTK_WINDOW (ancestor),
438 GDK_WINDOW_EDGE_SOUTH_WEST,
439 1,
440 org_x - event.GetX() + GetSize().x ,
441 org_y + event.GetY(),
442 0);
443 }
444 else
445 {
446 gtk_window_begin_resize_drag (GTK_WINDOW (ancestor),
2b5f62a0
VZ
447 GDK_WINDOW_EDGE_SOUTH_EAST,
448 1,
8aa2cea1
RD
449 org_x + event.GetX(),
450 org_y + event.GetY(),
2b5f62a0 451 0);
fc2d4209 452 }
2b5f62a0
VZ
453 }
454 else
455 {
ca65c044 456 event.Skip( true );
2b5f62a0
VZ
457 }
458#else
ca65c044 459 event.Skip( true );
2b5f62a0
VZ
460#endif
461}
462
463void wxStatusBarGeneric::OnRightDown(wxMouseEvent& event)
464{
465#ifdef __WXGTK20__
466 int width, height;
467 GetClientSize(&width, &height);
8aa2cea1 468
7422d7c4 469 if ( ShowsSizeGrip() && (event.GetX() > width-height) )
2b5f62a0
VZ
470 {
471 GtkWidget *ancestor = gtk_widget_get_toplevel( m_widget );
472
473 if (!GTK_IS_WINDOW (ancestor))
474 return;
475
08f53168 476 GdkWindow *source = GTKGetDrawingWindow();
2b5f62a0
VZ
477
478 int org_x = 0;
479 int org_y = 0;
480 gdk_window_get_origin( source, &org_x, &org_y );
8aa2cea1 481
2b5f62a0
VZ
482 gtk_window_begin_move_drag (GTK_WINDOW (ancestor),
483 2,
8aa2cea1
RD
484 org_x + event.GetX(),
485 org_y + event.GetY(),
2b5f62a0
VZ
486 0);
487 }
488 else
489 {
ca65c044 490 event.Skip( true );
2b5f62a0
VZ
491 }
492#else
ca65c044 493 event.Skip( true );
2b5f62a0 494#endif
390015c0
VZ
495}
496
1e6feb95 497#endif // wxUSE_STATUSBAR