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