|  | /************************************************************** | 
|  | * | 
|  | * 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 | 
|  | * | 
|  | * Unless required by applicable law or agreed to in writing, | 
|  | * software distributed under the License is distributed on an | 
|  | * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY | 
|  | * KIND, either express or implied.  See the License for the | 
|  | * specific language governing permissions and limitations | 
|  | * under the License. | 
|  | * | 
|  | *************************************************************/ | 
|  |  | 
|  |  | 
|  |  | 
|  | // MARKER(update_precomp.py): autogen include statement, do not remove | 
|  | #include "precompiled_svx.hxx" | 
|  | #include "svx/dialcontrol.hxx" | 
|  | #include "bmpmask.hrc" | 
|  | #include <svx/dialmgr.hxx> | 
|  | #include <tools/rcid.h> | 
|  | #include <math.h> | 
|  | #include <vcl/virdev.hxx> | 
|  | #include <vcl/svapp.hxx> | 
|  | #include <vcl/bitmap.hxx> | 
|  | #include <vcl/field.hxx> | 
|  | #include <svtools/colorcfg.hxx> | 
|  |  | 
|  | namespace svx { | 
|  |  | 
|  | // ============================================================================ | 
|  |  | 
|  | const long DIAL_OUTER_WIDTH = 8; | 
|  |  | 
|  | // ============================================================================ | 
|  |  | 
|  |  | 
|  | // ---------------------------------------------------------------------------- | 
|  |  | 
|  | DialControlBmp::DialControlBmp( Window& rParent ) : | 
|  | VirtualDevice( rParent, 0, 0 ), | 
|  | mbEnabled( true ), | 
|  | mrParent( rParent ) | 
|  | { | 
|  | EnableRTL( sal_False ); | 
|  | } | 
|  |  | 
|  | void DialControlBmp::InitBitmap( const Size& rSize, const Font& rFont ) | 
|  | { | 
|  | Init( rSize ); | 
|  | SetFont( rFont ); | 
|  | } | 
|  |  | 
|  | void DialControlBmp::CopyBackground( const DialControlBmp& rSrc ) | 
|  | { | 
|  | Init( rSrc.maRect.GetSize() ); | 
|  | mbEnabled = rSrc.mbEnabled; | 
|  | Point aPos; | 
|  | DrawBitmapEx( aPos, rSrc.GetBitmapEx( aPos, maRect.GetSize() ) ); | 
|  | } | 
|  |  | 
|  | void DialControlBmp::DrawBackground( const Size& rSize, bool bEnabled ) | 
|  | { | 
|  | Init( rSize ); | 
|  | mbEnabled = bEnabled; | 
|  | DrawBackground(); | 
|  | } | 
|  |  | 
|  | void DialControlBmp::DrawElements( const String& rText, sal_Int32 nAngle ) | 
|  | { | 
|  | // *** rotated text *** | 
|  |  | 
|  | Font aFont( GetFont() ); | 
|  | aFont.SetColor( GetTextColor() ); | 
|  | aFont.SetOrientation( static_cast< short >( (nAngle + 5) / 10 ) );  // Font uses 1/10 degrees | 
|  | aFont.SetWeight( WEIGHT_BOLD ); | 
|  | SetFont( aFont ); | 
|  |  | 
|  | double fAngle = nAngle * F_PI180 / 100.0; | 
|  | double fSin = sin( fAngle ); | 
|  | double fCos = cos( fAngle ); | 
|  | double fWidth = GetTextWidth( rText ) / 2.0; | 
|  | double fHeight = GetTextHeight() / 2.0; | 
|  | long nX = static_cast< long >( mnCenterX - fWidth * fCos - fHeight * fSin ); | 
|  | long nY = static_cast< long >( mnCenterY + fWidth * fSin - fHeight * fCos ); | 
|  | Rectangle aRect( nX, nY, 2 * mnCenterX - nX, 2 * mnCenterY - nY ); | 
|  | DrawText( aRect, rText, mbEnabled ? 0 : TEXT_DRAW_DISABLE ); | 
|  |  | 
|  | // *** drag button *** | 
|  |  | 
|  | bool bMain = (nAngle % 4500) != 0; | 
|  | SetLineColor( GetButtonLineColor() ); | 
|  | SetFillColor( GetButtonFillColor( bMain ) ); | 
|  |  | 
|  | nX = mnCenterX - static_cast< long >( (DIAL_OUTER_WIDTH / 2 - mnCenterX) * fCos ); | 
|  | nY = mnCenterY - static_cast< long >( (mnCenterY - DIAL_OUTER_WIDTH / 2) * fSin ); | 
|  | long nSize = bMain ? (DIAL_OUTER_WIDTH / 4) : (DIAL_OUTER_WIDTH / 2 - 1); | 
|  | DrawEllipse( Rectangle( nX - nSize, nY - nSize, nX + nSize, nY + nSize ) ); | 
|  | } | 
|  |  | 
|  | // private -------------------------------------------------------------------- | 
|  |  | 
|  | const Color& DialControlBmp::GetBackgroundColor() const | 
|  | { | 
|  | return GetSettings().GetStyleSettings().GetDialogColor(); | 
|  | } | 
|  |  | 
|  | const Color& DialControlBmp::GetTextColor() const | 
|  | { | 
|  | return GetSettings().GetStyleSettings().GetLabelTextColor(); | 
|  | } | 
|  |  | 
|  | const Color& DialControlBmp::GetScaleLineColor() const | 
|  | { | 
|  | const StyleSettings& rSett = GetSettings().GetStyleSettings(); | 
|  | return mbEnabled ? rSett.GetButtonTextColor() : rSett.GetDisableColor(); | 
|  | } | 
|  |  | 
|  | const Color& DialControlBmp::GetButtonLineColor() const | 
|  | { | 
|  | const StyleSettings& rSett = GetSettings().GetStyleSettings(); | 
|  | return mbEnabled ? rSett.GetButtonTextColor() : rSett.GetDisableColor(); | 
|  | } | 
|  |  | 
|  | const Color& DialControlBmp::GetButtonFillColor( bool bMain ) const | 
|  | { | 
|  | const StyleSettings& rSett = GetSettings().GetStyleSettings(); | 
|  | return mbEnabled ? (bMain ? rSett.GetMenuColor() : rSett.GetHighlightColor()) : rSett.GetDisableColor(); | 
|  | } | 
|  |  | 
|  | void DialControlBmp::Init( const Size& rSize ) | 
|  | { | 
|  | SetSettings( mrParent.GetSettings() ); | 
|  | maRect.SetPos( Point( 0, 0 ) ); | 
|  | maRect.SetSize( rSize ); | 
|  | mnCenterX = rSize.Width() / 2; | 
|  | mnCenterY = rSize.Height() / 2; | 
|  | SetOutputSize( rSize ); | 
|  | SetBackground(); | 
|  | } | 
|  |  | 
|  | void DialControlBmp::DrawBackground() | 
|  | { | 
|  | // *** background with 3D effect *** | 
|  |  | 
|  | SetLineColor(); | 
|  | SetFillColor(); | 
|  | Erase(); | 
|  |  | 
|  | EnableRTL( sal_True ); // #107807# draw 3D effect in correct direction | 
|  |  | 
|  | sal_uInt8 nDiff = mbEnabled ? 0x18 : 0x10; | 
|  | Color aColor; | 
|  |  | 
|  | aColor = GetBackgroundColor(); | 
|  | SetFillColor( aColor ); | 
|  | DrawPie( maRect, maRect.TopRight(), maRect.TopCenter() ); | 
|  | DrawPie( maRect, maRect.BottomLeft(), maRect.BottomCenter() ); | 
|  |  | 
|  | aColor.DecreaseLuminance( nDiff ); | 
|  | SetFillColor( aColor ); | 
|  | DrawPie( maRect, maRect.BottomCenter(), maRect.TopRight() ); | 
|  |  | 
|  | aColor.DecreaseLuminance( nDiff ); | 
|  | SetFillColor( aColor ); | 
|  | DrawPie( maRect, maRect.BottomRight(), maRect.RightCenter() ); | 
|  |  | 
|  | aColor = GetBackgroundColor(); | 
|  | aColor.IncreaseLuminance( nDiff ); | 
|  | SetFillColor( aColor ); | 
|  | DrawPie( maRect, maRect.TopCenter(), maRect.BottomLeft() ); | 
|  |  | 
|  | aColor.IncreaseLuminance( nDiff ); | 
|  | SetFillColor( aColor ); | 
|  | DrawPie( maRect, maRect.TopLeft(), maRect.LeftCenter() ); | 
|  |  | 
|  | EnableRTL( sal_False ); | 
|  |  | 
|  | // *** calibration *** | 
|  |  | 
|  | Point aStartPos( mnCenterX, mnCenterY ); | 
|  | Color aFullColor( GetScaleLineColor() ); | 
|  | Color aLightColor( GetBackgroundColor() ); | 
|  | aLightColor.Merge( aFullColor, 128 ); | 
|  |  | 
|  | for( int nAngle = 0; nAngle < 360; nAngle += 15 ) | 
|  | { | 
|  | SetLineColor( (nAngle % 45) ? aLightColor : aFullColor ); | 
|  | double fAngle = nAngle * F_PI180; | 
|  | long nX = static_cast< long >( -mnCenterX * cos( fAngle ) ); | 
|  | long nY = static_cast< long >( mnCenterY * sin( fAngle ) ); | 
|  | DrawLine( aStartPos, Point( mnCenterX - nX, mnCenterY - nY ) ); | 
|  | } | 
|  |  | 
|  | // *** clear inner area *** | 
|  |  | 
|  | SetLineColor(); | 
|  | SetFillColor( GetBackgroundColor() ); | 
|  | DrawEllipse( Rectangle( maRect.Left() + DIAL_OUTER_WIDTH, maRect.Top() + DIAL_OUTER_WIDTH, | 
|  | maRect.Right() - DIAL_OUTER_WIDTH, maRect.Bottom() - DIAL_OUTER_WIDTH ) ); | 
|  | } | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  | // ---------------------------------------------------------------------------- | 
|  |  | 
|  | DialControl::DialControl_Impl::DialControl_Impl ( | 
|  | Window& rParent ) : | 
|  | mpBmpEnabled(new DialControlBmp(rParent)), | 
|  | mpBmpDisabled(new DialControlBmp(rParent)), | 
|  | mpBmpBuffered(new DialControlBmp(rParent)), | 
|  | mpLinkField( 0 ), | 
|  | mnAngle( 0 ), | 
|  | mbNoRot( false ) | 
|  | { | 
|  | } | 
|  |  | 
|  | void DialControl::DialControl_Impl::Init( const Size& rWinSize, const Font& rWinFont ) | 
|  | { | 
|  | // "(x - 1) | 1" creates odd value <= x, to have a well-defined center pixel position | 
|  | maWinSize = Size( (rWinSize.Width() - 1) | 1, (rWinSize.Height() - 1) | 1 ); | 
|  | maWinFont = rWinFont; | 
|  |  | 
|  | mnCenterX = maWinSize.Width() / 2; | 
|  | mnCenterY = maWinSize.Height() / 2; | 
|  | maWinFont.SetTransparent( sal_True ); | 
|  |  | 
|  | mpBmpEnabled->DrawBackground( maWinSize, true ); | 
|  | mpBmpDisabled->DrawBackground( maWinSize, false ); | 
|  | mpBmpBuffered->InitBitmap( maWinSize, maWinFont ); | 
|  | } | 
|  |  | 
|  | // ============================================================================ | 
|  |  | 
|  | DialControl::DialControl( Window* pParent, const Size& rSize, const Font& rFont, WinBits nWinStyle ) : | 
|  | Control( pParent, nWinStyle ), | 
|  | mpImpl( new DialControl_Impl( *this ) ) | 
|  | { | 
|  | Init( rSize, rFont ); | 
|  | } | 
|  |  | 
|  | DialControl::DialControl( Window* pParent, const Size& rSize, WinBits nWinStyle ) : | 
|  | Control( pParent, nWinStyle ), | 
|  | mpImpl( new DialControl_Impl( *this ) ) | 
|  | { | 
|  | if( pParent ) | 
|  | Init( rSize, pParent->GetFont() ); | 
|  | else | 
|  | Init( rSize ); | 
|  | } | 
|  |  | 
|  | DialControl::DialControl( Window* pParent, const ResId& rResId ) : | 
|  | Control( pParent, rResId ), | 
|  | mpImpl( new DialControl_Impl( *this ) ) | 
|  | { | 
|  | Init( GetOutputSizePixel() ); | 
|  | } | 
|  |  | 
|  | DialControl::~DialControl() | 
|  | { | 
|  | } | 
|  |  | 
|  | void DialControl::Paint( const Rectangle&  ) | 
|  | { | 
|  | Point aPos; | 
|  | DrawBitmapEx( aPos, mpImpl->mpBmpBuffered->GetBitmapEx( aPos, mpImpl->maWinSize ) ); | 
|  | } | 
|  |  | 
|  | void DialControl::StateChanged( StateChangedType nStateChange ) | 
|  | { | 
|  | if( nStateChange == STATE_CHANGE_ENABLE ) | 
|  | InvalidateControl(); | 
|  |  | 
|  | // update the linked edit field | 
|  | if( mpImpl->mpLinkField ) | 
|  | { | 
|  | NumericField& rField = *mpImpl->mpLinkField; | 
|  | switch( nStateChange ) | 
|  | { | 
|  | case STATE_CHANGE_VISIBLE:  rField.Show( IsVisible() );     break; | 
|  | case STATE_CHANGE_ENABLE:   rField.Enable( IsEnabled() );   break; | 
|  | } | 
|  | } | 
|  |  | 
|  | Control::StateChanged( nStateChange ); | 
|  | } | 
|  |  | 
|  | void DialControl::DataChanged( const DataChangedEvent& rDCEvt ) | 
|  | { | 
|  | if( (rDCEvt.GetType() == DATACHANGED_SETTINGS) && (rDCEvt.GetFlags() & SETTINGS_STYLE) ) | 
|  | { | 
|  | Init( mpImpl->maWinSize, mpImpl->maWinFont ); | 
|  | InvalidateControl(); | 
|  | } | 
|  | Control::DataChanged( rDCEvt ); | 
|  | } | 
|  |  | 
|  | void DialControl::MouseButtonDown( const MouseEvent& rMEvt ) | 
|  | { | 
|  | if( rMEvt.IsLeft() ) | 
|  | { | 
|  | GrabFocus(); | 
|  | CaptureMouse(); | 
|  | mpImpl->mnOldAngle = mpImpl->mnAngle; | 
|  | HandleMouseEvent( rMEvt.GetPosPixel(), true ); | 
|  | } | 
|  | Control::MouseButtonDown( rMEvt ); | 
|  | } | 
|  |  | 
|  | void DialControl::MouseMove( const MouseEvent& rMEvt ) | 
|  | { | 
|  | if( IsMouseCaptured() && rMEvt.IsLeft() ) | 
|  | HandleMouseEvent( rMEvt.GetPosPixel(), false ); | 
|  | Control::MouseMove(rMEvt ); | 
|  | } | 
|  |  | 
|  | void DialControl::MouseButtonUp( const MouseEvent& rMEvt ) | 
|  | { | 
|  | if( IsMouseCaptured() ) | 
|  | { | 
|  | ReleaseMouse(); | 
|  | if( mpImpl->mpLinkField ) | 
|  | mpImpl->mpLinkField->GrabFocus(); | 
|  | } | 
|  | Control::MouseButtonUp( rMEvt ); | 
|  | } | 
|  |  | 
|  | void DialControl::KeyInput( const KeyEvent& rKEvt ) | 
|  | { | 
|  | const KeyCode& rKCode = rKEvt.GetKeyCode(); | 
|  | if( !rKCode.GetModifier() && (rKCode.GetCode() == KEY_ESCAPE) ) | 
|  | HandleEscapeEvent(); | 
|  | else | 
|  | Control::KeyInput( rKEvt ); | 
|  | } | 
|  |  | 
|  | void DialControl::LoseFocus() | 
|  | { | 
|  | // release captured mouse | 
|  | HandleEscapeEvent(); | 
|  | Control::LoseFocus(); | 
|  | } | 
|  |  | 
|  | bool DialControl::HasRotation() const | 
|  | { | 
|  | return !mpImpl->mbNoRot; | 
|  | } | 
|  |  | 
|  | void DialControl::SetNoRotation() | 
|  | { | 
|  | if( !mpImpl->mbNoRot ) | 
|  | { | 
|  | mpImpl->mbNoRot = true; | 
|  | InvalidateControl(); | 
|  | if( mpImpl->mpLinkField ) | 
|  | mpImpl->mpLinkField->SetText( String() ); | 
|  | } | 
|  | } | 
|  |  | 
|  | sal_Int32 DialControl::GetRotation() const | 
|  | { | 
|  | return mpImpl->mnAngle; | 
|  | } | 
|  |  | 
|  | void DialControl::SetRotation( sal_Int32 nAngle ) | 
|  | { | 
|  | SetRotation( nAngle, false ); | 
|  | } | 
|  |  | 
|  | void DialControl::SetLinkedField( NumericField* pField ) | 
|  | { | 
|  | // remove modify handler from old linked field | 
|  | ImplSetFieldLink( Link() ); | 
|  | // remember the new linked field | 
|  | mpImpl->mpLinkField = pField; | 
|  | // set modify handler at new linked field | 
|  | ImplSetFieldLink( LINK( this, DialControl, LinkedFieldModifyHdl ) ); | 
|  | } | 
|  |  | 
|  | NumericField* DialControl::GetLinkedField() const | 
|  | { | 
|  | return mpImpl->mpLinkField; | 
|  | } | 
|  |  | 
|  | void DialControl::SetModifyHdl( const Link& rLink ) | 
|  | { | 
|  | mpImpl->maModifyHdl = rLink; | 
|  | } | 
|  |  | 
|  | const Link& DialControl::GetModifyHdl() const | 
|  | { | 
|  | return mpImpl->maModifyHdl; | 
|  | } | 
|  |  | 
|  | // private -------------------------------------------------------------------- | 
|  |  | 
|  | void DialControl::Init( const Size& rWinSize, const Font& rWinFont ) | 
|  | { | 
|  | mpImpl->Init( rWinSize, rWinFont ); | 
|  | EnableRTL( sal_False ); // #107807# don't mirror mouse handling | 
|  | SetOutputSizePixel( mpImpl->maWinSize ); | 
|  | SetBackground(); | 
|  | } | 
|  |  | 
|  | void DialControl::Init( const Size& rWinSize ) | 
|  | { | 
|  | Font aFont( OutputDevice::GetDefaultFont( | 
|  | DEFAULTFONT_UI_SANS, Application::GetSettings().GetUILanguage(), DEFAULTFONT_FLAGS_ONLYONE ) ); | 
|  | Init( rWinSize, aFont ); | 
|  | } | 
|  |  | 
|  | void DialControl::InvalidateControl() | 
|  | { | 
|  | mpImpl->mpBmpBuffered->CopyBackground( IsEnabled() ? *mpImpl->mpBmpEnabled : *mpImpl->mpBmpDisabled ); | 
|  | if( !mpImpl->mbNoRot ) | 
|  | mpImpl->mpBmpBuffered->DrawElements( GetText(), mpImpl->mnAngle ); | 
|  | Invalidate(); | 
|  | } | 
|  |  | 
|  | void DialControl::SetRotation( sal_Int32 nAngle, bool bBroadcast ) | 
|  | { | 
|  | bool bOldSel = mpImpl->mbNoRot; | 
|  | mpImpl->mbNoRot = false; | 
|  |  | 
|  | while( nAngle < 0 ) nAngle += 36000; | 
|  | nAngle = (((nAngle + 50) / 100) * 100) % 36000; | 
|  | if( !bOldSel || (mpImpl->mnAngle != nAngle) ) | 
|  | { | 
|  | mpImpl->mnAngle = nAngle; | 
|  | InvalidateControl(); | 
|  | if( mpImpl->mpLinkField ) | 
|  | mpImpl->mpLinkField->SetValue( static_cast< long >( GetRotation() / 100 ) ); | 
|  | if( bBroadcast ) | 
|  | mpImpl->maModifyHdl.Call( this ); | 
|  | } | 
|  | } | 
|  |  | 
|  | void DialControl::ImplSetFieldLink( const Link& rLink ) | 
|  | { | 
|  | if( mpImpl->mpLinkField ) | 
|  | { | 
|  | NumericField& rField = *mpImpl->mpLinkField; | 
|  | rField.SetModifyHdl( rLink ); | 
|  | rField.SetUpHdl( rLink ); | 
|  | rField.SetDownHdl( rLink ); | 
|  | rField.SetFirstHdl( rLink ); | 
|  | rField.SetLastHdl( rLink ); | 
|  | rField.SetLoseFocusHdl( rLink ); | 
|  | } | 
|  | } | 
|  |  | 
|  | void DialControl::HandleMouseEvent( const Point& rPos, bool bInitial ) | 
|  | { | 
|  | long nX = rPos.X() - mpImpl->mnCenterX; | 
|  | long nY = mpImpl->mnCenterY - rPos.Y(); | 
|  | double fH = sqrt( static_cast< double >( nX ) * nX + static_cast< double >( nY ) * nY ); | 
|  | if( fH != 0.0 ) | 
|  | { | 
|  | double fAngle = acos( nX / fH ); | 
|  | sal_Int32 nAngle = static_cast< sal_Int32 >( fAngle / F_PI180 * 100.0 ); | 
|  | if( nY < 0 ) | 
|  | nAngle = 36000 - nAngle; | 
|  | if( bInitial )  // round to entire 15 degrees | 
|  | nAngle = ((nAngle + 750) / 1500) * 1500; | 
|  | SetRotation( nAngle, true ); | 
|  | } | 
|  | } | 
|  |  | 
|  | void DialControl::HandleEscapeEvent() | 
|  | { | 
|  | if( IsMouseCaptured() ) | 
|  | { | 
|  | ReleaseMouse(); | 
|  | SetRotation( mpImpl->mnOldAngle, true ); | 
|  | if( mpImpl->mpLinkField ) | 
|  | mpImpl->mpLinkField->GrabFocus(); | 
|  | } | 
|  | } | 
|  |  | 
|  | IMPL_LINK( DialControl, LinkedFieldModifyHdl, NumericField*, pField ) | 
|  | { | 
|  | if( pField ) | 
|  | SetRotation( static_cast< sal_Int32 >( pField->GetValue() * 100 ), false ); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // ============================================================================ | 
|  |  | 
|  | DialControlWrapper::DialControlWrapper( DialControl& rDial ) : | 
|  | SingleControlWrapperType( rDial ) | 
|  | { | 
|  | } | 
|  |  | 
|  | bool DialControlWrapper::IsControlDontKnow() const | 
|  | { | 
|  | return !GetControl().HasRotation(); | 
|  | } | 
|  |  | 
|  | void DialControlWrapper::SetControlDontKnow( bool bSet ) | 
|  | { | 
|  | if( bSet ) | 
|  | GetControl().SetNoRotation(); | 
|  | } | 
|  |  | 
|  | sal_Int32 DialControlWrapper::GetControlValue() const | 
|  | { | 
|  | return GetControl().GetRotation(); | 
|  | } | 
|  |  | 
|  | void DialControlWrapper::SetControlValue( sal_Int32 nValue ) | 
|  | { | 
|  | GetControl().SetRotation( nValue ); | 
|  | } | 
|  |  | 
|  | // ============================================================================ | 
|  |  | 
|  | } // namespace svx |