]> git.saurik.com Git - wxWidgets.git/blame - src/msw/statbr95.cpp
Dialogs can now show a question mark for context-sensitive help.
[wxWidgets.git] / src / msw / statbr95.cpp
CommitLineData
2bda0e17
KB
1///////////////////////////////////////////////////////////////////////////////
2// Name: msw/statbr95.cpp
3// Purpose: native implementation of wxStatusBar
4// Author: Vadim Zeitlin
ed791986 5// Modified by:
2bda0e17
KB
6// Created: 04.04.98
7// RCS-ID: $Id$
8// Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9// Licence: wxWindows license
10///////////////////////////////////////////////////////////////////////////////
11
cfe780fb
JS
12#ifdef __GNUG__
13#pragma implementation "statbr95.h"
14#endif
15
2bda0e17
KB
16// for compilers that support precompilation, includes "wx.h".
17#include "wx/wxprec.h"
18
19#ifdef __BORLANDC__
20 #pragma hdrstop
21#endif
22
23#ifndef WX_PRECOMP
24 #include "wx/setup.h"
25 #include "wx/frame.h"
26 #include "wx/settings.h"
27 #include "wx/dcclient.h"
28#endif
29
ed791986 30#if defined(__WIN95__) && wxUSE_NATIVE_STATUSBAR
2432b92d 31
608eeb0f 32#include "wx/intl.h"
2bda0e17 33#include "wx/log.h"
ed791986 34#include "wx/statusbr.h"
2bda0e17 35
42e69d6b
VZ
36#include "wx/msw/private.h"
37#include <windowsx.h>
2bda0e17 38
c42404a5
VZ
39#if !(defined(__GNUWIN32_OLD__) || defined(__TWIN32__))
40 #include <commctrl.h>
ce3ed50d
JS
41#endif
42
ed791986
VZ
43// ----------------------------------------------------------------------------
44// wxWindows macros
45// ----------------------------------------------------------------------------
2bda0e17 46
ed791986
VZ
47IMPLEMENT_DYNAMIC_CLASS(wxStatusBar95, wxWindow);
48IMPLEMENT_DYNAMIC_CLASS(wxStatusBar, wxStatusBar95)
2bda0e17 49
2bda0e17
KB
50// ----------------------------------------------------------------------------
51// macros
52// ----------------------------------------------------------------------------
53
54// windowsx.h and commctrl.h don't define those, so we do it here
55#define StatusBar_SetParts(h, n, w) SendMessage(h, SB_SETPARTS, (WPARAM)n, (LPARAM)w)
837e5743 56#define StatusBar_SetText(h, n, t) SendMessage(h, SB_SETTEXT, (WPARAM)n, (LPARAM)(LPCTSTR)t)
2bda0e17 57#define StatusBar_GetTextLen(h, n) LOWORD(SendMessage(h, SB_GETTEXTLENGTH, (WPARAM)n, 0))
837e5743 58#define StatusBar_GetText(h, n, s) LOWORD(SendMessage(h, SB_GETTEXT, (WPARAM)n, (LPARAM)(LPTSTR)s))
2bda0e17 59
ed791986
VZ
60// ----------------------------------------------------------------------------
61//
62// ----------------------------------------------------------------------------
63
457e6c54
JS
64// static WNDPROC gs_wndprocStatBar = NULL;
65static WXFARPROC gs_wndprocStatBar = (WXFARPROC) NULL;
ed791986
VZ
66
67LRESULT APIENTRY wxStatusBarProc(HWND hwnd,
68 UINT message,
69 WPARAM wParam,
70 LPARAM lParam)
71{
bae41109
RD
72 switch (message) {
73 case WM_COMMAND:
f6bcfd97
BP
74 case WM_DRAWITEM:
75 case WM_MEASUREITEM:
bae41109
RD
76 case WM_SIZE:
77 case WM_MOVE:
78 case WM_MOUSEMOVE:
bae41109
RD
79 case WM_LBUTTONUP:
80 case WM_LBUTTONDBLCLK:
81 case WM_RBUTTONDOWN:
82 case WM_RBUTTONUP:
83 case WM_RBUTTONDBLCLK:
84 case WM_MBUTTONDOWN:
85 case WM_MBUTTONUP:
86 case WM_MBUTTONDBLCLK:
87 wxStatusBar95 *sb = (wxStatusBar95 *)GetWindowLong(hwnd, GWL_USERDATA);
88 sb->MSWWindowProc(message, wParam, lParam);
89 break;
ed791986
VZ
90 }
91
66e33344 92 return ::CallWindowProc(CASTWNDPROC gs_wndprocStatBar, hwnd, message, wParam, lParam);
ed791986 93}
2bda0e17
KB
94
95// ============================================================================
96// implementation
97// ============================================================================
98
99// ----------------------------------------------------------------------------
100// wxStatusBar95 class
101// ----------------------------------------------------------------------------
102
103wxStatusBar95::wxStatusBar95()
104{
105 SetParent(NULL);
106 m_hWnd = 0;
107 m_windowId = 0;
108}
109
ed791986
VZ
110bool wxStatusBar95::Create(wxWindow *parent,
111 wxWindowID id,
112 long style,
113 const wxString& name)
2bda0e17 114{
cbc66a27
VZ
115 wxCHECK_MSG( parent, FALSE, wxT("status bar must have a parent") );
116
ed791986 117 SetName(name);
cbc66a27 118 SetWindowStyleFlag(style);
ed791986
VZ
119 SetParent(parent);
120
cbc66a27
VZ
121 parent->AddChild(this);
122
123 m_windowId = id == -1 ? NewControlId() : id;
ed791986 124
f6bcfd97 125 DWORD wstyle = WS_CHILD | WS_VISIBLE /* | WS_CLIPSIBLINGS */ ;
43b5058d
VZ
126
127 // setting SBARS_SIZEGRIP is perfectly useless: it's always on by default
128 // (at least in the version of comctl32.dll I'm using), and the only way to
129 // turn it off is to use CCS_TOP style - as we position the status bar
130 // manually anyhow (see DoMoveWindow), use CCS_TOP style if wxST_SIZEGRIP
131 // is not given
132 if ( !(style & wxST_SIZEGRIP) )
133 {
134 wstyle |= CCS_TOP;
135 }
136 else
137 {
138 // may be some versions of comctl32.dll do need it - anyhow, it won't
139 // do any harm
ed791986 140 wstyle |= SBARS_SIZEGRIP;
43b5058d 141 }
ed791986
VZ
142
143 m_hWnd = (WXHWND)CreateStatusWindow(wstyle,
144 wxEmptyString,
145 GetHwndOf(parent),
146 m_windowId);
147 if ( m_hWnd == 0 )
148 {
149 wxLogSysError(_("Failed to create a status bar."));
150
151 return FALSE;
152 }
2bda0e17 153
c6e7d14f
RD
154 SetFieldsCount(1);
155
cbc66a27
VZ
156 // we can't subclass this window as usual because the status bar window
157 // proc processes WM_SIZE and WM_PAINT specially
ed791986
VZ
158 // SubclassWin(m_hWnd);
159
cbc66a27
VZ
160 // but we want to process the messages from it still, so do custom
161 // subclassing here
457e6c54 162 gs_wndprocStatBar = (WXFARPROC)GetWindowLong(GetHwnd(), GWL_WNDPROC);
ed791986
VZ
163 SetWindowLong(GetHwnd(), GWL_WNDPROC, (LONG)wxStatusBarProc);
164 SetWindowLong(GetHwnd(), GWL_USERDATA, (LONG)this);
2bda0e17 165
ed791986
VZ
166 return TRUE;
167}
2bda0e17 168
ed791986
VZ
169wxStatusBar95::~wxStatusBar95()
170{
171 delete [] m_statusWidths;
2bda0e17
KB
172}
173
8b9518ee 174void wxStatusBar95::CopyFieldsWidth(const int widths[])
2bda0e17
KB
175{
176 if (widths && !m_statusWidths)
177 m_statusWidths = new int[m_nFields];
178
179 if ( widths != NULL ) {
180 for ( int i = 0; i < m_nFields; i++ )
181 m_statusWidths[i] = widths[i];
182 }
183 else {
184 delete [] m_statusWidths;
185 m_statusWidths = NULL;
186 }
187}
188
790ad94f 189void wxStatusBar95::SetFieldsCount(int nFields, const int *widths)
2bda0e17 190{
f6bcfd97
BP
191 // this is a Windows limitation
192 wxASSERT_MSG( (nFields > 0) && (nFields < 255), _T("too many fields") );
2bda0e17 193
f6bcfd97 194 m_nFields = nFields;
2bda0e17 195
f6bcfd97
BP
196 CopyFieldsWidth(widths);
197 SetFieldsWidth();
2bda0e17
KB
198}
199
8b9518ee 200void wxStatusBar95::SetStatusWidths(int n, const int widths[])
2bda0e17 201{
f6bcfd97 202 wxASSERT_MSG( n == m_nFields, _T("field number mismatch") );
2bda0e17 203
f6bcfd97
BP
204 CopyFieldsWidth(widths);
205 SetFieldsWidth();
2bda0e17
KB
206}
207
208void wxStatusBar95::SetFieldsWidth()
209{
3175c712
VZ
210 if ( !m_nFields )
211 return;
212
ed791986
VZ
213 int aBorders[3];
214 SendMessage(GetHwnd(), SB_GETBORDERS, 0, (LPARAM)aBorders);
2bda0e17 215
ed791986 216 int extraWidth = aBorders[2]; // space between fields
2bda0e17 217
ed791986 218 int *pWidths = new int[m_nFields];
2bda0e17 219
ed791986
VZ
220 int nWindowWidth, y;
221 GetClientSize(&nWindowWidth, &y);
2bda0e17 222
ed791986
VZ
223 if ( m_statusWidths == NULL ) {
224 // default: all fields have the same width
225 int nWidth = nWindowWidth / m_nFields;
226 for ( int i = 0; i < m_nFields; i++ )
227 pWidths[i] = (i + 1) * nWidth;
2bda0e17 228 }
ed791986
VZ
229 else {
230 // -1 doesn't mean the same thing for wxWindows and Win32, recalc
231 int nTotalWidth = 0,
232 nVarCount = 0,
233 i;
234 for ( i = 0; i < m_nFields; i++ ) {
235 if ( m_statusWidths[i] == -1 )
236 nVarCount++;
237 else
238 nTotalWidth += m_statusWidths[i] + extraWidth;
239 }
240
241 if ( nVarCount == 0 ) {
242 wxFAIL_MSG( _T("at least one field must be of variable width") );
243
244 nVarCount++;
245 }
246
247 int nVarWidth = (nWindowWidth - nTotalWidth) / nVarCount;
248
249 // do fill the array
250 int nCurPos = 0;
251 for ( i = 0; i < m_nFields; i++ ) {
252 if ( m_statusWidths[i] == -1 )
253 nCurPos += nVarWidth;
254 else
255 nCurPos += m_statusWidths[i] + extraWidth;
256 pWidths[i] = nCurPos;
257 }
2bda0e17 258 }
2bda0e17 259
ed791986
VZ
260 if ( !StatusBar_SetParts(GetHwnd(), m_nFields, pWidths) ) {
261 wxLogLastError(wxT("StatusBar_SetParts"));
262 }
2bda0e17 263
ed791986 264 delete [] pWidths;
2bda0e17
KB
265}
266
debe6624 267void wxStatusBar95::SetStatusText(const wxString& strText, int nField)
2bda0e17 268{
ed791986
VZ
269 wxCHECK_RET( (nField >= 0) && (nField < m_nFields),
270 _T("invalid statusbar field index") );
271
272 if ( !StatusBar_SetText(GetHwnd(), nField, strText) ) {
223d09f6 273 wxLogLastError(wxT("StatusBar_SetText"));
2bda0e17
KB
274 }
275}
276
277wxString wxStatusBar95::GetStatusText(int nField) const
278{
ed791986
VZ
279 wxCHECK_MSG( (nField >= 0) && (nField < m_nFields), wxEmptyString,
280 _T("invalid statusbar field index") );
2bda0e17 281
ed791986
VZ
282 wxString str;
283 int len = StatusBar_GetTextLen(GetHwnd(), nField);
b7346a70
JS
284 if (len > 0)
285 {
ed791986
VZ
286 StatusBar_GetText(GetHwnd(), nField, str.GetWriteBuf(len));
287 str.UngetWriteBuf();
b7346a70 288 }
2bda0e17
KB
289 return str;
290}
291
ed791986
VZ
292int wxStatusBar95::GetBorderX() const
293{
294 int aBorders[3];
295 SendMessage(GetHwnd(), SB_GETBORDERS, 0, (LPARAM)aBorders);
296
297 return aBorders[0];
298}
299
300int wxStatusBar95::GetBorderY() const
301{
302 int aBorders[3];
303 SendMessage(GetHwnd(), SB_GETBORDERS, 0, (LPARAM)aBorders);
304
305 return aBorders[1];
306}
307
308void wxStatusBar95::SetMinHeight(int height)
309{
310 SendMessage(GetHwnd(), SB_SETMINHEIGHT, height + 2*GetBorderY(), 0);
311
312 // have to send a (dummy) WM_SIZE to redraw it now
313 SendMessage(GetHwnd(), WM_SIZE, 0, 0);
314}
315
316bool wxStatusBar95::GetFieldRect(int i, wxRect& rect) const
317{
318 wxCHECK_MSG( (i >= 0) && (i < m_nFields), FALSE,
319 _T("invalid statusbar field index") );
320
321 RECT r;
322 if ( !::SendMessage(GetHwnd(), SB_GETRECT, i, (LPARAM)&r) )
323 {
f6bcfd97 324 wxLogLastError(wxT("SendMessage(SB_GETRECT)"));
ed791986
VZ
325 }
326
327 wxCopyRECTToRect(r, rect);
328
329 return TRUE;
330}
331
cbc66a27 332void wxStatusBar95::DoMoveWindow(int x, int y, int width, int height)
2bda0e17 333{
43b5058d
VZ
334 // the status bar wnd proc must be forwarded the WM_SIZE message whenever
335 // the stat bar position/size is changed because it normally positions the
336 // control itself along bottom or top side of the parent window - failing
337 // to do so will result in nasty visual effects
338 FORWARD_WM_SIZE(GetHwnd(), SIZE_RESTORED, x, y, SendMessage);
339
340 // but now, when the standard status bar wnd proc did all it wanted to do,
341 // move the status bar to its correct location - usually this call may be
342 // omitted because for normal status bars (positioned along the bottom
343 // edge) the position is already set correctly, but if the user wants to
344 // position them in some exotic location, this is really needed
345 wxWindow::DoMoveWindow(x, y, width, height);
346
347 // adjust fields widths to the new size
348 SetFieldsWidth();
2bda0e17
KB
349}
350
ed791986 351#endif // __WIN95__ && wxUSE_NATIVE_STATUSBAR
ba681060 352