///////////////////////////////////////////////////////////////////////////// // Name: src/msw/scrolbar.cpp // Purpose: wxScrollBar // Author: Julian Smart // Modified by: // Created: 04/01/98 // RCS-ID: $Id$ // Copyright: (c) Julian Smart // Licence: wxWindows licence ///////////////////////////////////////////////////////////////////////////// // For compilers that support precompilation, includes "wx.h". #include "wx/wxprec.h" #ifdef __BORLANDC__ #pragma hdrstop #endif #if wxUSE_SCROLLBAR #include "wx/scrolbar.h" #ifndef WX_PRECOMP #include "wx/utils.h" #include "wx/settings.h" #endif #include "wx/msw/private.h" // Scrollbar bool wxScrollBar::Create(wxWindow *parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style, const wxValidator& validator, const wxString& name) { if ( !CreateControl(parent, id, pos, size, style, validator, name) ) return false; if (!MSWCreateControl(wxT("ScrollBar"), wxEmptyString, pos, size)) return false; SetScrollbar(0, 1, 2, 1, false); return true; } wxScrollBar::~wxScrollBar(void) { } bool wxScrollBar::MSWOnScroll(int WXUNUSED(orientation), WXWORD wParam, WXWORD WXUNUSED(pos), WXHWND WXUNUSED(control)) { // don't use pos parameter because it is limited to 16 bits, get the full // 32 bit position from the control itself instead WinStruct scrollInfo; scrollInfo.fMask = SIF_RANGE | SIF_POS | SIF_TRACKPOS; if ( !::GetScrollInfo(GetHwnd(), SB_CTL, &scrollInfo) ) { wxLogLastError(wxT("GetScrollInfo")); return false; } int maxPos = scrollInfo.nMax; // A page size greater than one has the effect of reducing the effective // range, therefore the range has already been boosted artificially - so // reduce it again. if ( m_pageSize > 1 ) maxPos -= (m_pageSize - 1); int position = scrollInfo.nPos; wxEventType scrollEvent = wxEVT_NULL; switch ( wParam ) { case SB_TOP: position = 0; scrollEvent = wxEVT_SCROLL_TOP; break; case SB_BOTTOM: position = maxPos; scrollEvent = wxEVT_SCROLL_BOTTOM; break; case SB_LINEUP: position--; scrollEvent = wxEVT_SCROLL_LINEUP; break; case SB_LINEDOWN: position++; scrollEvent = wxEVT_SCROLL_LINEDOWN; break; case SB_PAGEUP: position -= GetPageSize(); scrollEvent = wxEVT_SCROLL_PAGEUP; break; case SB_PAGEDOWN: position += GetPageSize(); scrollEvent = wxEVT_SCROLL_PAGEDOWN; break; case SB_THUMBPOSITION: case SB_THUMBTRACK: position = scrollInfo.nTrackPos; scrollEvent = wParam == SB_THUMBPOSITION ? wxEVT_SCROLL_THUMBRELEASE : wxEVT_SCROLL_THUMBTRACK; break; case SB_ENDSCROLL: scrollEvent = wxEVT_SCROLL_CHANGED; break; } if ( position != scrollInfo.nPos ) { if ( position < 0 ) position = 0; if ( position > maxPos ) position = maxPos; SetThumbPosition(position); } else if ( scrollEvent != wxEVT_SCROLL_THUMBRELEASE && scrollEvent != wxEVT_SCROLL_CHANGED ) { // don't process the event if there is no displacement, // unless this is a thumb release or end scroll event. return false; } wxScrollEvent event(scrollEvent, m_windowId); event.SetOrientation(IsVertical() ? wxVERTICAL : wxHORIZONTAL); event.SetPosition(position); event.SetEventObject( this ); return HandleWindowEvent(event); } void wxScrollBar::SetThumbPosition(int viewStart) { SCROLLINFO info; info.cbSize = sizeof(SCROLLINFO); info.nPage = 0; info.nMin = 0; info.nPos = viewStart; info.fMask = SIF_POS ; ::SetScrollInfo((HWND) GetHWND(), SB_CTL, &info, TRUE); } int wxScrollBar::GetThumbPosition(void) const { WinStruct scrollInfo; scrollInfo.fMask = SIF_POS; if ( !::GetScrollInfo(GetHwnd(), SB_CTL, &scrollInfo) ) { wxLogLastError(wxT("GetScrollInfo")); } return scrollInfo.nPos; } void wxScrollBar::SetScrollbar(int position, int thumbSize, int range, int pageSize, bool refresh) { m_viewSize = pageSize; m_pageSize = thumbSize; m_objectSize = range; // The range (number of scroll steps) is the // object length minus the page size. int range1 = wxMax((m_objectSize - m_pageSize), 0) ; // Try to adjust the range to cope with page size > 1 // (see comment for SetPageLength) if ( m_pageSize > 1 ) { range1 += (m_pageSize - 1); } SCROLLINFO info; info.cbSize = sizeof(SCROLLINFO); info.nPage = m_pageSize; info.nMin = 0; info.nMax = range1; info.nPos = position; info.fMask = SIF_PAGE | SIF_RANGE | SIF_POS; ::SetScrollInfo((HWND) GetHWND(), SB_CTL, &info, refresh); } void wxScrollBar::Command(wxCommandEvent& event) { SetThumbPosition(event.GetInt()); ProcessCommand(event); } wxSize wxScrollBar::DoGetBestSize() const { int w = 100; int h = 100; if ( IsVertical() ) { w = wxSystemSettings::GetMetric(wxSYS_VSCROLL_X); } else { h = wxSystemSettings::GetMetric(wxSYS_HSCROLL_Y); } wxSize best(w, h); CacheBestSize(best); return best; } WXDWORD wxScrollBar::MSWGetStyle(long style, WXDWORD *exstyle) const { // we never have an external border WXDWORD msStyle = wxControl::MSWGetStyle ( (style & ~wxBORDER_MASK) | wxBORDER_NONE, exstyle ); // SBS_HORZ is 0 anyhow, but do mention it explicitly for clarity msStyle |= style & wxSB_HORIZONTAL ? SBS_HORZ : SBS_VERT; return msStyle; } WXHBRUSH wxScrollBar::MSWControlColor(WXHDC pDC, WXHWND hWnd) { // unless we have an explicitly set bg colour, use default (gradient under // XP) brush instead of GetBackgroundColour() one as the base class would // // note that fg colour isn't used for a scrollbar return UseBgCol() ? wxControl::MSWControlColor(pDC, hWnd) : NULL; } #endif // wxUSE_SCROLLBAR