1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: msw/statbr95.cpp
3 // Purpose: native implementation of wxStatusBar
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows license
10 ///////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "statbr95.h"
16 // for compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
26 #include "wx/settings.h"
27 #include "wx/dcclient.h"
30 #if defined(__WIN95__) && wxUSE_NATIVE_STATUSBAR
34 #include "wx/statusbr.h"
36 #include "wx/msw/private.h"
39 #if !(defined(__GNUWIN32_OLD__) || defined(__TWIN32__))
43 // ----------------------------------------------------------------------------
45 // ----------------------------------------------------------------------------
47 IMPLEMENT_DYNAMIC_CLASS(wxStatusBar95
, wxWindow
);
48 IMPLEMENT_DYNAMIC_CLASS(wxStatusBar
, wxStatusBar95
)
50 // ----------------------------------------------------------------------------
52 // ----------------------------------------------------------------------------
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)
56 #define StatusBar_SetText(h, n, t) SendMessage(h, SB_SETTEXT, (WPARAM)n, (LPARAM)(LPCTSTR)t)
57 #define StatusBar_GetTextLen(h, n) LOWORD(SendMessage(h, SB_GETTEXTLENGTH, (WPARAM)n, 0))
58 #define StatusBar_GetText(h, n, s) LOWORD(SendMessage(h, SB_GETTEXT, (WPARAM)n, (LPARAM)(LPTSTR)s))
60 // ----------------------------------------------------------------------------
62 // ----------------------------------------------------------------------------
64 // static WNDPROC gs_wndprocStatBar = NULL;
65 static WXFARPROC gs_wndprocStatBar
= (WXFARPROC
) NULL
;
67 LRESULT APIENTRY
wxStatusBarProc(HWND hwnd
,
72 if ( message
== WM_COMMAND
)
74 wxStatusBar95
*sb
= (wxStatusBar95
*)GetWindowLong(hwnd
, GWL_USERDATA
);
75 sb
->MSWWindowProc(message
, wParam
, lParam
);
78 return ::CallWindowProc(CASTWNDPROC gs_wndprocStatBar
, hwnd
, message
, wParam
, lParam
);
81 // ============================================================================
83 // ============================================================================
85 // ----------------------------------------------------------------------------
86 // wxStatusBar95 class
87 // ----------------------------------------------------------------------------
89 wxStatusBar95::wxStatusBar95()
96 bool wxStatusBar95::Create(wxWindow
*parent
,
101 wxCHECK_MSG( parent
, FALSE
, wxT("status bar must have a parent") );
104 SetWindowStyleFlag(style
);
107 parent
->AddChild(this);
109 m_windowId
= id
== -1 ? NewControlId() : id
;
111 DWORD wstyle
= WS_CHILD
| WS_VISIBLE
;
113 // setting SBARS_SIZEGRIP is perfectly useless: it's always on by default
114 // (at least in the version of comctl32.dll I'm using), and the only way to
115 // turn it off is to use CCS_TOP style - as we position the status bar
116 // manually anyhow (see DoMoveWindow), use CCS_TOP style if wxST_SIZEGRIP
118 if ( !(style
& wxST_SIZEGRIP
) )
124 // may be some versions of comctl32.dll do need it - anyhow, it won't
126 wstyle
|= SBARS_SIZEGRIP
;
129 m_hWnd
= (WXHWND
)CreateStatusWindow(wstyle
,
135 wxLogSysError(_("Failed to create a status bar."));
142 // we can't subclass this window as usual because the status bar window
143 // proc processes WM_SIZE and WM_PAINT specially
144 // SubclassWin(m_hWnd);
146 // but we want to process the messages from it still, so do custom
148 gs_wndprocStatBar
= (WXFARPROC
)GetWindowLong(GetHwnd(), GWL_WNDPROC
);
149 SetWindowLong(GetHwnd(), GWL_WNDPROC
, (LONG
)wxStatusBarProc
);
150 SetWindowLong(GetHwnd(), GWL_USERDATA
, (LONG
)this);
155 wxStatusBar95::~wxStatusBar95()
157 delete [] m_statusWidths
;
160 void wxStatusBar95::CopyFieldsWidth(const int widths
[])
162 if (widths
&& !m_statusWidths
)
163 m_statusWidths
= new int[m_nFields
];
165 if ( widths
!= NULL
) {
166 for ( int i
= 0; i
< m_nFields
; i
++ )
167 m_statusWidths
[i
] = widths
[i
];
170 delete [] m_statusWidths
;
171 m_statusWidths
= NULL
;
175 void wxStatusBar95::SetFieldsCount(int nFields
, const int *widths
)
177 // this is Windows limitation
178 wxASSERT_MSG( (nFields
> 0) && (nFields
< 255), _T("too many fields") );
182 CopyFieldsWidth(widths
);
186 void wxStatusBar95::SetStatusWidths(int n
, const int widths
[])
188 wxASSERT_MSG( n
== m_nFields
, _T("field number mismatch") );
190 CopyFieldsWidth(widths
);
194 void wxStatusBar95::SetFieldsWidth()
200 SendMessage(GetHwnd(), SB_GETBORDERS
, 0, (LPARAM
)aBorders
);
202 int extraWidth
= aBorders
[2]; // space between fields
204 int *pWidths
= new int[m_nFields
];
207 GetClientSize(&nWindowWidth
, &y
);
209 if ( m_statusWidths
== NULL
) {
210 // default: all fields have the same width
211 int nWidth
= nWindowWidth
/ m_nFields
;
212 for ( int i
= 0; i
< m_nFields
; i
++ )
213 pWidths
[i
] = (i
+ 1) * nWidth
;
216 // -1 doesn't mean the same thing for wxWindows and Win32, recalc
220 for ( i
= 0; i
< m_nFields
; i
++ ) {
221 if ( m_statusWidths
[i
] == -1 )
224 nTotalWidth
+= m_statusWidths
[i
] + extraWidth
;
227 if ( nVarCount
== 0 ) {
228 wxFAIL_MSG( _T("at least one field must be of variable width") );
233 int nVarWidth
= (nWindowWidth
- nTotalWidth
) / nVarCount
;
237 for ( i
= 0; i
< m_nFields
; i
++ ) {
238 if ( m_statusWidths
[i
] == -1 )
239 nCurPos
+= nVarWidth
;
241 nCurPos
+= m_statusWidths
[i
] + extraWidth
;
242 pWidths
[i
] = nCurPos
;
246 if ( !StatusBar_SetParts(GetHwnd(), m_nFields
, pWidths
) ) {
247 wxLogLastError(wxT("StatusBar_SetParts"));
253 void wxStatusBar95::SetStatusText(const wxString
& strText
, int nField
)
255 wxCHECK_RET( (nField
>= 0) && (nField
< m_nFields
),
256 _T("invalid statusbar field index") );
258 if ( !StatusBar_SetText(GetHwnd(), nField
, strText
) ) {
259 wxLogLastError(wxT("StatusBar_SetText"));
263 wxString
wxStatusBar95::GetStatusText(int nField
) const
265 wxCHECK_MSG( (nField
>= 0) && (nField
< m_nFields
), wxEmptyString
,
266 _T("invalid statusbar field index") );
269 int len
= StatusBar_GetTextLen(GetHwnd(), nField
);
272 StatusBar_GetText(GetHwnd(), nField
, str
.GetWriteBuf(len
));
278 int wxStatusBar95::GetBorderX() const
281 SendMessage(GetHwnd(), SB_GETBORDERS
, 0, (LPARAM
)aBorders
);
286 int wxStatusBar95::GetBorderY() const
289 SendMessage(GetHwnd(), SB_GETBORDERS
, 0, (LPARAM
)aBorders
);
294 void wxStatusBar95::SetMinHeight(int height
)
296 SendMessage(GetHwnd(), SB_SETMINHEIGHT
, height
+ 2*GetBorderY(), 0);
298 // have to send a (dummy) WM_SIZE to redraw it now
299 SendMessage(GetHwnd(), WM_SIZE
, 0, 0);
302 bool wxStatusBar95::GetFieldRect(int i
, wxRect
& rect
) const
304 wxCHECK_MSG( (i
>= 0) && (i
< m_nFields
), FALSE
,
305 _T("invalid statusbar field index") );
308 if ( !::SendMessage(GetHwnd(), SB_GETRECT
, i
, (LPARAM
)&r
) )
310 wxLogLastError("SendMessage(SB_GETRECT)");
313 wxCopyRECTToRect(r
, rect
);
318 void wxStatusBar95::DoMoveWindow(int x
, int y
, int width
, int height
)
320 // the status bar wnd proc must be forwarded the WM_SIZE message whenever
321 // the stat bar position/size is changed because it normally positions the
322 // control itself along bottom or top side of the parent window - failing
323 // to do so will result in nasty visual effects
324 FORWARD_WM_SIZE(GetHwnd(), SIZE_RESTORED
, x
, y
, SendMessage
);
326 // but now, when the standard status bar wnd proc did all it wanted to do,
327 // move the status bar to its correct location - usually this call may be
328 // omitted because for normal status bars (positioned along the bottom
329 // edge) the position is already set correctly, but if the user wants to
330 // position them in some exotic location, this is really needed
331 wxWindow::DoMoveWindow(x
, y
, width
, height
);
333 // adjust fields widths to the new size
337 #endif // __WIN95__ && wxUSE_NATIVE_STATUSBAR