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
,
80 case WM_LBUTTONDBLCLK
:
83 case WM_RBUTTONDBLCLK
:
86 case WM_MBUTTONDBLCLK
:
87 wxStatusBar95
*sb
= (wxStatusBar95
*)GetWindowLong(hwnd
, GWL_USERDATA
);
88 sb
->MSWWindowProc(message
, wParam
, lParam
);
92 return ::CallWindowProc(CASTWNDPROC gs_wndprocStatBar
, hwnd
, message
, wParam
, lParam
);
95 // ============================================================================
97 // ============================================================================
99 // ----------------------------------------------------------------------------
100 // wxStatusBar95 class
101 // ----------------------------------------------------------------------------
103 wxStatusBar95::wxStatusBar95()
110 bool wxStatusBar95::Create(wxWindow
*parent
,
113 const wxString
& name
)
115 wxCHECK_MSG( parent
, FALSE
, wxT("status bar must have a parent") );
118 SetWindowStyleFlag(style
);
121 parent
->AddChild(this);
123 m_windowId
= id
== -1 ? NewControlId() : id
;
125 DWORD wstyle
= WS_CHILD
| WS_VISIBLE
/* | WS_CLIPSIBLINGS */ ;
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
132 if ( !(style
& wxST_SIZEGRIP
) )
138 // may be some versions of comctl32.dll do need it - anyhow, it won't
140 wstyle
|= SBARS_SIZEGRIP
;
143 m_hWnd
= (WXHWND
)CreateStatusWindow(wstyle
,
149 wxLogSysError(_("Failed to create a status bar."));
156 // we can't subclass this window as usual because the status bar window
157 // proc processes WM_SIZE and WM_PAINT specially
158 // SubclassWin(m_hWnd);
160 // but we want to process the messages from it still, so do custom
162 gs_wndprocStatBar
= (WXFARPROC
)GetWindowLong(GetHwnd(), GWL_WNDPROC
);
163 SetWindowLong(GetHwnd(), GWL_WNDPROC
, (LONG
)wxStatusBarProc
);
164 SetWindowLong(GetHwnd(), GWL_USERDATA
, (LONG
)this);
169 wxStatusBar95::~wxStatusBar95()
171 delete [] m_statusWidths
;
174 void wxStatusBar95::CopyFieldsWidth(const int widths
[])
176 if (widths
&& !m_statusWidths
)
177 m_statusWidths
= new int[m_nFields
];
179 if ( widths
!= NULL
) {
180 for ( int i
= 0; i
< m_nFields
; i
++ )
181 m_statusWidths
[i
] = widths
[i
];
184 delete [] m_statusWidths
;
185 m_statusWidths
= NULL
;
189 void wxStatusBar95::SetFieldsCount(int nFields
, const int *widths
)
191 // this is a Windows limitation
192 wxASSERT_MSG( (nFields
> 0) && (nFields
< 255), _T("too many fields") );
196 CopyFieldsWidth(widths
);
200 void wxStatusBar95::SetStatusWidths(int n
, const int widths
[])
202 wxASSERT_MSG( n
== m_nFields
, _T("field number mismatch") );
204 CopyFieldsWidth(widths
);
208 void wxStatusBar95::SetFieldsWidth()
214 SendMessage(GetHwnd(), SB_GETBORDERS
, 0, (LPARAM
)aBorders
);
216 int extraWidth
= aBorders
[2]; // space between fields
218 int *pWidths
= new int[m_nFields
];
221 GetClientSize(&nWindowWidth
, &y
);
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
;
230 // -1 doesn't mean the same thing for wxWindows and Win32, recalc
234 for ( i
= 0; i
< m_nFields
; i
++ ) {
235 if ( m_statusWidths
[i
] == -1 )
238 nTotalWidth
+= m_statusWidths
[i
] + extraWidth
;
241 if ( nVarCount
== 0 ) {
242 wxFAIL_MSG( _T("at least one field must be of variable width") );
247 int nVarWidth
= (nWindowWidth
- nTotalWidth
) / nVarCount
;
251 for ( i
= 0; i
< m_nFields
; i
++ ) {
252 if ( m_statusWidths
[i
] == -1 )
253 nCurPos
+= nVarWidth
;
255 nCurPos
+= m_statusWidths
[i
] + extraWidth
;
256 pWidths
[i
] = nCurPos
;
260 if ( !StatusBar_SetParts(GetHwnd(), m_nFields
, pWidths
) ) {
261 wxLogLastError(wxT("StatusBar_SetParts"));
267 void wxStatusBar95::SetStatusText(const wxString
& strText
, int nField
)
269 wxCHECK_RET( (nField
>= 0) && (nField
< m_nFields
),
270 _T("invalid statusbar field index") );
272 if ( !StatusBar_SetText(GetHwnd(), nField
, strText
) ) {
273 wxLogLastError(wxT("StatusBar_SetText"));
277 wxString
wxStatusBar95::GetStatusText(int nField
) const
279 wxCHECK_MSG( (nField
>= 0) && (nField
< m_nFields
), wxEmptyString
,
280 _T("invalid statusbar field index") );
283 int len
= StatusBar_GetTextLen(GetHwnd(), nField
);
286 StatusBar_GetText(GetHwnd(), nField
, str
.GetWriteBuf(len
));
292 int wxStatusBar95::GetBorderX() const
295 SendMessage(GetHwnd(), SB_GETBORDERS
, 0, (LPARAM
)aBorders
);
300 int wxStatusBar95::GetBorderY() const
303 SendMessage(GetHwnd(), SB_GETBORDERS
, 0, (LPARAM
)aBorders
);
308 void wxStatusBar95::SetMinHeight(int height
)
310 SendMessage(GetHwnd(), SB_SETMINHEIGHT
, height
+ 2*GetBorderY(), 0);
312 // have to send a (dummy) WM_SIZE to redraw it now
313 SendMessage(GetHwnd(), WM_SIZE
, 0, 0);
316 bool wxStatusBar95::GetFieldRect(int i
, wxRect
& rect
) const
318 wxCHECK_MSG( (i
>= 0) && (i
< m_nFields
), FALSE
,
319 _T("invalid statusbar field index") );
322 if ( !::SendMessage(GetHwnd(), SB_GETRECT
, i
, (LPARAM
)&r
) )
324 wxLogLastError(wxT("SendMessage(SB_GETRECT)"));
327 wxCopyRECTToRect(r
, rect
);
332 void wxStatusBar95::DoMoveWindow(int x
, int y
, int width
, int height
)
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
);
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
);
347 // adjust fields widths to the new size
351 #endif // __WIN95__ && wxUSE_NATIVE_STATUSBAR