]> git.saurik.com Git - wxWidgets.git/blame - src/generic/statusbr.cpp
added missing refresh when changing focus in wxListCtrl, added test for it in the...
[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$
8// Copyright: (c) Julian Smart and Markus Holzem
ed791986 9// Licence: wxWindows license
c801d85f
KB
10/////////////////////////////////////////////////////////////////////////////
11
12#ifdef __GNUG__
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
412e4edf 32#include "wx/statusbr.h"
c801d85f 33
412e4edf
VZ
34// with wxUSE_NATIVE_STATUSBAR it is not included from wx/statusbr.h
35#include "wx/generic/statusbr.h"
c801d85f 36
ed791986
VZ
37IMPLEMENT_DYNAMIC_CLASS(wxStatusBarGeneric, wxWindow)
38
39#if !defined(__WIN32__) || !wxUSE_NATIVE_STATUSBAR
40 IMPLEMENT_DYNAMIC_CLASS(wxStatusBar, wxStatusBarGeneric)
41#endif // Win32 && wxUSE_NATIVE_STATUSBAR
c801d85f 42
ed791986
VZ
43BEGIN_EVENT_TABLE(wxStatusBarGeneric, wxWindow)
44 EVT_PAINT(wxStatusBarGeneric::OnPaint)
45 EVT_SYS_COLOUR_CHANGED(wxStatusBarGeneric::OnSysColourChanged)
c801d85f 46END_EVENT_TABLE()
c801d85f
KB
47
48// Default status border dimensions
49#define wxTHICK_LINE_BORDER 2
50#define wxTHICK_LINE_WIDTH 1
51
ed791986 52wxStatusBarGeneric::wxStatusBarGeneric()
c801d85f 53{
c67daf87
UR
54 m_statusWidths = (int *) NULL;
55 m_statusStrings = (wxString *) NULL;
c801d85f
KB
56 m_nFields = 0;
57 m_borderX = wxTHICK_LINE_BORDER;
58 m_borderY = wxTHICK_LINE_BORDER;
59}
60
ed791986 61wxStatusBarGeneric::~wxStatusBarGeneric()
c801d85f 62{
f51f94ea
VZ
63# ifdef __WXMSW__
64 SetFont(wxNullFont);
65# endif // MSW
66
67 if ( m_statusWidths )
68 delete[] m_statusWidths;
69 if ( m_statusStrings )
70 delete[] m_statusStrings;
c801d85f
KB
71}
72
ed791986
VZ
73bool wxStatusBarGeneric::Create(wxWindow *parent,
74 wxWindowID id,
75 long style,
76 const wxString& name)
c801d85f 77{
c67daf87
UR
78 m_statusWidths = (int *) NULL;
79 m_statusStrings = (wxString *) NULL;
c801d85f
KB
80 m_nFields = 0;
81 m_borderX = wxTHICK_LINE_BORDER;
82 m_borderY = wxTHICK_LINE_BORDER;
83
ed791986
VZ
84 bool success = wxWindow::Create(parent, id,
85 wxDefaultPosition, wxDefaultSize,
86 style | wxTAB_TRAVERSAL, name);
c801d85f
KB
87
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
KB
92 InitColours();
93
94 SetFont(m_defaultStatusBarFont);
95
96 return success;
97}
98
cc56206f 99void wxStatusBarGeneric::SetFieldsCount(int number, const int *widths)
c801d85f 100{
633d67bb
VZ
101 if ( number != m_nFields )
102 {
103 m_nFields = number;
c801d85f 104
f51f94ea 105 delete[] m_statusStrings;
633d67bb 106 m_statusStrings = new wxString[number];
c801d85f 107
633d67bb
VZ
108#if 0 // VZ: what is this for?
109 int i;
110 for (i = 0; i < number; i++)
111 m_statusStrings[i] = "";
112#endif
113 }
c801d85f 114
633d67bb 115 SetStatusWidths(number, widths);
c801d85f
KB
116}
117
ed791986 118void wxStatusBarGeneric::SetStatusText(const wxString& text, int number)
c801d85f 119{
633d67bb
VZ
120 wxCHECK_RET( (number >= 0) && (number < m_nFields),
121 _T("invalid status bar field index") );
c801d85f 122
633d67bb 123 m_statusStrings[number] = text;
c801d85f 124
633d67bb 125 Refresh();
c801d85f
KB
126}
127
ed791986 128wxString wxStatusBarGeneric::GetStatusText(int n) const
c801d85f 129{
ed791986
VZ
130 wxCHECK_MSG( (n >= 0) && (n < m_nFields), wxEmptyString,
131 _T("invalid status bar field index") );
132
c801d85f
KB
133 return m_statusStrings[n];
134}
135
ed791986 136void wxStatusBarGeneric::SetStatusWidths(int n, const int widths_field[])
c801d85f 137{
633d67bb
VZ
138 // only set status widths, when n == number of statuswindows
139 wxCHECK_RET( n == m_nFields, _T("status bar field count mismatch") );
140
141 // delete the old widths in any case - this function may be used to reset
142 // the widths to the default (all equal)
143 delete [] m_statusWidths;
144
145 if ( !widths_field )
146 {
147 // not an error, see the comment above
148 m_statusWidths = (int *)NULL;
149
150 return;
151 }
152
153 int i;
154
155 // VZ: this doesn't do anything as is_variable is unused later
156#if 0
c801d85f
KB
157 // when one window (minimum) is variable (width <= 0)
158 bool is_variable = FALSE;
c801d85f
KB
159 for (i = 0; i < m_nFields; i++)
160 {
633d67bb
VZ
161 if (widths_field[i] <= 0)
162 is_variable = TRUE;
c801d85f 163 }
633d67bb 164#endif // 0
c801d85f
KB
165
166 // set widths
167 m_statusWidths = new int[n];
168 for (i = 0; i < m_nFields; i++)
169 {
633d67bb 170 m_statusWidths[i] = widths_field[i];
c801d85f 171 }
c801d85f
KB
172}
173
ed791986 174void wxStatusBarGeneric::OnPaint(wxPaintEvent& WXUNUSED(event) )
c801d85f
KB
175{
176 wxPaintDC dc(this);
177
5b3ed311 178
c801d85f 179 int i;
c0ed460c
JS
180 if ( GetFont().Ok() )
181 dc.SetFont(GetFont());
c801d85f
KB
182 dc.SetBackgroundMode(wxTRANSPARENT);
183
28be2e8a 184#ifdef __WXPM__
b34590eb
DW
185 wxColour vColor;
186
702c190d 187 vColor.InitFromName("GREY");
b34590eb 188 ::WinFillRect(dc.m_hPS, &dc.m_vRclPaint, vColor.GetPixel());
28be2e8a
DW
189#endif
190
c801d85f 191 for ( i = 0; i < m_nFields; i ++ )
f51f94ea 192 DrawField(dc, i);
e97f20a0 193
28be2e8a
DW
194#ifdef __WXMSW__
195 dc.SetFont(wxNullFont);
196#endif // MSW
c801d85f
KB
197}
198
ed791986 199void wxStatusBarGeneric::DrawFieldText(wxDC& dc, int i)
c801d85f
KB
200{
201 int leftMargin = 2;
202
16e93305 203 wxRect rect;
c801d85f
KB
204 GetFieldRect(i, rect);
205
206 wxString text(GetStatusText(i));
207
208 long x, y;
209
210 dc.GetTextExtent(text, &x, &y);
211
212 int xpos = rect.x + leftMargin;
213 int ypos = (int) (((rect.height - y) / 2 ) + rect.y + 0.5) ;
ed791986 214
7c74e7fe 215#if defined( __WXGTK__ ) || defined(__WXMAC__)
92976ab6
RR
216 xpos++;
217 ypos++;
218#endif
c801d85f
KB
219
220 dc.SetClippingRegion(rect.x, rect.y, rect.width, rect.height);
221
222 dc.DrawText(text, xpos, ypos);
223
224 dc.DestroyClippingRegion();
225}
226
ed791986 227void wxStatusBarGeneric::DrawField(wxDC& dc, int i)
c801d85f 228{
16e93305 229 wxRect rect;
c801d85f
KB
230 GetFieldRect(i, rect);
231
232 // Draw border
233 // Have grey background, plus 3-d border -
234 // One black rectangle.
235 // Inside this, left and top sides - dark grey. Bottom and right -
236 // white.
237
f51f94ea 238 dc.SetPen(m_hilightPen);
c801d85f 239
b34590eb
DW
240#ifndef __WXPM__
241
c801d85f
KB
242 // Right and bottom white lines
243 dc.DrawLine(rect.x + rect.width, rect.y,
244 rect.x + rect.width, rect.y + rect.height);
245 dc.DrawLine(rect.x + rect.width, rect.y + rect.height,
f51f94ea 246 rect.x, rect.y + rect.height);
c801d85f 247
f51f94ea 248 dc.SetPen(m_mediumShadowPen);
c801d85f
KB
249
250 // Left and top grey lines
251 dc.DrawLine(rect.x, rect.y + rect.height,
f51f94ea 252 rect.x, rect.y);
c801d85f 253 dc.DrawLine(rect.x, rect.y,
f51f94ea 254 rect.x + rect.width, rect.y);
b34590eb
DW
255#else
256 // Right
702c190d
DW
257 dc.DrawLine(rect.x + rect.width, rect.y,
258 rect.x + rect.width, rect.y + rect.height + 2);
b34590eb 259 dc.SetPen(m_mediumShadowPen);
702c190d
DW
260 dc.DrawLine(rect.x + rect.width + 1, rect.y,
261 rect.x + rect.width + 1, rect.y + rect.height + 2);
262 dc.DrawLine(rect.x + rect.width + 2, rect.y,
263 rect.x + rect.width + 2, rect.y + rect.height + 2);
b34590eb 264 // Top
702c190d
DW
265 dc.DrawLine(rect.x + rect.width + 2, rect.y,
266 rect.x - 2, rect.y);
267 dc.DrawLine(rect.x + rect.width + 1, rect.y - 1,
268 rect.x - 2, rect.y - 1);
b34590eb 269 dc.SetPen(m_hilightPen);
702c190d
DW
270 dc.DrawLine(rect.x + rect.width, rect.y - 2,
271 rect.x - 2, rect.y - 2);
b34590eb
DW
272
273#endif
c801d85f 274
f51f94ea 275 DrawFieldText(dc, i);
c801d85f
KB
276}
277
278 // Get the position and size of the field's internal bounding rectangle
ed791986 279bool wxStatusBarGeneric::GetFieldRect(int n, wxRect& rect) const
c801d85f 280{
ed791986
VZ
281 wxCHECK_MSG( (n >= 0) && (n < m_nFields), FALSE,
282 _T("invalid status bar field index") );
c801d85f
KB
283
284 int width, height;
28be2e8a
DW
285#ifdef __WXPM__
286 GetSize(&width, &height);
287#else
c801d85f 288 GetClientSize(&width, &height);
28be2e8a 289#endif
c801d85f
KB
290
291 int i;
292 int sum_of_nonvar = 0;
293 int num_of_var = 0;
294 bool do_same_width = FALSE;
295
296 int fieldWidth = 0;
297 int fieldPosition = 0;
298
299 if (m_statusWidths)
300 {
301 // if sum(not variable Windows) > c_width - (20 points per variable_window)
302 // then do_same_width = TRUE;
303 for (i = 0; i < m_nFields; i++)
304 {
305 if (m_statusWidths[i] > 0) sum_of_nonvar += m_statusWidths[i];
306 else num_of_var++;
307 }
308 if (sum_of_nonvar > (width - 20*num_of_var)) do_same_width = TRUE;
309 }
310 else do_same_width = TRUE;
311 if (do_same_width)
312 {
313 for (i = 0; i < m_nFields; i++)
314 {
315 fieldWidth = (int)(width/m_nFields);
f51f94ea
VZ
316 fieldPosition = i*fieldWidth;
317 if ( i == n )
318 break;
c801d85f
KB
319 }
320 }
321 else // no_same_width
322 {
323 int *tempwidth = new int[m_nFields];
324 int temppos = 0;
325 for (i = 0; i < m_nFields; i++)
326 {
327 if (m_statusWidths[i] > 0) tempwidth[i] = m_statusWidths[i];
328 else tempwidth[i] = (width - sum_of_nonvar) / num_of_var;
329 }
330 for (i = 0; i < m_nFields; i++)
331 {
f51f94ea
VZ
332 fieldWidth = tempwidth[i];
333 fieldPosition = temppos;
c801d85f 334
f51f94ea 335 temppos += tempwidth[i];
c801d85f 336
f51f94ea
VZ
337 if ( i == n )
338 break;
c801d85f
KB
339 }
340 delete [] tempwidth;
341 }
342
343 rect.x = fieldPosition + wxTHICK_LINE_BORDER;
f51f94ea 344 rect.y = wxTHICK_LINE_BORDER;
c801d85f 345
f51f94ea
VZ
346 rect.width = fieldWidth - 2 * wxTHICK_LINE_BORDER ;
347 rect.height = height - 2 * wxTHICK_LINE_BORDER ;
c801d85f 348
f51f94ea 349 return TRUE;
c801d85f
KB
350}
351
352// Initialize colours
ed791986 353void wxStatusBarGeneric::InitColours()
c801d85f
KB
354{
355 // Shadow colours
5b3ed311 356#if defined(__WIN95__)
c801d85f
KB
357 wxColour mediumShadowColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DSHADOW));
358 m_mediumShadowPen = wxPen(mediumShadowColour, 1, wxSOLID);
359
360 wxColour hilightColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DHILIGHT));
361 m_hilightPen = wxPen(hilightColour, 1, wxSOLID);
b34590eb
DW
362#elif defined(__WXPM__)
363 m_mediumShadowPen = wxPen("LIGHT GREY", 1, wxSOLID);
364 m_hilightPen = wxPen("WHITE", 1, wxSOLID);
c801d85f
KB
365#else
366 m_mediumShadowPen = wxPen("GREY", 1, wxSOLID);
367 m_hilightPen = wxPen("WHITE", 1, wxSOLID);
368#endif
369
16c1f7f3 370 m_defaultStatusBarFont = wxSystemSettings::GetSystemFont(wxSYS_DEFAULT_GUI_FONT);
c801d85f
KB
371 SetBackgroundColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DFACE));
372}
373
374// Responds to colour changes, and passes event on to children.
ed791986 375void wxStatusBarGeneric::OnSysColourChanged(wxSysColourChangedEvent& event)
c801d85f
KB
376{
377 InitColours();
378 Refresh();
379
380 // Propagate the event to the non-top-level children
381 wxWindow::OnSysColourChanged(event);
382}
383
ed791986
VZ
384void wxStatusBarGeneric::SetMinHeight(int height)
385{
386 // check that this min height is not less than minimal height for the
387 // current font
388 wxClientDC dc(this);
389 wxCoord y;
390 dc.GetTextExtent( _T("X"), NULL, &y );
391
392 if ( height > (11*y)/10 )
393 {
394 SetSize(-1, -1, -1, height + 2*m_borderY);
395 }
396}
397
1e6feb95 398#endif // wxUSE_STATUSBAR