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
,
78 case WM_LBUTTONDBLCLK
:
81 case WM_RBUTTONDBLCLK
:
84 case WM_MBUTTONDBLCLK
:
85 wxStatusBar95
*sb
= (wxStatusBar95
*)GetWindowLong(hwnd
, GWL_USERDATA
);
86 sb
->MSWWindowProc(message
, wParam
, lParam
);
90 return ::CallWindowProc(CASTWNDPROC gs_wndprocStatBar
, hwnd
, message
, wParam
, lParam
);
93 // ============================================================================
95 // ============================================================================
97 // ----------------------------------------------------------------------------
98 // wxStatusBar95 class
99 // ----------------------------------------------------------------------------
101 wxStatusBar95::wxStatusBar95()
108 bool wxStatusBar95::Create(wxWindow
*parent
,
111 const wxString
& name
)
113 wxCHECK_MSG( parent
, FALSE
, wxT("status bar must have a parent") );
116 SetWindowStyleFlag(style
);
119 parent
->AddChild(this);
121 m_windowId
= id
== -1 ? NewControlId() : id
;
123 DWORD wstyle
= WS_CHILD
| WS_VISIBLE
;
125 // setting SBARS_SIZEGRIP is perfectly useless: it's always on by default
126 // (at least in the version of comctl32.dll I'm using), and the only way to
127 // turn it off is to use CCS_TOP style - as we position the status bar
128 // manually anyhow (see DoMoveWindow), use CCS_TOP style if wxST_SIZEGRIP
130 if ( !(style
& wxST_SIZEGRIP
) )
136 // may be some versions of comctl32.dll do need it - anyhow, it won't
138 wstyle
|= SBARS_SIZEGRIP
;
141 m_hWnd
= (WXHWND
)CreateStatusWindow(wstyle
,
147 wxLogSysError(_("Failed to create a status bar."));
154 // we can't subclass this window as usual because the status bar window
155 // proc processes WM_SIZE and WM_PAINT specially
156 // SubclassWin(m_hWnd);
158 // but we want to process the messages from it still, so do custom
160 gs_wndprocStatBar
= (WXFARPROC
)GetWindowLong(GetHwnd(), GWL_WNDPROC
);
161 SetWindowLong(GetHwnd(), GWL_WNDPROC
, (LONG
)wxStatusBarProc
);
162 SetWindowLong(GetHwnd(), GWL_USERDATA
, (LONG
)this);
167 wxStatusBar95::~wxStatusBar95()
169 delete [] m_statusWidths
;
172 void wxStatusBar95::CopyFieldsWidth(const int widths
[])
174 if (widths
&& !m_statusWidths
)
175 m_statusWidths
= new int[m_nFields
];
177 if ( widths
!= NULL
) {
178 for ( int i
= 0; i
< m_nFields
; i
++ )
179 m_statusWidths
[i
] = widths
[i
];
182 delete [] m_statusWidths
;
183 m_statusWidths
= NULL
;
187 void wxStatusBar95::SetFieldsCount(int nFields
, const int *widths
)
189 // this is Windows limitation
190 wxASSERT_MSG( (nFields
> 0) && (nFields
< 255), _T("too many fields") );
194 CopyFieldsWidth(widths
);
198 void wxStatusBar95::SetStatusWidths(int n
, const int widths
[])
200 wxASSERT_MSG( n
== m_nFields
, _T("field number mismatch") );
202 CopyFieldsWidth(widths
);
206 void wxStatusBar95::SetFieldsWidth()
212 SendMessage(GetHwnd(), SB_GETBORDERS
, 0, (LPARAM
)aBorders
);
214 int extraWidth
= aBorders
[2]; // space between fields
216 int *pWidths
= new int[m_nFields
];
219 GetClientSize(&nWindowWidth
, &y
);
221 if ( m_statusWidths
== NULL
) {
222 // default: all fields have the same width
223 int nWidth
= nWindowWidth
/ m_nFields
;
224 for ( int i
= 0; i
< m_nFields
; i
++ )
225 pWidths
[i
] = (i
+ 1) * nWidth
;
228 // -1 doesn't mean the same thing for wxWindows and Win32, recalc
232 for ( i
= 0; i
< m_nFields
; i
++ ) {
233 if ( m_statusWidths
[i
] == -1 )
236 nTotalWidth
+= m_statusWidths
[i
] + extraWidth
;
239 if ( nVarCount
== 0 ) {
240 wxFAIL_MSG( _T("at least one field must be of variable width") );
245 int nVarWidth
= (nWindowWidth
- nTotalWidth
) / nVarCount
;
249 for ( i
= 0; i
< m_nFields
; i
++ ) {
250 if ( m_statusWidths
[i
] == -1 )
251 nCurPos
+= nVarWidth
;
253 nCurPos
+= m_statusWidths
[i
] + extraWidth
;
254 pWidths
[i
] = nCurPos
;
258 if ( !StatusBar_SetParts(GetHwnd(), m_nFields
, pWidths
) ) {
259 wxLogLastError(wxT("StatusBar_SetParts"));
265 void wxStatusBar95::SetStatusText(const wxString
& strText
, int nField
)
267 wxCHECK_RET( (nField
>= 0) && (nField
< m_nFields
),
268 _T("invalid statusbar field index") );
270 if ( !StatusBar_SetText(GetHwnd(), nField
, strText
) ) {
271 wxLogLastError(wxT("StatusBar_SetText"));
275 wxString
wxStatusBar95::GetStatusText(int nField
) const
277 wxCHECK_MSG( (nField
>= 0) && (nField
< m_nFields
), wxEmptyString
,
278 _T("invalid statusbar field index") );
281 int len
= StatusBar_GetTextLen(GetHwnd(), nField
);
284 StatusBar_GetText(GetHwnd(), nField
, str
.GetWriteBuf(len
));
290 int wxStatusBar95::GetBorderX() const
293 SendMessage(GetHwnd(), SB_GETBORDERS
, 0, (LPARAM
)aBorders
);
298 int wxStatusBar95::GetBorderY() const
301 SendMessage(GetHwnd(), SB_GETBORDERS
, 0, (LPARAM
)aBorders
);
306 void wxStatusBar95::SetMinHeight(int height
)
308 SendMessage(GetHwnd(), SB_SETMINHEIGHT
, height
+ 2*GetBorderY(), 0);
310 // have to send a (dummy) WM_SIZE to redraw it now
311 SendMessage(GetHwnd(), WM_SIZE
, 0, 0);
314 bool wxStatusBar95::GetFieldRect(int i
, wxRect
& rect
) const
316 wxCHECK_MSG( (i
>= 0) && (i
< m_nFields
), FALSE
,
317 _T("invalid statusbar field index") );
320 if ( !::SendMessage(GetHwnd(), SB_GETRECT
, i
, (LPARAM
)&r
) )
322 wxLogLastError("SendMessage(SB_GETRECT)");
325 wxCopyRECTToRect(r
, rect
);
330 void wxStatusBar95::DoMoveWindow(int x
, int y
, int width
, int height
)
332 // the status bar wnd proc must be forwarded the WM_SIZE message whenever
333 // the stat bar position/size is changed because it normally positions the
334 // control itself along bottom or top side of the parent window - failing
335 // to do so will result in nasty visual effects
336 FORWARD_WM_SIZE(GetHwnd(), SIZE_RESTORED
, x
, y
, SendMessage
);
338 // but now, when the standard status bar wnd proc did all it wanted to do,
339 // move the status bar to its correct location - usually this call may be
340 // omitted because for normal status bars (positioned along the bottom
341 // edge) the position is already set correctly, but if the user wants to
342 // position them in some exotic location, this is really needed
343 wxWindow::DoMoveWindow(x
, y
, width
, height
);
345 // adjust fields widths to the new size
349 #endif // __WIN95__ && wxUSE_NATIVE_STATUSBAR