Files
libreoffice/vcl/source/control/slider.cxx
Jan Holesovsky 5ca1936975 rendercontext: Mass fix of using the wrong size.
In the paints, we must use the size of the Window for the computations, not of
the RenderContext - the RenderContext can be much bigger than the Window in
the double-buffering case.

Fixes for example the list boxes, and many others.

Change-Id: I4c7607569f88b2d097587140858d0862e54b5ea6
2015-06-23 10:32:28 +02:00

1037 lines
30 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* This file incorporates work covered by the following license notice:
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
*/
#include <tools/rc.h>
#include <vcl/event.hxx>
#include <vcl/decoview.hxx>
#include <vcl/slider.hxx>
#include <vcl/settings.hxx>
#include "thumbpos.hxx"
#define SLIDER_DRAW_THUMB ((sal_uInt16)0x0001)
#define SLIDER_DRAW_CHANNEL1 ((sal_uInt16)0x0002)
#define SLIDER_DRAW_CHANNEL2 ((sal_uInt16)0x0004)
#define SLIDER_DRAW_CHANNEL (SLIDER_DRAW_CHANNEL1 | SLIDER_DRAW_CHANNEL2)
#define SLIDER_DRAW_ALL (SLIDER_DRAW_THUMB | SLIDER_DRAW_CHANNEL)
#define SLIDER_STATE_CHANNEL1_DOWN ((sal_uInt16)0x0001)
#define SLIDER_STATE_CHANNEL2_DOWN ((sal_uInt16)0x0002)
#define SLIDER_STATE_THUMB_DOWN ((sal_uInt16)0x0004)
#define SLIDER_THUMB_SIZE 9
#define SLIDER_THUMB_HALFSIZE 4
#define SLIDER_CHANNEL_OFFSET 0
#define SLIDER_CHANNEL_SIZE 4
#define SLIDER_CHANNEL_HALFSIZE 2
#define SLIDER_HEIGHT 16
#define SLIDER_VIEW_STYLE (WB_3DLOOK | WB_HORZ | WB_VERT)
void Slider::ImplInit( vcl::Window* pParent, WinBits nStyle )
{
mnThumbPixOffset = 0;
mnThumbPixRange = 0;
mnThumbPixPos = 0; // between mnThumbPixOffset and mnThumbPixOffset+mnThumbPixRange
mnChannelPixOffset = 0;
mnChannelPixRange = 0;
mnChannelPixTop = 0;
mnChannelPixBottom = 0;
mnMinRange = 0;
mnMaxRange = 100;
mnThumbPos = 0;
mnLineSize = 1;
mnPageSize = 1;
mnDelta = 0;
mnDragDraw = 0;
mnDrawFlags = SLIDER_DRAW_ALL;
mnStateFlags = 0;
meScrollType = SCROLL_DONTKNOW;
mbCalcSize = true;
mbFullDrag = true;
mpLinkedField = nullptr;
Control::ImplInit( pParent, nStyle, NULL );
ImplInitSettings();
SetSizePixel( CalcWindowSizePixel() );
}
Slider::Slider( vcl::Window* pParent, WinBits nStyle ) :
Control(WINDOW_SLIDER)
{
ImplInit( pParent, nStyle );
}
Slider::~Slider()
{
disposeOnce();
}
void Slider::dispose()
{
mpLinkedField.clear();
Control::dispose();
}
void Slider::ImplInitSettings()
{
vcl::Window* pParent = GetParent();
if ( pParent->IsChildTransparentModeEnabled() && !IsControlBackground() )
{
EnableChildTransparentMode( true );
SetParentClipMode( ParentClipMode::NoClip );
SetPaintTransparent( true );
SetBackground();
}
else
{
EnableChildTransparentMode( false );
SetParentClipMode( ParentClipMode::NONE );
SetPaintTransparent( false );
if ( IsControlBackground() )
SetBackground( GetControlBackground() );
else
SetBackground( pParent->GetBackground() );
}
}
void Slider::ImplUpdateRects( bool bUpdate )
{
Rectangle aOldThumbRect = maThumbRect;
bool bInvalidateAll = false;
if ( mnThumbPixRange )
{
if ( GetStyle() & WB_HORZ )
{
maThumbRect.Left() = mnThumbPixPos-SLIDER_THUMB_HALFSIZE;
maThumbRect.Right() = maThumbRect.Left()+SLIDER_THUMB_SIZE-1;
if ( mnChannelPixOffset < maThumbRect.Left() )
{
maChannel1Rect.Left() = mnChannelPixOffset;
maChannel1Rect.Right() = maThumbRect.Left()-1;
maChannel1Rect.Top() = mnChannelPixTop;
maChannel1Rect.Bottom() = mnChannelPixBottom;
}
else
maChannel1Rect.SetEmpty();
if ( mnChannelPixOffset+mnChannelPixRange-1 > maThumbRect.Right() )
{
maChannel2Rect.Left() = maThumbRect.Right()+1;
maChannel2Rect.Right() = mnChannelPixOffset+mnChannelPixRange-1;
maChannel2Rect.Top() = mnChannelPixTop;
maChannel2Rect.Bottom() = mnChannelPixBottom;
}
else
maChannel2Rect.SetEmpty();
const Rectangle aControlRegion( Rectangle( Point(0,0), Size( SLIDER_THUMB_SIZE, 10 ) ) );
Rectangle aThumbBounds, aThumbContent;
if ( GetNativeControlRegion( CTRL_SLIDER, PART_THUMB_HORZ,
aControlRegion, ControlState::NONE, ImplControlValue(), OUString(),
aThumbBounds, aThumbContent ) )
{
maThumbRect.Left() = mnThumbPixPos - aThumbBounds.GetWidth()/2;
maThumbRect.Right() = maThumbRect.Left() + aThumbBounds.GetWidth() - 1;
bInvalidateAll = true;
}
}
else
{
maThumbRect.Top() = mnThumbPixPos-SLIDER_THUMB_HALFSIZE;
maThumbRect.Bottom() = maThumbRect.Top()+SLIDER_THUMB_SIZE-1;
if ( mnChannelPixOffset < maThumbRect.Top() )
{
maChannel1Rect.Top() = mnChannelPixOffset;
maChannel1Rect.Bottom() = maThumbRect.Top()-1;
maChannel1Rect.Left() = mnChannelPixTop;
maChannel1Rect.Right() = mnChannelPixBottom;
}
else
maChannel1Rect.SetEmpty();
if ( mnChannelPixOffset+mnChannelPixRange-1 > maThumbRect.Bottom() )
{
maChannel2Rect.Top() = maThumbRect.Bottom()+1;
maChannel2Rect.Bottom() = mnChannelPixOffset+mnChannelPixRange-1;
maChannel2Rect.Left() = mnChannelPixTop;
maChannel2Rect.Right() = mnChannelPixBottom;
}
else
maChannel2Rect.SetEmpty();
const Rectangle aControlRegion( Rectangle( Point(0,0), Size( 10, SLIDER_THUMB_SIZE ) ) );
Rectangle aThumbBounds, aThumbContent;
if ( GetNativeControlRegion( CTRL_SLIDER, PART_THUMB_VERT,
aControlRegion, ControlState::NONE, ImplControlValue(), OUString(),
aThumbBounds, aThumbContent ) )
{
maThumbRect.Top() = mnThumbPixPos - aThumbBounds.GetHeight()/2;
maThumbRect.Bottom() = maThumbRect.Top() + aThumbBounds.GetHeight() - 1;
bInvalidateAll = true;
}
}
}
else
{
maChannel1Rect.SetEmpty();
maChannel2Rect.SetEmpty();
maThumbRect.SetEmpty();
}
if ( bUpdate )
{
if ( aOldThumbRect != maThumbRect )
{
mnDrawFlags = SLIDER_DRAW_ALL;
if( bInvalidateAll )
Invalidate();
else
{
vcl::Region aInvalidRegion( aOldThumbRect );
aInvalidRegion.Union( maThumbRect );
if( !IsBackground() && GetParent() )
{
const Point aPos( GetPosPixel() );
aInvalidRegion.Move( aPos.X(), aPos.Y() );
GetParent()->Invalidate( aInvalidRegion, InvalidateFlags::Transparent | InvalidateFlags::Update );
}
else
Invalidate( aInvalidRegion );
}
}
}
}
void Slider::ImplSetFieldLink(const Link<>& rLink)
{
if (mpLinkedField != nullptr)
{
mpLinkedField->SetModifyHdl(rLink);
mpLinkedField->SetUpHdl(rLink);
mpLinkedField->SetDownHdl(rLink);
mpLinkedField->SetFirstHdl(rLink);
mpLinkedField->SetLastHdl(rLink);
mpLinkedField->SetLoseFocusHdl(rLink);
}
}
void Slider::ImplUpdateLinkedField()
{
if (mpLinkedField)
mpLinkedField->SetValue(mnThumbPos);
}
IMPL_LINK(Slider, LinkedFieldModifyHdl, NumericField*, pField)
{
if (pField)
{
SetThumbPos(pField->GetValue());
}
return 0;
}
long Slider::ImplCalcThumbPos( long nPixPos )
{
// calculate position
long nCalcThumbPos;
nCalcThumbPos = ImplMulDiv( nPixPos-mnThumbPixOffset, mnMaxRange-mnMinRange, mnThumbPixRange-1 );
nCalcThumbPos += mnMinRange;
return nCalcThumbPos;
}
long Slider::ImplCalcThumbPosPix( long nPos )
{
// calculate position
long nCalcThumbPos;
nCalcThumbPos = ImplMulDiv( nPos-mnMinRange, mnThumbPixRange-1, mnMaxRange-mnMinRange );
// at the beginning and end we try to display Slider correctly
if ( !nCalcThumbPos && (mnThumbPos > mnMinRange) )
nCalcThumbPos = 1;
if ( nCalcThumbPos &&
(nCalcThumbPos == mnThumbPixRange-1) &&
(mnThumbPos < mnMaxRange) )
nCalcThumbPos--;
return nCalcThumbPos+mnThumbPixOffset;
}
void Slider::ImplCalc( bool bUpdate )
{
bool bInvalidateAll = false;
if ( mbCalcSize )
{
long nOldChannelPixOffset = mnChannelPixOffset;
long nOldChannelPixRange = mnChannelPixRange;
long nOldChannelPixTop = mnChannelPixTop;
long nOldChannelPixBottom = mnChannelPixBottom;
long nCalcWidth;
long nCalcHeight;
maChannel1Rect.SetEmpty();
maChannel2Rect.SetEmpty();
maThumbRect.SetEmpty();
Size aSize = GetOutputSizePixel();
if ( GetStyle() & WB_HORZ )
{
nCalcWidth = aSize.Width();
nCalcHeight = aSize.Height();
maThumbRect.Top() = 0;
maThumbRect.Bottom()= aSize.Height()-1;
}
else
{
nCalcWidth = aSize.Height();
nCalcHeight = aSize.Width();
maThumbRect.Left() = 0;
maThumbRect.Right() = aSize.Width()-1;
}
if ( nCalcWidth >= SLIDER_THUMB_SIZE )
{
mnThumbPixOffset = SLIDER_THUMB_HALFSIZE;
mnThumbPixRange = nCalcWidth-(SLIDER_THUMB_HALFSIZE*2);
mnThumbPixPos = 0;
mnChannelPixOffset = SLIDER_CHANNEL_OFFSET;
mnChannelPixRange = nCalcWidth-(SLIDER_CHANNEL_OFFSET*2);
mnChannelPixTop = (nCalcHeight/2)-SLIDER_CHANNEL_HALFSIZE;
mnChannelPixBottom = mnChannelPixTop+SLIDER_CHANNEL_SIZE-1;
}
else
{
mnThumbPixRange = 0;
mnChannelPixRange = 0;
}
if ( (nOldChannelPixOffset != mnChannelPixOffset) ||
(nOldChannelPixRange != mnChannelPixRange) ||
(nOldChannelPixTop != mnChannelPixTop) ||
(nOldChannelPixBottom != mnChannelPixBottom) )
bInvalidateAll = true;
mbCalcSize = false;
}
if ( mnThumbPixRange )
mnThumbPixPos = ImplCalcThumbPosPix( mnThumbPos );
if ( bUpdate && bInvalidateAll )
{
mnDrawFlags = SLIDER_DRAW_ALL;
Invalidate();
bUpdate = false;
}
ImplUpdateRects( bUpdate );
}
void Slider::ImplDraw(vcl::RenderContext& rRenderContext, sal_uInt16 nDrawFlags)
{
DecorationView aDecoView(&rRenderContext);
DrawButtonFlags nStyle;
const StyleSettings& rStyleSettings = rRenderContext.GetSettings().GetStyleSettings();
bool bEnabled = IsEnabled();
// do missing calculations
if (mbCalcSize)
ImplCalc(false);
ControlPart nPart = (GetStyle() & WB_HORZ) ? PART_TRACK_HORZ_AREA : PART_TRACK_VERT_AREA;
ControlState nState = (IsEnabled() ? ControlState::ENABLED : ControlState::NONE);
nState |= (HasFocus() ? ControlState::FOCUSED : ControlState::NONE);
SliderValue sldValue;
sldValue.mnMin = mnMinRange;
sldValue.mnMax = mnMaxRange;
sldValue.mnCur = mnThumbPos;
sldValue.maThumbRect = maThumbRect;
if (IsMouseOver())
{
if (maThumbRect.IsInside(GetPointerPosPixel()))
sldValue.mnThumbState |= ControlState::ROLLOVER;
}
const Rectangle aCtrlRegion(Point(0,0), GetOutputSizePixel());
bool bNativeOK = rRenderContext.DrawNativeControl(CTRL_SLIDER, nPart, aCtrlRegion, nState, sldValue, OUString());
if (bNativeOK)
return;
if ((nDrawFlags & SLIDER_DRAW_CHANNEL1) && !maChannel1Rect.IsEmpty())
{
long nRectSize;
Rectangle aRect = maChannel1Rect;
rRenderContext.SetLineColor(rStyleSettings.GetShadowColor());
if (GetStyle() & WB_HORZ)
{
rRenderContext.DrawLine(aRect.TopLeft(), Point(aRect.Left(), aRect.Bottom() - 1));
rRenderContext.DrawLine(aRect.TopLeft(), aRect.TopRight());
}
else
{
rRenderContext.DrawLine(aRect.TopLeft(), Point(aRect.Right() - 1, aRect.Top()));
rRenderContext.DrawLine(aRect.TopLeft(), aRect.BottomLeft());
}
rRenderContext.SetLineColor(rStyleSettings.GetLightColor());
if (GetStyle() & WB_HORZ)
{
rRenderContext.DrawLine(aRect.BottomLeft(), aRect.BottomRight());
nRectSize = aRect.GetWidth();
}
else
{
rRenderContext.DrawLine(aRect.TopRight(), aRect.BottomRight());
nRectSize = aRect.GetHeight();
}
if (nRectSize > 1)
{
aRect.Left()++;
aRect.Top()++;
if (GetStyle() & WB_HORZ)
aRect.Bottom()--;
else
aRect.Right()--;
rRenderContext.SetLineColor();
if (mnStateFlags & SLIDER_STATE_CHANNEL1_DOWN)
rRenderContext.SetFillColor(rStyleSettings.GetShadowColor());
else
rRenderContext.SetFillColor(rStyleSettings.GetCheckedColor());
rRenderContext.DrawRect(aRect);
}
}
if ((nDrawFlags & SLIDER_DRAW_CHANNEL2) && !maChannel2Rect.IsEmpty())
{
long nRectSize;
Rectangle aRect = maChannel2Rect;
rRenderContext.SetLineColor(rStyleSettings.GetLightColor());
if (GetStyle() & WB_HORZ)
{
rRenderContext.DrawLine(aRect.TopRight(), aRect.BottomRight());
rRenderContext.DrawLine(aRect.BottomLeft(), aRect.BottomRight());
nRectSize = aRect.GetWidth();
}
else
{
rRenderContext.DrawLine(aRect.BottomLeft(), aRect.BottomRight());
rRenderContext.DrawLine(aRect.TopRight(), aRect.BottomRight());
nRectSize = aRect.GetHeight();
}
if (nRectSize > 1)
{
rRenderContext.SetLineColor(rStyleSettings.GetShadowColor());
if (GetStyle() & WB_HORZ)
rRenderContext.DrawLine(aRect.TopLeft(), Point(aRect.Right() - 1, aRect.Top()));
else
rRenderContext.DrawLine(aRect.TopLeft(), Point(aRect.Left(), aRect.Bottom() - 1));
aRect.Right()--;
aRect.Bottom()--;
if (GetStyle() & WB_HORZ)
aRect.Top()++;
else
aRect.Left()++;
rRenderContext.SetLineColor();
if (mnStateFlags & SLIDER_STATE_CHANNEL2_DOWN)
rRenderContext.SetFillColor(rStyleSettings.GetShadowColor());
else
rRenderContext.SetFillColor(rStyleSettings.GetCheckedColor());
rRenderContext.DrawRect(aRect);
}
}
if (nDrawFlags & SLIDER_DRAW_THUMB)
{
if (!maThumbRect.IsEmpty())
{
if (bEnabled)
{
nStyle = DrawButtonFlags::NONE;
if (mnStateFlags & SLIDER_STATE_THUMB_DOWN)
nStyle |= DrawButtonFlags::Pressed;
aDecoView.DrawButton(maThumbRect, nStyle);
}
else
{
rRenderContext.SetLineColor(rStyleSettings.GetShadowColor());
rRenderContext.SetFillColor(rStyleSettings.GetCheckedColor());
rRenderContext.DrawRect(maThumbRect);
}
}
}
}
bool Slider::ImplIsPageUp( const Point& rPos )
{
Size aSize = GetOutputSizePixel();
Rectangle aRect = maChannel1Rect;
if ( GetStyle() & WB_HORZ )
{
aRect.Top() = 0;
aRect.Bottom() = aSize.Height()-1;
}
else
{
aRect.Left() = 0;
aRect.Right() = aSize.Width()-1;
}
return aRect.IsInside( rPos );
}
bool Slider::ImplIsPageDown( const Point& rPos )
{
Size aSize = GetOutputSizePixel();
Rectangle aRect = maChannel2Rect;
if ( GetStyle() & WB_HORZ )
{
aRect.Top() = 0;
aRect.Bottom() = aSize.Height()-1;
}
else
{
aRect.Left() = 0;
aRect.Right() = aSize.Width()-1;
}
return aRect.IsInside( rPos );
}
long Slider::ImplSlide( long nNewPos, bool bCallEndSlide )
{
long nOldPos = mnThumbPos;
SetThumbPos( nNewPos );
long nDelta = mnThumbPos-nOldPos;
if ( nDelta )
{
mnDelta = nDelta;
Slide();
if ( bCallEndSlide )
EndSlide();
mnDelta = 0;
}
return nDelta;
}
long Slider::ImplDoAction( bool bCallEndSlide )
{
long nDelta = 0;
switch ( meScrollType )
{
case SCROLL_LINEUP:
nDelta = ImplSlide( mnThumbPos-mnLineSize, bCallEndSlide );
break;
case SCROLL_LINEDOWN:
nDelta = ImplSlide( mnThumbPos+mnLineSize, bCallEndSlide );
break;
case SCROLL_PAGEUP:
nDelta = ImplSlide( mnThumbPos-mnPageSize, bCallEndSlide );
break;
case SCROLL_PAGEDOWN:
nDelta = ImplSlide( mnThumbPos+mnPageSize, bCallEndSlide );
break;
case SCROLL_SET:
nDelta = ImplSlide( ImplCalcThumbPos( GetPointerPosPixel().X() ), bCallEndSlide );
break;
default:
break;
}
return nDelta;
}
void Slider::ImplDoMouseAction( const Point& rMousePos, bool bCallAction )
{
sal_uInt16 nOldStateFlags = mnStateFlags;
bool bAction = false;
switch ( meScrollType )
{
case( SCROLL_SET ):
{
const bool bUp = ImplIsPageUp( rMousePos ), bDown = ImplIsPageDown( rMousePos );
if ( bUp || bDown )
{
bAction = bCallAction;
mnStateFlags |= ( bUp ? SLIDER_STATE_CHANNEL1_DOWN : SLIDER_STATE_CHANNEL2_DOWN );
}
else
mnStateFlags &= ~( SLIDER_STATE_CHANNEL1_DOWN | SLIDER_STATE_CHANNEL2_DOWN );
break;
}
case SCROLL_PAGEUP:
if ( ImplIsPageUp( rMousePos ) )
{
bAction = bCallAction;
mnStateFlags |= SLIDER_STATE_CHANNEL1_DOWN;
}
else
mnStateFlags &= ~SLIDER_STATE_CHANNEL1_DOWN;
break;
case SCROLL_PAGEDOWN:
if ( ImplIsPageDown( rMousePos ) )
{
bAction = bCallAction;
mnStateFlags |= SLIDER_STATE_CHANNEL2_DOWN;
}
else
mnStateFlags &= ~SLIDER_STATE_CHANNEL2_DOWN;
break;
default:
break;
}
if ( bAction )
{
if ( ImplDoAction( false ) )
{
// Update the channel complete
if ( mnDragDraw & SLIDER_DRAW_CHANNEL )
{
Update();
mnDrawFlags = mnDragDraw;
Invalidate();
}
}
}
else if ( nOldStateFlags != mnStateFlags )
{
mnDrawFlags = mnDragDraw;
Invalidate();
}
}
long Slider::ImplDoSlide( long nNewPos )
{
if ( meScrollType != SCROLL_DONTKNOW )
return 0;
meScrollType = SCROLL_DRAG;
long nDelta = ImplSlide( nNewPos, true );
meScrollType = SCROLL_DONTKNOW;
return nDelta;
}
long Slider::ImplDoSlideAction( ScrollType eScrollType )
{
if ( (meScrollType != SCROLL_DONTKNOW) ||
(eScrollType == SCROLL_DONTKNOW) ||
(eScrollType == SCROLL_DRAG) )
return 0;
meScrollType = eScrollType;
long nDelta = ImplDoAction( true );
meScrollType = SCROLL_DONTKNOW;
return nDelta;
}
void Slider::MouseButtonDown( const MouseEvent& rMEvt )
{
if ( rMEvt.IsLeft() )
{
const Point& rMousePos = rMEvt.GetPosPixel();
StartTrackingFlags nTrackFlags = StartTrackingFlags::NONE;
if ( maThumbRect.IsInside( rMousePos ) )
{
meScrollType = SCROLL_DRAG;
mnDragDraw = SLIDER_DRAW_THUMB;
// calculate additional values
Point aCenterPos = maThumbRect.Center();
if ( GetStyle() & WB_HORZ )
mnMouseOff = rMousePos.X()-aCenterPos.X();
else
mnMouseOff = rMousePos.Y()-aCenterPos.Y();
}
else if ( ImplIsPageUp( rMousePos ) )
{
if( GetStyle() & WB_SLIDERSET )
meScrollType = SCROLL_SET;
else
{
nTrackFlags = StartTrackingFlags::ButtonRepeat;
meScrollType = SCROLL_PAGEUP;
}
mnDragDraw = SLIDER_DRAW_CHANNEL;
}
else if ( ImplIsPageDown( rMousePos ) )
{
if( GetStyle() & WB_SLIDERSET )
meScrollType = SCROLL_SET;
else
{
nTrackFlags = StartTrackingFlags::ButtonRepeat;
meScrollType = SCROLL_PAGEDOWN;
}
mnDragDraw = SLIDER_DRAW_CHANNEL;
}
// Shall we start Tracking?
if( meScrollType != SCROLL_DONTKNOW )
{
// store Start position for cancel and EndScroll delta
mnStartPos = mnThumbPos;
ImplDoMouseAction( rMousePos, meScrollType != SCROLL_SET );
Update();
if( meScrollType != SCROLL_SET )
StartTracking( nTrackFlags );
}
}
}
void Slider::MouseButtonUp( const MouseEvent& )
{
if( SCROLL_SET == meScrollType )
{
// reset Button and PageRect state
const sal_uInt16 nOldStateFlags = mnStateFlags;
mnStateFlags &= ~( SLIDER_STATE_CHANNEL1_DOWN | SLIDER_STATE_CHANNEL2_DOWN | SLIDER_STATE_THUMB_DOWN );
if ( nOldStateFlags != mnStateFlags )
{
mnDrawFlags = mnDragDraw;
Invalidate();
}
mnDragDraw = 0;
ImplDoAction( true );
meScrollType = SCROLL_DONTKNOW;
}
}
void Slider::Tracking( const TrackingEvent& rTEvt )
{
if ( rTEvt.IsTrackingEnded() )
{
// reset Button and PageRect state
sal_uInt16 nOldStateFlags = mnStateFlags;
mnStateFlags &= ~(SLIDER_STATE_CHANNEL1_DOWN | SLIDER_STATE_CHANNEL2_DOWN |
SLIDER_STATE_THUMB_DOWN);
if ( nOldStateFlags != mnStateFlags )
{
mnDrawFlags = mnDragDraw;
Invalidate();
}
mnDragDraw = 0;
// on cancel, reset the previous Thumb position
if ( rTEvt.IsTrackingCanceled() )
{
long nOldPos = mnThumbPos;
SetThumbPos( mnStartPos );
mnDelta = mnThumbPos-nOldPos;
Slide();
}
if ( meScrollType == SCROLL_DRAG )
{
// after dragging, recalculate to a rounded Thumb position
ImplCalc();
Update();
if ( !mbFullDrag && (mnStartPos != mnThumbPos) )
{
mnDelta = mnThumbPos-mnStartPos;
Slide();
mnDelta = 0;
}
}
mnDelta = mnThumbPos-mnStartPos;
EndSlide();
mnDelta = 0;
meScrollType = SCROLL_DONTKNOW;
}
else
{
const Point rMousePos = rTEvt.GetMouseEvent().GetPosPixel();
// special handling for dragging
if ( meScrollType == SCROLL_DRAG )
{
long nMovePix;
Point aCenterPos = maThumbRect.Center();
if ( GetStyle() & WB_HORZ )
nMovePix = rMousePos.X()-(aCenterPos.X()+mnMouseOff);
else
nMovePix = rMousePos.Y()-(aCenterPos.Y()+mnMouseOff);
// only if the mouse moves in Scroll direction we have to act
if ( nMovePix )
{
mnThumbPixPos += nMovePix;
if ( mnThumbPixPos < mnThumbPixOffset )
mnThumbPixPos = mnThumbPixOffset;
if ( mnThumbPixPos > (mnThumbPixOffset+mnThumbPixRange-1) )
mnThumbPixPos = mnThumbPixOffset+mnThumbPixRange-1;
long nOldPos = mnThumbPos;
mnThumbPos = ImplCalcThumbPos( mnThumbPixPos );
if ( nOldPos != mnThumbPos )
{
ImplUpdateRects();
Update();
ImplUpdateLinkedField();
if ( mbFullDrag && (nOldPos != mnThumbPos) )
{
mnDelta = mnThumbPos-nOldPos;
Slide();
mnDelta = 0;
}
}
}
}
else
ImplDoMouseAction( rMousePos, rTEvt.IsTrackingRepeat() );
// end tracking if ScrollBar values indicate we are done
if ( !IsVisible() )
EndTracking();
}
}
void Slider::KeyInput( const KeyEvent& rKEvt )
{
if ( !rKEvt.GetKeyCode().GetModifier() )
{
switch ( rKEvt.GetKeyCode().GetCode() )
{
case KEY_HOME:
ImplDoSlide( GetRangeMin() );
break;
case KEY_END:
ImplDoSlide( GetRangeMax() );
break;
case KEY_LEFT:
case KEY_UP:
ImplDoSlideAction( SCROLL_LINEUP );
break;
case KEY_RIGHT:
case KEY_DOWN:
ImplDoSlideAction( SCROLL_LINEDOWN );
break;
case KEY_PAGEUP:
ImplDoSlideAction( SCROLL_PAGEUP );
break;
case KEY_PAGEDOWN:
ImplDoSlideAction( SCROLL_PAGEDOWN );
break;
default:
Control::KeyInput( rKEvt );
break;
}
}
else
Control::KeyInput( rKEvt );
}
void Slider::Paint(vcl::RenderContext& rRenderContext, const Rectangle& /*rRect*/)
{
if (mnDrawFlags)
{
ImplDraw(rRenderContext, mnDrawFlags);
mnDrawFlags = 0;
}
}
void Slider::Resize()
{
Control::Resize();
mbCalcSize = true;
if ( IsReallyVisible() )
ImplCalc( false );
mnDrawFlags = SLIDER_DRAW_ALL;
Invalidate();
}
void Slider::SetLinkedField(VclPtr<NumericField> pField)
{
ImplSetFieldLink(Link<>());
mpLinkedField = pField;
ImplSetFieldLink(LINK(this, Slider, LinkedFieldModifyHdl));
}
void Slider::RequestHelp( const HelpEvent& rHEvt )
{
Control::RequestHelp( rHEvt );
}
void Slider::StateChanged( StateChangedType nType )
{
Control::StateChanged( nType );
if ( nType == StateChangedType::InitShow )
ImplCalc( false );
else if ( nType == StateChangedType::Data )
{
if ( IsReallyVisible() && IsUpdateMode() )
ImplCalc( true );
}
else if ( nType == StateChangedType::UpdateMode )
{
if ( IsReallyVisible() && IsUpdateMode() )
{
ImplCalc( false );
mnDrawFlags = SLIDER_DRAW_ALL;
Invalidate();
}
}
else if ( nType == StateChangedType::Enable )
{
if ( IsReallyVisible() && IsUpdateMode() )
{
mnDrawFlags = SLIDER_DRAW_ALL;
Invalidate();
}
}
else if ( nType == StateChangedType::Style )
{
if ( IsReallyVisible() && IsUpdateMode() )
{
if ( (GetPrevStyle() & SLIDER_VIEW_STYLE) !=
(GetStyle() & SLIDER_VIEW_STYLE) )
{
mbCalcSize = true;
ImplCalc( false );
mnDrawFlags = SLIDER_DRAW_ALL;
Invalidate();
}
}
}
else if ( nType == StateChangedType::ControlBackground )
{
ImplInitSettings();
mnDrawFlags = SLIDER_DRAW_ALL;
Invalidate();
}
}
void Slider::DataChanged( const DataChangedEvent& rDCEvt )
{
Control::DataChanged( rDCEvt );
if ( (rDCEvt.GetType() == DataChangedEventType::SETTINGS) &&
(rDCEvt.GetFlags() & AllSettingsFlags::STYLE) )
{
ImplInitSettings();
mnDrawFlags = SLIDER_DRAW_ALL;
Invalidate();
}
}
void Slider::Slide()
{
maSlideHdl.Call( this );
}
void Slider::EndSlide()
{
maEndSlideHdl.Call( this );
}
void Slider::SetRangeMin(long nNewRange)
{
SetRange(Range(nNewRange, GetRangeMax()));
}
void Slider::SetRangeMax(long nNewRange)
{
SetRange(Range(GetRangeMin(), nNewRange));
}
void Slider::SetRange( const Range& rRange )
{
// adjust Range
Range aRange = rRange;
aRange.Justify();
long nNewMinRange = aRange.Min();
long nNewMaxRange = aRange.Max();
// reset Range if different
if ( (mnMinRange != nNewMinRange) ||
(mnMaxRange != nNewMaxRange) )
{
mnMinRange = nNewMinRange;
mnMaxRange = nNewMaxRange;
// adjust Thumb
if ( mnThumbPos > mnMaxRange )
mnThumbPos = mnMaxRange;
if ( mnThumbPos < mnMinRange )
mnThumbPos = mnMinRange;
ImplUpdateLinkedField();
CompatStateChanged( StateChangedType::Data );
}
}
void Slider::SetThumbPos( long nNewThumbPos )
{
if ( nNewThumbPos < mnMinRange )
nNewThumbPos = mnMinRange;
if ( nNewThumbPos > mnMaxRange )
nNewThumbPos = mnMaxRange;
if ( mnThumbPos != nNewThumbPos )
{
mnThumbPos = nNewThumbPos;
ImplUpdateLinkedField();
CompatStateChanged( StateChangedType::Data );
}
}
Size Slider::CalcWindowSizePixel()
{
long nWidth = mnMaxRange-mnMinRange+(SLIDER_THUMB_HALFSIZE*2)+1;
long nHeight = SLIDER_HEIGHT;
Size aSize;
if ( GetStyle() & WB_HORZ )
{
aSize.Width() = nWidth;
aSize.Height() = nHeight;
}
else
{
aSize.Height() = nWidth;
aSize.Width() = nHeight;
}
return aSize;
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */