diff options
Diffstat (limited to 'sd/source/ui/animations/CustomAnimationPane.cxx')
-rw-r--r-- | sd/source/ui/animations/CustomAnimationPane.cxx | 2489 |
1 files changed, 2489 insertions, 0 deletions
diff --git a/sd/source/ui/animations/CustomAnimationPane.cxx b/sd/source/ui/animations/CustomAnimationPane.cxx new file mode 100644 index 000000000000..8b96e08b4a8c --- /dev/null +++ b/sd/source/ui/animations/CustomAnimationPane.cxx @@ -0,0 +1,2489 @@ +/************************************************************************* + * + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * Copyright 2000, 2010 Oracle and/or its affiliates. + * + * OpenOffice.org - a multi-platform office productivity suite + * + * This file is part of OpenOffice.org. + * + * OpenOffice.org is free software: you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License version 3 + * only, as published by the Free Software Foundation. + * + * OpenOffice.org is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Lesser General Public License version 3 for more details + * (a copy is included in the LICENSE file that accompanied this code). + * + * You should have received a copy of the GNU Lesser General Public License + * version 3 along with OpenOffice.org. If not, see + * <http://www.openoffice.org/license.html> + * for a copy of the LGPLv3 License. + * + ************************************************************************/ + +// MARKER(update_precomp.py): autogen include statement, do not remove +#include "precompiled_sd.hxx" + +#include <com/sun/star/presentation/EffectPresetClass.hpp> +#include <com/sun/star/animations/XAnimationNodeSupplier.hpp> +#include <com/sun/star/view/XSelectionSupplier.hpp> +#include <com/sun/star/drawing/XDrawView.hpp> +#include <com/sun/star/drawing/XShape.hpp> +#include <com/sun/star/beans/XPropertySet.hpp> +#include <com/sun/star/presentation/EffectNodeType.hpp> +#include <com/sun/star/presentation/EffectCommands.hpp> +#include <com/sun/star/animations/AnimationTransformType.hpp> +#include <com/sun/star/text/XTextRangeCompare.hpp> +#include <com/sun/star/container/XEnumerationAccess.hpp> +#include <com/sun/star/container/XIndexAccess.hpp> +#include <com/sun/star/presentation/ParagraphTarget.hpp> +#include <com/sun/star/text/XText.hpp> +#include <com/sun/star/awt/XWindow.hpp> +#include <com/sun/star/drawing/LineStyle.hpp> +#include <com/sun/star/drawing/FillStyle.hpp> +#include <comphelper/processfactory.hxx> +#include <sfx2/dispatch.hxx> +#include "STLPropertySet.hxx" +#include "CustomAnimationPane.hxx" +#include "CustomAnimationDialog.hxx" +#include "CustomAnimationCreateDialog.hxx" +#include "CustomAnimationPane.hrc" +#include "CustomAnimation.hrc" +#include "CustomAnimationList.hxx" +#include <vcl/lstbox.hxx> +#include <vcl/fixed.hxx> + +#include <vcl/button.hxx> +#include <vcl/combobox.hxx> +#include <vcl/scrbar.hxx> + +#include <comphelper/sequence.hxx> +#include <sfx2/frame.hxx> + +#include <svx/unoapi.hxx> +#include <svx/svxids.hrc> +#include <DrawDocShell.hxx> +#include <ViewShellBase.hxx> +#include "DrawViewShell.hxx" +#include "DrawController.hxx" +#include "sdresid.hxx" +#include "drawview.hxx" +#include "slideshow.hxx" +#include "undoanim.hxx" +#include "optsitem.hxx" +#include "sddll.hxx" +#include "framework/FrameworkHelper.hxx" + +#include "EventMultiplexer.hxx" +#include "DialogListBox.hxx" + +#include "glob.hrc" +#include "sdpage.hxx" +#include "drawdoc.hxx" +#include "app.hrc" + +#include <memory> +#include <algorithm> + +#include <basegfx/polygon/b2dpolypolygontools.hxx> +#include <basegfx/matrix/b2dhommatrix.hxx> +#include <basegfx/range/b2drange.hxx> + +using namespace ::com::sun::star; +using namespace ::com::sun::star::animations; +using namespace ::com::sun::star::presentation; +using namespace ::com::sun::star::text; + +using ::rtl::OUString; +using namespace ::com::sun::star::uno; +using namespace ::com::sun::star::drawing; +using ::com::sun::star::view::XSelectionSupplier; +using ::com::sun::star::view::XSelectionChangeListener; +using ::com::sun::star::frame::XController; +using ::com::sun::star::frame::XModel; +using ::com::sun::star::beans::XPropertySet; +using ::com::sun::star::beans::XPropertyChangeListener; +using ::com::sun::star::container::XIndexAccess; +using ::com::sun::star::container::XEnumerationAccess; +using ::com::sun::star::container::XEnumeration; +using ::com::sun::star::text::XText; +using ::sd::framework::FrameworkHelper; + +namespace sd { + +// -------------------------------------------------------------------- + +void fillDurationComboBox( ComboBox* pBox ) +{ + static const double gdVerySlow = 5.0; + static const double gdSlow = 3.0; + static const double gdNormal = 2.0; + static const double gdFast = 1.0; + static const double gdVeryFast = 0.5; + + String aVerySlow( SdResId( STR_CUSTOMANIMATION_DURATION_VERY_SLOW ) ); + pBox->SetEntryData( pBox->InsertEntry( aVerySlow ), (void*)&gdVerySlow ); + + String aSlow( SdResId( STR_CUSTOMANIMATION_DURATION_SLOW ) ); + pBox->SetEntryData( pBox->InsertEntry( aSlow ), (void*)&gdSlow ); + + String aNormal( SdResId( STR_CUSTOMANIMATION_DURATION_NORMAL ) ); + pBox->SetEntryData( pBox->InsertEntry( aNormal ), (void*)&gdNormal ); + + String aFast( SdResId( STR_CUSTOMANIMATION_DURATION_FAST ) ); + pBox->SetEntryData( pBox->InsertEntry( aFast ), (void*)&gdFast ); + + String aVeryFast( SdResId( STR_CUSTOMANIMATION_DURATION_VERY_FAST ) ); + pBox->SetEntryData( pBox->InsertEntry( aVeryFast ), (void*)&gdVeryFast ); +} + +void fillRepeatComboBox( ComboBox* pBox ) +{ + String aNone( SdResId( STR_CUSTOMANIMATION_REPEAT_NONE ) ); + pBox->SetEntryData( pBox->InsertEntry( aNone ), (void*)((sal_Int32)0) ); + + pBox->SetEntryData( pBox->InsertEntry( String::CreateFromInt32( 2 ) ), (void*)((sal_Int32)1) ); + pBox->SetEntryData( pBox->InsertEntry( String::CreateFromInt32( 3 ) ), (void*)((sal_Int32)3) ); + pBox->SetEntryData( pBox->InsertEntry( String::CreateFromInt32( 4 ) ), (void*)((sal_Int32)4) ); + pBox->SetEntryData( pBox->InsertEntry( String::CreateFromInt32( 5 ) ), (void*)((sal_Int32)5) ); + pBox->SetEntryData( pBox->InsertEntry( String::CreateFromInt32( 10 ) ), (void*)((sal_Int32)10) ); + + String aUntilClick( SdResId( STR_CUSTOMANIMATION_REPEAT_UNTIL_NEXT_CLICK ) ); + pBox->SetEntryData( pBox->InsertEntry( aUntilClick ), (void*)((sal_Int32)-1) ); + + String aEndOfSlide( SdResId( STR_CUSTOMANIMATION_REPEAT_UNTIL_END_OF_SLIDE ) ); + pBox->SetEntryData( pBox->InsertEntry( aEndOfSlide ), (void*)((sal_Int32)-2) ); +} + +// -------------------------------------------------------------------- + +CustomAnimationPane::CustomAnimationPane( ::Window* pParent, ViewShellBase& rBase, const Size& rMinSize ) +: Control( pParent, SdResId(DLG_CUSTOMANIMATIONPANE) ), + mrBase( rBase ), + mpCustomAnimationPresets(NULL), + mnPropertyType( nPropertyTypeNone ), + maMinSize( rMinSize ), + mxModel( rBase.GetDocShell()->GetDoc()->getUnoModel(), UNO_QUERY ), + maLateInitTimer() +{ + // load resources + mpFLEffect = new FixedLine( this, SdResId( FL_EFFECT ) ); + + mpPBAddEffect = new PushButton( this, SdResId( PB_ADD_EFFECT ) ); + mpPBChangeEffect = new PushButton( this, SdResId( PB_CHANGE_EFFECT ) ); + mpPBRemoveEffect = new PushButton( this, SdResId( PB_REMOVE_EFFECT ) ); + + mpFLModify = new FixedLine( this, SdResId( FL_MODIFY ) ); + + mpFTStart = new FixedText( this, SdResId( FT_START ) ); + mpLBStart = new ListBox( this, SdResId( LB_START ) ); + mpFTProperty = new FixedText( this, SdResId( FT_PROPERTY ) ); + mpLBProperty = new PropertyControl( this, SdResId( LB_PROPERTY ) ); + mpPBPropertyMore = new PushButton( this, SdResId( PB_PROPERTY_MORE ) ); + + mpFTSpeed = new FixedText( this, SdResId( FT_SPEED ) ); + mpCBSpeed = new ComboBox( this, SdResId( CB_SPEED ) ); + + mpCustomAnimationList = new CustomAnimationList( this, SdResId( CT_CUSTOM_ANIMATION_LIST ), this ); + + mpPBMoveUp = new PushButton( this, SdResId( PB_MOVE_UP ) ); + mpPBMoveDown = new PushButton( this, SdResId( PB_MOVE_DOWN ) ); + mpFTChangeOrder = new FixedText( this, SdResId( FT_CHANGE_ORDER ) ); + mpFLSeperator1 = new FixedLine( this, SdResId( FL_SEPERATOR1 ) ); + mpPBPlay = new PushButton( this, SdResId( PB_PLAY ) ); + mpPBSlideShow = new PushButton( this, SdResId( PB_SLIDE_SHOW ) ); + mpFLSeperator2 = new FixedLine( this, SdResId( FL_SEPERATOR2 ) ); + mpCBAutoPreview = new CheckBox( this, SdResId( CB_AUTOPREVIEW ) ); + + maStrProperty = mpFTProperty->GetText(); + + FreeResource(); + + // use bold font for group headings (same font for all fixed lines): + Font font( mpFLEffect->GetFont() ); + font.SetWeight( WEIGHT_BOLD ); + mpFLEffect->SetFont( font ); + mpFLModify->SetFont( font ); + + fillDurationComboBox( mpCBSpeed ); + mpPBMoveUp->SetSymbol( SYMBOL_ARROW_UP ); + mpPBMoveDown->SetSymbol( SYMBOL_ARROW_DOWN ); + + mpPBAddEffect->SetClickHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + mpPBChangeEffect->SetClickHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + mpPBRemoveEffect->SetClickHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + mpLBStart->SetSelectHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + mpCBSpeed->SetSelectHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + mpPBPropertyMore->SetClickHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + mpPBMoveUp->SetClickHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + mpPBMoveDown->SetClickHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + mpPBPlay->SetClickHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + mpPBSlideShow->SetClickHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + mpCBAutoPreview->SetClickHdl( LINK( this, CustomAnimationPane, implControlHdl ) ); + + maStrModify = mpFLEffect->GetText(); + + // resize controls according to current size + updateLayout(); + + // get current controller and initialize listeners + try + { + mxView = Reference< XDrawView >::query(mrBase.GetController()); + addListener(); + } + catch( Exception& e ) + { + (void)e; + DBG_ERROR( "sd::CustomAnimationPane::CustomAnimationPane(), Exception cought!" ); + } + + // get current page and update custom animation list + onChangeCurrentPage(); + + // Wait a short time before the presets list is created. This gives the + // system time to paint the control. + maLateInitTimer.SetTimeout(100); + maLateInitTimer.SetTimeoutHdl(LINK(this, CustomAnimationPane, lateInitCallback)); + maLateInitTimer.Start(); +} + +CustomAnimationPane::~CustomAnimationPane() +{ + maLateInitTimer.Stop(); + + removeListener(); + + MotionPathTagVector aTags; + aTags.swap( maMotionPathTags ); + MotionPathTagVector::iterator aIter; + for( aIter = aTags.begin(); aIter != aTags.end(); aIter++ ) + (*aIter)->Dispose(); + + delete mpFLModify; + delete mpPBAddEffect; + delete mpPBChangeEffect; + delete mpPBRemoveEffect; + delete mpFLEffect; + delete mpFTStart; + delete mpLBStart; + delete mpFTProperty; + delete mpLBProperty; + delete mpPBPropertyMore; + delete mpFTSpeed; + delete mpCBSpeed; + delete mpCustomAnimationList; + delete mpFTChangeOrder; + delete mpPBMoveUp; + delete mpPBMoveDown; + delete mpFLSeperator1; + delete mpPBPlay; + delete mpPBSlideShow; + delete mpFLSeperator2; + delete mpCBAutoPreview; +} + +void CustomAnimationPane::addUndo() +{ + SfxUndoManager* pManager = mrBase.GetDocShell()->GetUndoManager(); + if( pManager ) + { + SdPage* pPage = SdPage::getImplementation( mxCurrentPage ); + if( pPage ) + pManager->AddUndoAction( new UndoAnimation( mrBase.GetDocShell()->GetDoc(), pPage ) ); + } +} + +void CustomAnimationPane::Resize() +{ + updateLayout(); +} + +void CustomAnimationPane::StateChanged( StateChangedType nStateChange ) +{ + Control::StateChanged( nStateChange ); + + if( nStateChange == STATE_CHANGE_VISIBLE ) + updateMotionPathTags(); +} + +void CustomAnimationPane::KeyInput( const KeyEvent& rKEvt ) +{ + if( mpCustomAnimationList ) + mpCustomAnimationList->KeyInput( rKEvt ); +} + +void CustomAnimationPane::addListener() +{ + Link aLink( LINK(this,CustomAnimationPane,EventMultiplexerListener) ); + mrBase.GetEventMultiplexer()->AddEventListener ( + aLink, + tools::EventMultiplexerEvent::EID_EDIT_VIEW_SELECTION + | tools::EventMultiplexerEvent::EID_CURRENT_PAGE + | tools::EventMultiplexerEvent::EID_MAIN_VIEW_REMOVED + | tools::EventMultiplexerEvent::EID_MAIN_VIEW_ADDED + | tools::EventMultiplexerEvent::EID_DISPOSING + | tools::EventMultiplexerEvent::EID_END_TEXT_EDIT); +} + +void CustomAnimationPane::removeListener() +{ + Link aLink( LINK(this,CustomAnimationPane,EventMultiplexerListener) ); + mrBase.GetEventMultiplexer()->RemoveEventListener( aLink ); +} + +IMPL_LINK(CustomAnimationPane,EventMultiplexerListener, + tools::EventMultiplexerEvent*,pEvent) +{ + switch (pEvent->meEventId) + { + case tools::EventMultiplexerEvent::EID_EDIT_VIEW_SELECTION: + onSelectionChanged(); + break; + + case tools::EventMultiplexerEvent::EID_CURRENT_PAGE: + onChangeCurrentPage(); + break; + + case tools::EventMultiplexerEvent::EID_MAIN_VIEW_ADDED: + // At this moment the controller may not yet been set at model + // or ViewShellBase. Take it from the view shell passed with + // the event. + if (mrBase.GetMainViewShell() != NULL) + { + if( mrBase.GetMainViewShell()->GetShellType() == ViewShell::ST_IMPRESS ) + { + mxView = Reference<XDrawView>::query(mrBase.GetDrawController()); + onSelectionChanged(); + onChangeCurrentPage(); + break; + } + } + // fall through intended + case tools::EventMultiplexerEvent::EID_MAIN_VIEW_REMOVED: + mxView = 0; + mxCurrentPage = 0; + updateControls(); + break; + + case tools::EventMultiplexerEvent::EID_DISPOSING: + mxView = Reference<XDrawView>(); + onSelectionChanged(); + onChangeCurrentPage(); + break; + case tools::EventMultiplexerEvent::EID_END_TEXT_EDIT: + if( mpMainSequence.get() && pEvent->mpUserData ) + mpCustomAnimationList->update( mpMainSequence ); + break; + } + return 0; +} + + +void CustomAnimationPane::updateLayout() +{ + Size aPaneSize( GetSizePixel() ); + if( aPaneSize.Width() < maMinSize.Width() ) + aPaneSize.Width() = maMinSize.Width(); + + if( aPaneSize.Height() < maMinSize.Height() ) + aPaneSize.Height() = maMinSize.Height(); + + Point aOffset( LogicToPixel( Point(3,3), MAP_APPFONT ) ); + Point aCursor( aOffset ); + + // place the modify fixed line + + // place the "modify effect" fixed line + Size aSize( mpFLModify->GetSizePixel() ); + aSize.Width() = aPaneSize.Width() - 2 * aOffset.X(); + + mpFLModify->SetPosSizePixel( aCursor, aSize ); + + aCursor.Y() += aSize.Height() + aOffset.Y(); + + const int nButtonExtraWidth = 4 * aOffset.X(); + + // the "add effect" button is placed top-left + Size aCtrlSize( mpPBAddEffect->GetSizePixel() ); + aCtrlSize.setWidth( mpPBAddEffect->CalcMinimumSize( aSize.Width() ).getWidth() + nButtonExtraWidth ); + mpPBAddEffect->SetPosSizePixel( aCursor, aCtrlSize ); + + aCursor.X() += aOffset.X() + aCtrlSize.Width(); + + // place the "change effect" button + + // if the "change" button does not fit right of the "add effect", put it on the next line + aCtrlSize = mpPBChangeEffect->GetSizePixel(); + aCtrlSize.setWidth( mpPBChangeEffect->CalcMinimumSize( aSize.Width() ).getWidth() + nButtonExtraWidth ); + if( ( aCursor.X() + aCtrlSize.Width() + aOffset.X() ) > aPaneSize.Width() ) + { + aCursor.X() = aOffset.X(); + aCursor.Y() += aCtrlSize.Height() + aOffset.Y(); + } + mpPBChangeEffect->SetPosSizePixel( aCursor, aCtrlSize ); + + aCursor.X() += aOffset.X() + aCtrlSize.Width(); + + // place the "remove effect" button + + // if the "remove" button does not fit right of the "add effect", put it on the next line + aCtrlSize = mpPBRemoveEffect->GetSizePixel(); + aCtrlSize.setWidth( mpPBRemoveEffect->CalcMinimumSize( aSize.Width() ).getWidth() + nButtonExtraWidth ); + if( ( aCursor.X() + aCtrlSize.Width() + aOffset.X() ) > aPaneSize.Width() ) + { + aCursor.X() = aOffset.X(); + aCursor.Y() += aCtrlSize.Height() + aOffset.Y(); + } + + mpPBRemoveEffect->SetPosSizePixel( aCursor, aCtrlSize ); + + aCursor.X() = aOffset.X(); + aCursor.Y() += aCtrlSize.Height() + 2 * aOffset.Y(); + + // place the "modify effect" fixed line + aSize = mpFLEffect->GetSizePixel(); + aSize.Width() = aPaneSize.Width() - 2 * aOffset.X(); + + mpFLEffect->SetPosSizePixel( aCursor, aSize ); + + aCursor.Y() += aSize.Height() + aOffset.Y(); + + // --------------------------------------------------------------------------- + // place the properties controls + + // calc minimum width for fixedtext + + Size aFixedTextSize( mpFTStart->CalcMinimumSize() ); + long nWidth = aFixedTextSize.Width(); + aFixedTextSize = mpFTProperty->CalcMinimumSize(); + nWidth = std::max( nWidth, aFixedTextSize.Width() ); + aFixedTextSize = mpFTSpeed->CalcMinimumSize(); + aFixedTextSize.Width() = std::max( nWidth, aFixedTextSize.Width() ) + aOffset.X(); + mpFTStart->SetSizePixel(aFixedTextSize); + mpFTProperty->SetSizePixel(aFixedTextSize); + mpFTSpeed->SetSizePixel(aFixedTextSize); + + aSize = mpPBPropertyMore->GetSizePixel(); + + // place the "start" fixed text + + Point aFTPos( aCursor ); + Point aLBPos( aCursor ); + Size aListBoxSize( LogicToPixel( Size( 60, 12 ), MAP_APPFONT ) ); + long nDeltaY = aListBoxSize.Height() + aOffset.Y(); + + // linebreak? + if( (aFixedTextSize.Width() + aListBoxSize.Width() + aSize.Width() + 4 * aOffset.X()) > aPaneSize.Width() ) + { + // y position for list box is below fixed text + aLBPos.Y() += aFixedTextSize.Height() + aOffset.Y(); + + // height of fixed text + list box + something = 2 * list box + nDeltaY = aListBoxSize.Height() + aFixedTextSize.Height() + 2*aOffset.Y(); + } + else + { + // x position for list box is right of fixed text + aLBPos.X() += aFixedTextSize.Width() + aOffset.X(); + + if( aListBoxSize.Height() > aFixedTextSize.Height() ) + aFTPos.Y() = aLBPos.Y() + ((aListBoxSize.Height() - aFixedTextSize.Height()) >> 1); + else + aLBPos.Y() = aFTPos.Y() + ((aFixedTextSize.Height() - aListBoxSize.Height()) >> 1); + } + + // width of the listbox is from its left side until end of pane + aListBoxSize.Width() = aPaneSize.Width() - aLBPos.X() - aSize.Width() - 2 * aOffset.X(); + + mpFTStart->SetPosPixel( aFTPos ); + mpLBStart->SetPosSizePixel( aLBPos, aListBoxSize ); + + aFTPos.Y() += nDeltaY; aLBPos.Y() += nDeltaY; + + mpFTProperty->SetPosPixel( aFTPos ); + mpLBProperty->SetPosSizePixel( aLBPos, aListBoxSize ); + mpLBProperty->Resize(); + + Point aMorePos( aLBPos ); + aMorePos.X() += aListBoxSize.Width() + aOffset.X(); + mpPBPropertyMore->SetPosPixel( aMorePos ); + + aFTPos.Y() += nDeltaY; aLBPos.Y() += nDeltaY; + + mpFTSpeed->SetPosPixel( aFTPos ); + mpCBSpeed->SetPosSizePixel( aLBPos, aListBoxSize ); + + aFTPos.Y() += nDeltaY + aOffset.Y(); + + Point aListPos( aFTPos ); + + // positionate the buttons on the bottom + + // place the auto preview checkbox + aCursor = Point( aOffset.X(), aPaneSize.Height() - mpCBAutoPreview->GetSizePixel().Height() - aOffset.Y() ); + mpCBAutoPreview->SetPosPixel( aCursor ); + + // place the seperator 2 fixed line + aCursor.Y() -= /* aOffset.Y() + */ mpFLSeperator2->GetSizePixel().Height(); + aSize = mpFLSeperator2->GetSizePixel(); + aSize.Width() = aPaneSize.Width() - 2 * aOffset.X(); + mpFLSeperator2->SetPosSizePixel( aCursor, aSize ); + + // next, layout and place the play and slide show buttons + aCtrlSize = mpPBSlideShow->GetSizePixel(); + aCtrlSize.setWidth( mpPBSlideShow->CalcMinimumSize( aSize.Width() ).getWidth() + nButtonExtraWidth ); + + Size aPlaySize( mpPBPlay->GetSizePixel() ); + aPlaySize.setWidth( mpPBPlay->CalcMinimumSize( aSize.Width() ).getWidth() + nButtonExtraWidth ); + + aCursor.Y() -= aCtrlSize.Height() /* + aOffset.Y() */; + + // do we need two lines for the buttons? + int aTestWidth = aCursor.X() + mpPBPlay->GetSizePixel().Width() + 2 * aOffset.X() + mpPBSlideShow->GetSizePixel().Width(); + if( aTestWidth > aPaneSize.Width() ) + { + mpPBSlideShow->SetPosSizePixel( aCursor, aCtrlSize ); + aCursor.Y() -= aCtrlSize.Height() + aOffset.Y(); + mpPBPlay->SetPosSizePixel( aCursor, aPlaySize ); + } + else + { + mpPBPlay->SetPosSizePixel( aCursor, aPlaySize ); + aCursor.X() += aPlaySize.Width() + aOffset.X(); + mpPBSlideShow->SetPosSizePixel( aCursor, aCtrlSize ); + } + + // place the seperator 1 fixed line + aCursor.X() = aOffset.X(); + aCursor.Y() -= /* aOffset.Y() + */ mpFLSeperator1->GetSizePixel().Height(); + aSize = mpFLSeperator1->GetSizePixel(); + aSize.Width() = aPaneSize.Width() - 2 * aOffset.X(); + mpFLSeperator1->SetPosSizePixel( aCursor, aSize ); + + // place the move down button + aSize = mpPBMoveDown->GetSizePixel(); + + aCursor.X() = aPaneSize.Width() - aOffset.X() - aSize.Width(); + aCursor.Y() -= aOffset.Y() + aSize.Height(); + mpPBMoveDown->SetPosPixel( aCursor ); + + aCursor.X() -= aOffset.X() + aSize.Width(); + mpPBMoveUp->SetPosPixel( aCursor ); + + // Place the change order label. + // Its width has to be calculated dynamically so that is can be + // displayed flush right without having too much space to the buttons + // with some languages or truncated text with others. + mpFTChangeOrder->SetSizePixel(mpFTChangeOrder->CalcMinimumSize()); + + aCursor.X() -= aOffset.X() + mpFTChangeOrder->GetSizePixel().Width(); + aCursor.Y() += (aSize.Height() - mpFTChangeOrder->GetSizePixel().Height()) >> 1; + mpFTChangeOrder->SetPosPixel( aCursor ); + + // positionate the custom animation list control + Size aCustomAnimationListSize( aPaneSize.Width() - aListPos.X() - aOffset.X(), aCursor.Y() - aListPos.Y() - 2 * aOffset.Y() ); + mpCustomAnimationList->SetPosSizePixel( aListPos, aCustomAnimationListSize ); +} + +static sal_Int32 getPropertyType( const OUString& rProperty ) +{ + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Direction") ) ) + return nPropertyTypeDirection; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Spokes") ) ) + return nPropertyTypeSpokes; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Zoom") ) ) + return nPropertyTypeZoom; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Accelerate") ) ) + return nPropertyTypeAccelerate; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Decelerate") ) ) + return nPropertyTypeDecelerate; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Color1") ) ) + return nPropertyTypeFirstColor; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Color2") ) ) + return nPropertyTypeSecondColor; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("FillColor") ) ) + return nPropertyTypeFillColor; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("ColorStyle") ) ) + return nPropertyTypeColorStyle; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("AutoReverse") ) ) + return nPropertyTypeAutoReverse; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("FontStyle") ) ) + return nPropertyTypeFont; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("CharColor") ) ) + return nPropertyTypeCharColor; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("CharHeight") ) ) + return nPropertyTypeCharHeight; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("CharDecoration") ) ) + return nPropertyTypeCharDecoration; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("LineColor") ) ) + return nPropertyTypeLineColor; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Rotate") ) ) + return nPropertyTypeRotate; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Transparency") ) ) + return nPropertyTypeTransparency; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Color") ) ) + return nPropertyTypeColor; + + if( rProperty.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("Scale") ) ) + return nPropertyTypeScale; + + return nPropertyTypeNone; +} + +OUString getPropertyName( sal_Int32 nPropertyType ) +{ + switch( nPropertyType ) + { + case nPropertyTypeDirection: + return OUString( String( SdResId( STR_CUSTOMANIMATION_DIRECTION_PROPERTY ) ) ); + + case nPropertyTypeSpokes: + return OUString( String( SdResId( STR_CUSTOMANIMATION_SPOKES_PROPERTY ) ) ); + + case nPropertyTypeFirstColor: + return OUString( String( SdResId( STR_CUSTOMANIMATION_FIRST_COLOR_PROPERTY ) ) ); + + case nPropertyTypeSecondColor: + return OUString( String( SdResId( STR_CUSTOMANIMATION_SECOND_COLOR_PROPERTY ) ) ); + + case nPropertyTypeZoom: + return OUString( String( SdResId( STR_CUSTOMANIMATION_ZOOM_PROPERTY ) ) ); + + case nPropertyTypeFillColor: + return OUString( String( SdResId( STR_CUSTOMANIMATION_FILL_COLOR_PROPERTY ) ) ); + + case nPropertyTypeColorStyle: + return OUString( String( SdResId( STR_CUSTOMANIMATION_STYLE_PROPERTY ) ) ); + + case nPropertyTypeFont: + return OUString( String( SdResId( STR_CUSTOMANIMATION_FONT_PROPERTY ) ) ); + + case nPropertyTypeCharHeight: + return OUString( String( SdResId( STR_CUSTOMANIMATION_SIZE_PROPERTY ) ) ); + + case nPropertyTypeCharColor: + return OUString( String( SdResId( STR_CUSTOMANIMATION_FONT_COLOR_PROPERTY ) ) ); + + case nPropertyTypeCharHeightStyle: + return OUString( String( SdResId( STR_CUSTOMANIMATION_FONT_SIZE_STYLE_PROPERTY ) ) ); + + case nPropertyTypeCharDecoration: + return OUString( String( SdResId( STR_CUSTOMANIMATION_FONT_STYLE_PROPERTY ) ) ); + + case nPropertyTypeLineColor: + return OUString( String( SdResId( STR_CUSTOMANIMATION_LINE_COLOR_PROPERTY ) ) ); + + case nPropertyTypeRotate: + return OUString( String( SdResId( STR_CUSTOMANIMATION_AMOUNT_PROPERTY ) ) ); + + case nPropertyTypeColor: + return OUString( String( SdResId( STR_CUSTOMANIMATION_COLOR_PROPERTY ) ) ); + + case nPropertyTypeTransparency: + return OUString( String( SdResId( STR_CUSTOMANIMATION_AMOUNT_PROPERTY ) ) ); + + case nPropertyTypeScale: + return OUString( String( SdResId( STR_CUSTOMANIMATION_SCALE_PROPERTY ) ) ); + } + + OUString aStr; + return aStr; +} + +void CustomAnimationPane::updateControls() +{ + mpFLModify->Enable( mxView.is() ); + mpFTSpeed->Enable( mxView.is() ); + mpCBSpeed->Enable( mxView.is() ); + mpCustomAnimationList->Enable( mxView.is() ); + mpFTChangeOrder->Enable( mxView.is() ); + mpPBMoveUp->Enable( mxView.is() ); + mpPBMoveDown->Enable( mxView.is() ); + mpFLSeperator1->Enable( mxView.is() ); + mpPBPlay->Enable( mxView.is() ); + mpPBSlideShow->Enable( mxView.is() ); + mpFLSeperator2->Enable( mxView.is() ); + mpCBAutoPreview->Enable( mxView.is() ); + + if( !mxView.is() ) + { + mpPBAddEffect->Enable( FALSE ); + mpPBChangeEffect->Enable( FALSE ); + mpPBRemoveEffect->Enable( FALSE ); + mpFLEffect->Enable( FALSE ); + mpFTStart->Enable( FALSE ); + mpLBStart->Enable( FALSE ); + mpPBPropertyMore->Enable( FALSE ); + mpLBProperty->Enable( FALSE ); + mpFTProperty->Enable( FALSE ); + mpCustomAnimationList->clear(); + return; + } + + const int nSelectionCount = maListSelection.size(); + + mpPBAddEffect->Enable( maViewSelection.hasValue() ); + mpPBChangeEffect->Enable( nSelectionCount); + mpPBRemoveEffect->Enable(nSelectionCount); + + mpFLEffect->Enable(nSelectionCount > 0); + mpFTStart->Enable(nSelectionCount > 0); + mpLBStart->Enable(nSelectionCount > 0); + mpPBPropertyMore->Enable(nSelectionCount > 0); + +// mpPBPlay->Enable(nSelectionCount > 0); + + mpFTProperty->SetText( maStrProperty ); + + mnPropertyType = nPropertyTypeNone; + + if( nSelectionCount == 1 ) + { + CustomAnimationEffectPtr pEffect = maListSelection.front(); + + OUString aUIName( getPresets().getUINameForPresetId( pEffect->getPresetId() ) ); + + OUString aTemp( maStrModify ); + + if( aUIName.getLength() ) + { + aTemp += OUString( (sal_Unicode)' ' ); + aTemp += aUIName; + } + mpFLEffect->SetText( aTemp ); + + CustomAnimationPresetPtr pDescriptor = getPresets().getEffectDescriptor( pEffect->getPresetId() ); + if( pDescriptor.get() ) + { + PropertySubControl* pSubControl = NULL; + + Any aValue; + + UStringList aProperties( pDescriptor->getProperties() ); + if( aProperties.size() >= 1 ) + { + OUString aProperty( aProperties.front() ); + + mnPropertyType = getPropertyType( aProperties.front() ); + + mpFTProperty->SetText( getPropertyName( mnPropertyType ) ); + + aValue = getProperty1Value( mnPropertyType, pEffect ); + } + + if( aValue.hasValue() ) + { + pSubControl = mpLBProperty->getSubControl(); + if( !pSubControl || (pSubControl->getControlType() != mnPropertyType) ) + { + pSubControl = PropertySubControl::create( mnPropertyType, this, aValue, pEffect->getPresetId(), LINK( this, CustomAnimationPane, implPropertyHdl ) ); + mpLBProperty->setSubControl( pSubControl ); + } + else + { + pSubControl->setValue( aValue, pEffect->getPresetId() ); + } + } + else + { + mpLBProperty->setSubControl( 0 ); + } + + bool bEnable = (pSubControl != 0) && (pSubControl->getControl()->IsEnabled()); + mpLBProperty->Enable( bEnable ); + mpFTProperty->Enable( bEnable ); + } + else + { + mpLBProperty->setSubControl( 0 ); + mpFTProperty->Enable( FALSE ); + mpLBProperty->Enable( FALSE ); + mpPBPropertyMore->Enable( FALSE ); + } + + // + // --- + // + USHORT nPos = 0xffff; + + sal_Int16 nNodeType = pEffect->getNodeType(); + switch( nNodeType ) + { + case EffectNodeType::ON_CLICK: nPos = 0; break; + case EffectNodeType::WITH_PREVIOUS: nPos = 1; break; + case EffectNodeType::AFTER_PREVIOUS: nPos = 2; break; + } + + mpLBStart->SelectEntryPos( nPos ); + + double fDuration = pEffect->getDuration(); + const bool bHasSpeed = fDuration > 0.001; + + mpFTSpeed->Enable(bHasSpeed); + mpCBSpeed->Enable(bHasSpeed); + + if( bHasSpeed ) + { + if( fDuration == 5.0 ) + nPos = 0; + else if( fDuration == 3.0 ) + nPos = 1; + else if( fDuration == 2.0 ) + nPos = 2; + else if( fDuration == 1.0 ) + nPos = 3; + else if( fDuration == 0.5 ) + nPos = 4; + else + nPos = 0xffff; + + mpCBSpeed->SelectEntryPos( nPos ); + } + + mpPBPropertyMore->Enable( TRUE ); + + mpFTChangeOrder->Enable( TRUE ); + } + else + { + mpLBProperty->setSubControl( 0 ); + mpFTProperty->Enable( FALSE ); + mpLBProperty->Enable( FALSE ); + mpPBPropertyMore->Enable( FALSE ); + mpFTSpeed->Enable(FALSE); + mpCBSpeed->Enable(FALSE); + mpFTChangeOrder->Enable( FALSE ); + mpLBStart->SetNoSelection(); + mpCBSpeed->SetNoSelection(); + mpFLEffect->SetText( maStrModify ); + } + + bool bEnableUp = true; + bool bEnableDown = true; + if( nSelectionCount == 0 ) + { + bEnableUp = false; + bEnableDown = false; + } + else + { + if( mpMainSequence->find( maListSelection.front() ) == mpMainSequence->getBegin() ) + bEnableUp = false; + + EffectSequence::iterator aIter( mpMainSequence->find( maListSelection.back() ) ); + if( aIter == mpMainSequence->getEnd() ) + { + bEnableDown = false; + } + else + { + do + { + aIter++; + } + while( (aIter != mpMainSequence->getEnd()) && !(mpCustomAnimationList->isExpanded((*aIter)) ) ); + + if( aIter == mpMainSequence->getEnd() ) + bEnableDown = false; + } + + if( bEnableUp || bEnableDown ) + { + MainSequenceRebuildGuard aGuard( mpMainSequence ); + + EffectSequenceHelper* pSequence = 0; + EffectSequence::iterator aRebuildIter( maListSelection.begin() ); + const EffectSequence::iterator aRebuildEnd( maListSelection.end() ); + while( aRebuildIter != aRebuildEnd ) + { + CustomAnimationEffectPtr pEffect = (*aRebuildIter++); + + if( pEffect.get() ) + { + if( pSequence == 0 ) + { + pSequence = pEffect->getEffectSequence(); + } + else + { + if( pSequence != pEffect->getEffectSequence() ) + { + bEnableUp = false; + bEnableDown = false; + break; + } + } + } + } + } + } + + mpPBMoveUp->Enable(bEnableUp); + mpPBMoveDown->Enable(bEnableDown); + + SdOptions* pOptions = SD_MOD()->GetSdOptions(DOCUMENT_TYPE_IMPRESS); + mpCBAutoPreview->Check( pOptions->IsPreviewChangedEffects() == sal_True ); + + updateMotionPathTags(); +} + +static bool updateMotionPathImpl( CustomAnimationPane& rPane, ::sd::View& rView, EffectSequence::iterator aIter, EffectSequence::iterator aEnd, MotionPathTagVector& rOldTags, MotionPathTagVector& rNewTags ) +{ + bool bChanges = false; + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect( (*aIter++) ); + if( pEffect.get() && pEffect->getPresetClass() == ::com::sun::star::presentation::EffectPresetClass::MOTIONPATH ) + { + rtl::Reference< MotionPathTag > xMotionPathTag; + // first try to find if there is already a tag for this + MotionPathTagVector::iterator aMIter( rOldTags.begin() ); + for( ; aMIter != rOldTags.end(); aMIter++ ) + { + rtl::Reference< MotionPathTag > xTag( (*aMIter) ); + if( xTag->getEffect() == pEffect ) + { + if( !xTag->isDisposed() ) + { + xMotionPathTag = xTag; + rOldTags.erase( aMIter ); + } + break; + } + } + + // if not found, create new one + if( !xMotionPathTag.is() ) + { + xMotionPathTag.set( new MotionPathTag( rPane, rView, pEffect ) ); + bChanges = true; + } + + if( xMotionPathTag.is() ) + rNewTags.push_back( xMotionPathTag ); + } + } + + return bChanges; +} + +void CustomAnimationPane::updateMotionPathTags() +{ + bool bChanges = false; + + MotionPathTagVector aTags; + aTags.swap( maMotionPathTags ); + + ::sd::View* pView = 0; + + if( mxView.is() ) + { + ::boost::shared_ptr<ViewShell> xViewShell( mrBase.GetMainViewShell() ); + if( xViewShell.get() ) + pView = xViewShell->GetView(); + } + + if( IsVisible() && mpMainSequence.get() && pView ) + { + bChanges = updateMotionPathImpl( *this, *pView, mpMainSequence->getBegin(), mpMainSequence->getEnd(), aTags, maMotionPathTags ); + + const InteractiveSequenceList& rISL = mpMainSequence->getInteractiveSequenceList(); + InteractiveSequenceList::const_iterator aISI( rISL.begin() ); + while( aISI != rISL.end() ) + { + InteractiveSequencePtr pIS( (*aISI++) ); + bChanges |= updateMotionPathImpl( *this, *pView, pIS->getBegin(), pIS->getEnd(), aTags, maMotionPathTags ); + } + } + + if( !aTags.empty() ) + { + bChanges = true; + MotionPathTagVector::iterator aIter( aTags.begin() ); + while( aIter != aTags.end() ) + { + rtl::Reference< MotionPathTag > xTag( (*aIter++) ); + xTag->Dispose(); + } + } + + if( bChanges && pView ) + pView->updateHandles(); +} + +void CustomAnimationPane::onSelectionChanged() +{ + if( !maSelectionLock.isLocked() ) + { + ScopeLockGuard aGuard( maSelectionLock ); + + if( mxView.is() ) try + { + Reference< XSelectionSupplier > xSel( mxView, UNO_QUERY_THROW ); + if (xSel.is()) + { + maViewSelection = xSel->getSelection(); + mpCustomAnimationList->onSelectionChanged( maViewSelection ); + updateControls(); + } + } + catch( Exception& ) + { + DBG_ERROR( "sd::CustomAnimationPane::onSelectionChanged(), Exception catched!" ); + } + } +} + +void CustomAnimationPane::onDoubleClick() +{ + showOptions(); +} + +void CustomAnimationPane::onContextMenu( USHORT nSelectedPopupEntry ) +{ + switch( nSelectedPopupEntry ) + { + case CM_WITH_CLICK: onChangeStart( EffectNodeType::ON_CLICK ); break; + case CM_WITH_PREVIOUS: onChangeStart( EffectNodeType::WITH_PREVIOUS ); break; + case CM_AFTER_PREVIOUS: onChangeStart( EffectNodeType::AFTER_PREVIOUS ); break; + case CM_OPTIONS: showOptions(); break; + case CM_DURATION: showOptions(RID_TP_CUSTOMANIMATION_DURATION); break; + case CM_REMOVE: onRemove(); break; + case CM_CREATE: if( maViewSelection.hasValue() ) onChange( true ); break; + } + + updateControls(); +} + +void addValue( STLPropertySet* pSet, sal_Int32 nHandle, const Any& rValue ) +{ + switch( pSet->getPropertyState( nHandle ) ) + { + case STLPropertyState_AMBIGUOUS: + // value is already ambiguous, do nothing + break; + case STLPropertyState_DIRECT: + // set to ambiguous if existing value is different + if( rValue != pSet->getPropertyValue( nHandle ) ) + pSet->setPropertyState( nHandle, STLPropertyState_AMBIGUOUS ); + break; + case STLPropertyState_DEFAULT: + // just set new value + pSet->setPropertyValue( nHandle, rValue ); + break; + } +} + +static sal_Int32 calcMaxParaDepth( Reference< XShape > xTargetShape ) +{ + sal_Int32 nMaxParaDepth = -1; + + if( xTargetShape.is() ) + { + Reference< XEnumerationAccess > xText( xTargetShape, UNO_QUERY ); + if( xText.is() ) + { + Reference< XPropertySet > xParaSet; + const OUString strNumberingLevel( RTL_CONSTASCII_USTRINGPARAM("NumberingLevel") ); + + Reference< XEnumeration > xEnumeration( xText->createEnumeration(), UNO_QUERY_THROW ); + while( xEnumeration->hasMoreElements() ) + { + xEnumeration->nextElement() >>= xParaSet; + if( xParaSet.is() ) + { + sal_Int32 nParaDepth = 0; + xParaSet->getPropertyValue( strNumberingLevel ) >>= nParaDepth; + + if( nParaDepth > nMaxParaDepth ) + nMaxParaDepth = nParaDepth; + } + } + } + } + + return nMaxParaDepth + 1; +} + +Any CustomAnimationPane::getProperty1Value( sal_Int32 nType, CustomAnimationEffectPtr pEffect ) +{ + switch( nType ) + { + case nPropertyTypeDirection: + case nPropertyTypeSpokes: + case nPropertyTypeZoom: + return makeAny( pEffect->getPresetSubType() ); + + case nPropertyTypeColor: + case nPropertyTypeFillColor: + case nPropertyTypeFirstColor: + case nPropertyTypeSecondColor: + case nPropertyTypeCharColor: + case nPropertyTypeLineColor: + { + const sal_Int32 nIndex = (nPropertyTypeFirstColor == nType) ? 0 : 1; + return pEffect->getColor( nIndex ); + } + + case nPropertyTypeFont: + return pEffect->getProperty( AnimationNodeType::SET, OUString( RTL_CONSTASCII_USTRINGPARAM("CharFontName") ), VALUE_TO ); + + case nPropertyTypeCharHeight: + { + const OUString aAttributeName( RTL_CONSTASCII_USTRINGPARAM( "CharHeight" ) ); + Any aValue( pEffect->getProperty( AnimationNodeType::SET, aAttributeName, VALUE_TO ) ); + if( !aValue.hasValue() ) + aValue = pEffect->getProperty( AnimationNodeType::ANIMATE, aAttributeName, VALUE_TO ); + return aValue; + } + + case nPropertyTypeRotate: + return pEffect->getTransformationProperty( AnimationTransformType::ROTATE, VALUE_BY); + + case nPropertyTypeTransparency: + return pEffect->getProperty( AnimationNodeType::SET, OUString(RTL_CONSTASCII_USTRINGPARAM("Opacity")), VALUE_TO ); + + case nPropertyTypeScale: + return pEffect->getTransformationProperty( AnimationTransformType::SCALE, VALUE_BY ); + + case nPropertyTypeCharDecoration: + { + Sequence< Any > aValues(3); + aValues[0] = pEffect->getProperty( AnimationNodeType::SET, OUString(RTL_CONSTASCII_USTRINGPARAM("CharWeight")), VALUE_TO ); + aValues[1] = pEffect->getProperty( AnimationNodeType::SET, OUString(RTL_CONSTASCII_USTRINGPARAM("CharPosture")), VALUE_TO ); + aValues[2] = pEffect->getProperty( AnimationNodeType::SET, OUString(RTL_CONSTASCII_USTRINGPARAM("CharUnderline")), VALUE_TO ); + return makeAny( aValues ); + } + } + + Any aAny; + return aAny; +} + +bool CustomAnimationPane::setProperty1Value( sal_Int32 nType, CustomAnimationEffectPtr pEffect, const Any& rValue ) +{ + bool bEffectChanged = false; + switch( nType ) + { + case nPropertyTypeDirection: + case nPropertyTypeSpokes: + case nPropertyTypeZoom: + { + OUString aPresetSubType; + rValue >>= aPresetSubType; + if( aPresetSubType != pEffect->getPresetSubType() ) + { + getPresets().changePresetSubType( pEffect, aPresetSubType ); + bEffectChanged = true; + } + } + break; + + case nPropertyTypeFillColor: + case nPropertyTypeColor: + case nPropertyTypeFirstColor: + case nPropertyTypeSecondColor: + case nPropertyTypeCharColor: + case nPropertyTypeLineColor: + { + const sal_Int32 nIndex = (nPropertyTypeFirstColor == nType) ? 0 : 1; + Any aOldColor( pEffect->getColor( nIndex ) ); + if( aOldColor != rValue ) + { + pEffect->setColor( nIndex, rValue ); + bEffectChanged = true; + } + } + break; + + case nPropertyTypeFont: + bEffectChanged = pEffect->setProperty( AnimationNodeType::SET, OUString( RTL_CONSTASCII_USTRINGPARAM( "CharFontName" ) ), VALUE_TO, rValue ); + break; + + case nPropertyTypeCharHeight: + { + const OUString aAttributeName( RTL_CONSTASCII_USTRINGPARAM( "CharHeight" ) ); + bEffectChanged = pEffect->setProperty( AnimationNodeType::SET, aAttributeName, VALUE_TO, rValue ); + if( !bEffectChanged ) + bEffectChanged = pEffect->setProperty( AnimationNodeType::ANIMATE, aAttributeName, VALUE_TO, rValue ); + } + break; + case nPropertyTypeRotate: + bEffectChanged = pEffect->setTransformationProperty( AnimationTransformType::ROTATE, VALUE_BY , rValue ); + break; + + case nPropertyTypeTransparency: + bEffectChanged = pEffect->setProperty( AnimationNodeType::SET, OUString( RTL_CONSTASCII_USTRINGPARAM("Opacity") ), VALUE_TO, rValue ); + break; + + case nPropertyTypeScale: + bEffectChanged = pEffect->setTransformationProperty( AnimationTransformType::SCALE, VALUE_BY, rValue ); + break; + + case nPropertyTypeCharDecoration: + { + Sequence< Any > aValues(3); + rValue >>= aValues; + bEffectChanged = pEffect->setProperty( AnimationNodeType::SET, OUString(RTL_CONSTASCII_USTRINGPARAM("CharWeight")), VALUE_TO, aValues[0] ); + bEffectChanged |= pEffect->setProperty( AnimationNodeType::SET, OUString(RTL_CONSTASCII_USTRINGPARAM("CharPosture")), VALUE_TO, aValues[1] ); + bEffectChanged |= pEffect->setProperty( AnimationNodeType::SET, OUString(RTL_CONSTASCII_USTRINGPARAM("CharUnderline")), VALUE_TO, aValues[2] ); + } + break; + + } + + return bEffectChanged; +} + +static sal_Bool hasVisibleShape( const Reference< XShape >& xShape ) +{ + try + { + const OUString sShapeType( xShape->getShapeType() ); + + if( sShapeType.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("com.sun.star.presentation.TitleTextShape") ) || + sShapeType.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("com.sun.star.presentation.OutlinerShape") ) || + sShapeType.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("com.sun.star.presentation.SubtitleShape") ) || + sShapeType.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("com.sun.star.drawing.TextShape") ) ) + { + const OUString sFillStyle( RTL_CONSTASCII_USTRINGPARAM("FillStyle" ) ); + const OUString sLineStyle( RTL_CONSTASCII_USTRINGPARAM("LineStyle" ) ); + Reference< XPropertySet > xSet( xShape, UNO_QUERY_THROW ); + + FillStyle eFillStyle; + xSet->getPropertyValue( sFillStyle ) >>= eFillStyle; + + ::com::sun::star::drawing::LineStyle eLineStyle; + xSet->getPropertyValue( sLineStyle ) >>= eLineStyle; + + return eFillStyle != FillStyle_NONE || eLineStyle != ::com::sun::star::drawing::LineStyle_NONE; + } + } + catch( Exception& e ) + { + (void)e; + } + return sal_True; +} + +STLPropertySet* CustomAnimationPane::createSelectionSet() +{ + STLPropertySet* pSet = CustomAnimationDialog::createDefaultSet(); + + pSet->setPropertyValue( nHandleCurrentPage, makeAny( mxCurrentPage ) ); + + sal_Int32 nMaxParaDepth = 0; + + // get options from selected effects + EffectSequence::iterator aIter( maListSelection.begin() ); + const EffectSequence::iterator aEnd( maListSelection.end() ); + const CustomAnimationPresets& rPresets (getPresets()); + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + + EffectSequenceHelper* pEffectSequence = pEffect->getEffectSequence(); + if( !pEffectSequence ) + pEffectSequence = mpMainSequence.get(); + + if( pEffect->hasText() ) + { + sal_Int32 n = calcMaxParaDepth(pEffect->getTargetShape()); + if( n > nMaxParaDepth ) + nMaxParaDepth = n; + } + + addValue( pSet, nHandleHasAfterEffect, makeAny( pEffect->hasAfterEffect() ) ); + addValue( pSet, nHandleAfterEffectOnNextEffect, makeAny( pEffect->IsAfterEffectOnNext() ? sal_True : sal_False ) ); + addValue( pSet, nHandleDimColor, pEffect->getDimColor() ); + addValue( pSet, nHandleIterateType, makeAny( pEffect->getIterateType() ) ); + + // convert absolute time to percentage value + // This calculation is done in float to avoid some rounding artifacts. + float fIterateInterval = (float)pEffect->getIterateInterval(); + if( pEffect->getDuration() ) + fIterateInterval = (float)(fIterateInterval / pEffect->getDuration() ); + fIterateInterval *= 100.0; + addValue( pSet, nHandleIterateInterval, makeAny( (double)fIterateInterval ) ); + + addValue( pSet, nHandleBegin, makeAny( pEffect->getBegin() ) ); + addValue( pSet, nHandleDuration, makeAny( pEffect->getDuration() ) ); + addValue( pSet, nHandleStart, makeAny( pEffect->getNodeType() ) ); + addValue( pSet, nHandleRepeat, makeAny( pEffect->getRepeatCount() ) ); + addValue( pSet, nHandleEnd, pEffect->getEnd() ); + addValue( pSet, nHandleRewind, makeAny( pEffect->getFill() ) ); + + addValue( pSet, nHandlePresetId, makeAny( pEffect->getPresetId() ) ); + + addValue( pSet, nHandleHasText, makeAny( (sal_Bool)pEffect->hasText() ) ); + + addValue( pSet, nHandleHasVisibleShape, Any( hasVisibleShape( pEffect->getTargetShape() ) ) ); + + Any aSoundSource; + if( pEffect->getAudio().is() ) + { + aSoundSource = pEffect->getAudio()->getSource(); + addValue( pSet, nHandleSoundVolumne, makeAny( pEffect->getAudio()->getVolume() ) ); +// todo addValue( pSet, nHandleSoundEndAfterSlide, makeAny( pEffect->getAudio()->getEndAfterSlide() ) ); +// this is now stored at the XCommand parameter sequence + } + else if( pEffect->getCommand() == EffectCommands::STOPAUDIO ) + { + aSoundSource = makeAny( (sal_Bool)sal_True ); + } + addValue( pSet, nHandleSoundURL, aSoundSource ); + + sal_Int32 nGroupId = pEffect->getGroupId(); + CustomAnimationTextGroupPtr pTextGroup; + if( nGroupId != -1 ) + pTextGroup = pEffectSequence->findGroup( nGroupId ); + + addValue( pSet, nHandleTextGrouping, makeAny( pTextGroup.get() ? pTextGroup->getTextGrouping() : (sal_Int32)-1 ) ); + addValue( pSet, nHandleAnimateForm, makeAny( pTextGroup.get() ? (sal_Bool)pTextGroup->getAnimateForm() : sal_True ) ); + addValue( pSet, nHandleTextGroupingAuto, makeAny( pTextGroup.get() ? pTextGroup->getTextGroupingAuto() : (double)-1.0 ) ); + addValue( pSet, nHandleTextReverse, makeAny( pTextGroup.get() ? (sal_Bool)pTextGroup->getTextReverse() : sal_False ) ); + + if( pEffectSequence->getSequenceType() == EffectNodeType::INTERACTIVE_SEQUENCE ) + { + InteractiveSequence* pIS = static_cast< InteractiveSequence* >( pEffectSequence ); + addValue( pSet, nHandleTrigger, makeAny( pIS->getTriggerShape() ) ); + } + + // + + CustomAnimationPresetPtr pDescriptor = rPresets.getEffectDescriptor( pEffect->getPresetId() ); + if( pDescriptor.get() ) + { + sal_Int32 nType = nPropertyTypeNone; + + UStringList aProperties( pDescriptor->getProperties() ); + if( aProperties.size() >= 1 ) + nType = getPropertyType( aProperties.front() ); + + if( nType != nPropertyTypeNone ) + { + addValue( pSet, nHandleProperty1Type, makeAny( nType ) ); + addValue( pSet, nHandleProperty1Value, getProperty1Value( nType, pEffect ) ); + } + + if( pDescriptor->hasProperty( OUString( RTL_CONSTASCII_USTRINGPARAM( "Accelerate" ) ) ) ) + { + addValue( pSet, nHandleAccelerate, makeAny( pEffect->getAcceleration() ) ); + } + + if( pDescriptor->hasProperty( OUString( RTL_CONSTASCII_USTRINGPARAM( "Decelerate" ) ) ) ) + { + addValue( pSet, nHandleDecelerate, makeAny( pEffect->getDecelerate() ) ); + } + + if( pDescriptor->hasProperty( OUString( RTL_CONSTASCII_USTRINGPARAM( "AutoReverse" ) ) ) ) + { + addValue( pSet, nHandleAutoReverse, makeAny( pEffect->getAutoReverse() ) ); + } + } + } + + addValue( pSet, nHandleMaxParaDepth, makeAny( nMaxParaDepth ) ); + + return pSet; +} + +void CustomAnimationPane::changeSelection( STLPropertySet* pResultSet, STLPropertySet* pOldSet ) +{ + // change selected effect + bool bChanged = false; + + MainSequenceRebuildGuard aGuard( mpMainSequence ); + + EffectSequence::iterator aIter( maListSelection.begin() ); + const EffectSequence::iterator aEnd( maListSelection.end() ); + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + + DBG_ASSERT( pEffect->getEffectSequence(), "sd::CustomAnimationPane::changeSelection(), dead effect in selection!" ); + if( !pEffect->getEffectSequence() ) + continue; + + double fDuration = 0.0; // we might need this for iterate-interval + if( pResultSet->getPropertyState( nHandleDuration ) == STLPropertyState_DIRECT ) + { + pResultSet->getPropertyValue( nHandleDuration ) >>= fDuration; + } + else + { + fDuration = pEffect->getDuration(); + } + + if( pResultSet->getPropertyState( nHandleIterateType ) == STLPropertyState_DIRECT ) + { + sal_Int16 nIterateType = 0; + pResultSet->getPropertyValue( nHandleIterateType ) >>= nIterateType; + if( pEffect->getIterateType() != nIterateType ) + { + pEffect->setIterateType( nIterateType ); + bChanged = true; + } + } + + if( pEffect->getIterateType() ) + { + if( pResultSet->getPropertyState( nHandleIterateInterval ) == STLPropertyState_DIRECT ) + { + double fIterateInterval = 0.0; + pResultSet->getPropertyValue( nHandleIterateInterval ) >>= fIterateInterval; + if( pEffect->getIterateInterval() != fIterateInterval ) + { + const double f = fIterateInterval * pEffect->getDuration() / 100; + pEffect->setIterateInterval( f ); + bChanged = true; + } + } + } + + if( pResultSet->getPropertyState( nHandleBegin ) == STLPropertyState_DIRECT ) + { + double fBegin = 0.0; + pResultSet->getPropertyValue( nHandleBegin ) >>= fBegin; + if( pEffect->getBegin() != fBegin ) + { + pEffect->setBegin( fBegin ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleDuration ) == STLPropertyState_DIRECT ) + { + if( pEffect->getDuration() != fDuration ) + { + pEffect->setDuration( fDuration ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleStart ) == STLPropertyState_DIRECT ) + { + sal_Int16 nNodeType = 0; + pResultSet->getPropertyValue( nHandleStart ) >>= nNodeType; + if( pEffect->getNodeType() != nNodeType ) + { + pEffect->setNodeType( nNodeType ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleRepeat ) == STLPropertyState_DIRECT ) + { + Any aRepeatCount( pResultSet->getPropertyValue( nHandleRepeat ) ); + if( aRepeatCount != pEffect->getRepeatCount() ) + { + pEffect->setRepeatCount( aRepeatCount ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleEnd ) == STLPropertyState_DIRECT ) + { + Any aEndValue( pResultSet->getPropertyValue( nHandleEnd ) ); + if( pEffect->getEnd() != aEndValue ) + { + pEffect->setEnd( aEndValue ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleRewind ) == STLPropertyState_DIRECT ) + { + sal_Int16 nFill = 0; + pResultSet->getPropertyValue( nHandleRewind ) >>= nFill; + if( pEffect->getFill() != nFill ) + { + pEffect->setFill( nFill ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleHasAfterEffect ) == STLPropertyState_DIRECT ) + { + sal_Bool bHasAfterEffect = sal_False; + if( pResultSet->getPropertyValue( nHandleHasAfterEffect ) >>= bHasAfterEffect ) + { + if( pEffect->hasAfterEffect() != bHasAfterEffect ) + { + pEffect->setHasAfterEffect( bHasAfterEffect ); + bChanged = true; + } + } + } + + if( pResultSet->getPropertyState( nHandleAfterEffectOnNextEffect ) == STLPropertyState_DIRECT ) + { + sal_Bool bAfterEffectOnNextEffect = sal_False; + if( (pResultSet->getPropertyValue( nHandleAfterEffectOnNextEffect ) >>= bAfterEffectOnNextEffect) && ((pEffect->IsAfterEffectOnNext() ? sal_True : sal_False) != bAfterEffectOnNextEffect) ) + { + pEffect->setAfterEffectOnNext( bAfterEffectOnNextEffect ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleDimColor ) == STLPropertyState_DIRECT ) + { + Any aDimColor( pResultSet->getPropertyValue( nHandleDimColor ) ); + if( pEffect->getDimColor() != aDimColor ) + { + pEffect->setDimColor( aDimColor ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleAccelerate ) == STLPropertyState_DIRECT ) + { + double fAccelerate = 0.0; + pResultSet->getPropertyValue( nHandleAccelerate ) >>= fAccelerate; + if( pEffect->getAcceleration() != fAccelerate ) + { + pEffect->setAcceleration( fAccelerate ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleDecelerate ) == STLPropertyState_DIRECT ) + { + double fDecelerate = 0.0; + pResultSet->getPropertyValue( nHandleDecelerate ) >>= fDecelerate; + if( pEffect->getDecelerate() != fDecelerate ) + { + pEffect->setDecelerate( fDecelerate ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleAutoReverse ) == STLPropertyState_DIRECT ) + { + sal_Bool bAutoReverse = sal_False; + pResultSet->getPropertyValue( nHandleAutoReverse ) >>= bAutoReverse; + if( pEffect->getAutoReverse() != bAutoReverse ) + { + pEffect->setAutoReverse( bAutoReverse ); + bChanged = true; + } + } + + if( pResultSet->getPropertyState( nHandleProperty1Value ) == STLPropertyState_DIRECT ) + { + sal_Int32 nType = 0; + pOldSet->getPropertyValue( nHandleProperty1Type ) >>= nType; + + bChanged |= setProperty1Value( nType, pEffect, pResultSet->getPropertyValue( nHandleProperty1Value ) ); + } + + if( pResultSet->getPropertyState( nHandleSoundURL ) == STLPropertyState_DIRECT ) + { + const Any aSoundSource( pResultSet->getPropertyValue( nHandleSoundURL ) ); + + if( aSoundSource.getValueType() == ::getCppuType((const sal_Bool*)0) ) + { + pEffect->setStopAudio(); + bChanged = true; + } + else + { + OUString aSoundURL; + aSoundSource >>= aSoundURL; + + if( aSoundURL.getLength() ) + { + if( !pEffect->getAudio().is() ) + { + pEffect->createAudio( aSoundSource ); + bChanged = true; + } + else + { + if( pEffect->getAudio()->getSource() != aSoundSource ) + { + pEffect->getAudio()->setSource( aSoundSource ); + bChanged = true; + } + } + } + else + { + if( pEffect->getAudio().is() || pEffect->getStopAudio() ) + { + pEffect->removeAudio(); + bChanged = true; + } + } + } + } + + if( pResultSet->getPropertyState( nHandleTrigger ) == STLPropertyState_DIRECT ) + { + Reference< XShape > xTriggerShape; + pResultSet->getPropertyValue( nHandleTrigger ) >>= xTriggerShape; + bChanged |= mpMainSequence->setTrigger( pEffect, xTriggerShape ); + } + } + + const bool bHasTextGrouping = pResultSet->getPropertyState( nHandleTextGrouping ) == STLPropertyState_DIRECT; + const bool bHasAnimateForm = pResultSet->getPropertyState( nHandleAnimateForm ) == STLPropertyState_DIRECT; + const bool bHasTextGroupingAuto = pResultSet->getPropertyState( nHandleTextGroupingAuto ) == STLPropertyState_DIRECT; + const bool bHasTextReverse = pResultSet->getPropertyState( nHandleTextReverse ) == STLPropertyState_DIRECT; + + if( bHasTextGrouping || bHasAnimateForm || bHasTextGroupingAuto || bHasTextReverse ) + { + // we need to do a second pass for text grouping options + // since changing them can cause effects to be removed + // or replaced, we do this after we aplied all other options + // above + + sal_Int32 nTextGrouping = 0; + sal_Bool bAnimateForm = sal_True, bTextReverse = sal_False; + double fTextGroupingAuto = -1.0; + + if( bHasTextGrouping ) + pResultSet->getPropertyValue(nHandleTextGrouping) >>= nTextGrouping; + + if( bHasAnimateForm ) + pResultSet->getPropertyValue(nHandleAnimateForm) >>= bAnimateForm; + + if( bHasTextGroupingAuto ) + pResultSet->getPropertyValue(nHandleTextGroupingAuto) >>= fTextGroupingAuto; + + if( bHasTextReverse ) + pResultSet->getPropertyValue(nHandleTextReverse) >>= bTextReverse; + + EffectSequence const aSelectedEffects( maListSelection ); + EffectSequence::const_iterator iter( aSelectedEffects.begin() ); + const EffectSequence::const_iterator iEnd( aSelectedEffects.end() ); + while( iter != iEnd ) + { + CustomAnimationEffectPtr const& pEffect = (*iter++); + + EffectSequenceHelper* pEffectSequence = pEffect->getEffectSequence(); + if( !pEffectSequence ) + pEffectSequence = mpMainSequence.get(); + + sal_Int32 nGroupId = pEffect->getGroupId(); + CustomAnimationTextGroupPtr pTextGroup; + if( (nGroupId != -1) ) + { + // use existing group + pTextGroup = pEffectSequence->findGroup( nGroupId ); + } + else + { + // somethings changed so we need a group now + pTextGroup = pEffectSequence->createTextGroup( pEffect, nTextGrouping, fTextGroupingAuto, bAnimateForm, bTextReverse ); + bChanged = true; + } + + if( bHasTextGrouping ) + { + if( (pTextGroup->getTextGrouping() != nTextGrouping) ) + { + pEffectSequence->setTextGrouping( pTextGroup, nTextGrouping ); + bChanged = true; + } + } + + if( bHasAnimateForm ) + { + if( pTextGroup->getAnimateForm() != bAnimateForm ) + { + pEffectSequence->setAnimateForm( pTextGroup, bAnimateForm ); + bChanged = true; + } + } + + if( bHasTextGroupingAuto ) + { + if( pTextGroup->getTextGroupingAuto() != fTextGroupingAuto ) + { + pEffectSequence->setTextGroupingAuto( pTextGroup, fTextGroupingAuto ); + bChanged = true; + } + } + + if( bHasTextReverse ) + { + if( pTextGroup->getTextReverse() != bTextReverse ) + { + pEffectSequence->setTextReverse( pTextGroup, bTextReverse ); + bChanged = true; + } + } + } + } + + if( bChanged ) + { + mpMainSequence->rebuild(); + updateControls(); + mrBase.GetDocShell()->SetModified(); + } +} + +void CustomAnimationPane::showOptions( USHORT nPage /* = 0 */ ) +{ + STLPropertySet* pSet = createSelectionSet(); + + CustomAnimationDialog* pDlg = new CustomAnimationDialog( this, pSet, nPage ); + if( pDlg->Execute() ) + { + addUndo(); + changeSelection( pDlg->getResultSet(), pSet ); + updateControls(); + } + + delete pDlg; +} + +void CustomAnimationPane::onChangeCurrentPage() +{ + if( mxView.is() ) try + { + Reference< XDrawPage > xNewPage( mxView->getCurrentPage() ); + if( xNewPage != mxCurrentPage ) + { + mxCurrentPage = xNewPage; + SdPage* pPage = SdPage::getImplementation( mxCurrentPage ); + if( pPage ) + { + mpMainSequence = pPage->getMainSequence(); + mpCustomAnimationList->update( mpMainSequence ); + } + updateControls(); + } + } + catch( Exception& ) + { + DBG_ERROR( "sd::CustomAnimationPane::onChangeCurrentPage(), exception catched!" ); + } +} + +bool getTextSelection( const Any& rSelection, Reference< XShape >& xShape, std::list< sal_Int16 >& rParaList ) +{ + Reference< XTextRange > xSelectedText; + rSelection >>= xSelectedText; + if( xSelectedText.is() ) try + { + xShape.set( xSelectedText->getText(), UNO_QUERY_THROW ); + + Reference< XTextRangeCompare > xTextRangeCompare( xShape, UNO_QUERY_THROW ); + Reference< XEnumerationAccess > xParaEnumAccess( xShape, UNO_QUERY_THROW ); + Reference< XEnumeration > xParaEnum( xParaEnumAccess->createEnumeration(), UNO_QUERY_THROW ); + Reference< XTextRange > xRange; + Reference< XTextRange > xStart( xSelectedText->getStart() ); + Reference< XTextRange > xEnd( xSelectedText->getEnd() ); + + if( xTextRangeCompare->compareRegionEnds( xStart, xEnd ) < 0 ) + { + Reference< XTextRange > xTemp( xStart ); + xStart = xEnd; + xEnd = xTemp; + } + + sal_Int16 nPara = 0; + while( xParaEnum->hasMoreElements() ) + { + xParaEnum->nextElement() >>= xRange; + + // break if start of selection is prior to end of current paragraph + if( xRange.is() && (xTextRangeCompare->compareRegionEnds( xStart, xRange ) >= 0 ) ) + break; + + nPara++; + } + + while( xRange.is() ) + { + if( xRange.is() && xRange->getString().getLength() ) + rParaList.push_back( nPara ); + + // break if end of selection is before or at end of current paragraph + if( xRange.is() && xTextRangeCompare->compareRegionEnds( xEnd, xRange ) >= 0 ) + break; + + nPara++; + + if( xParaEnum->hasMoreElements() ) + xParaEnum->nextElement() >>= xRange; + else + xRange.clear(); + } + + return true; + } + catch( Exception& e ) + { + (void)e; + DBG_ERROR( "sd::CustomAnimationPane::getTextSelection(), exception cought!" ); + } + + return false; +} + +void CustomAnimationPane::onChange( bool bCreate ) +{ + bool bHasText = true; + + // first create vector of targets for dialog preview + std::vector< Any > aTargets; + OUString sPresetId; + double fDuration = 2.0f; + + if( bCreate ) + { + // gather shapes from the selection + Reference< XSelectionSupplier > xSel( mxView, UNO_QUERY_THROW ); + maViewSelection = xSel->getSelection(); + + if( maViewSelection.getValueType() == ::getCppuType((const Reference< XShapes >*)0) ) + { + Reference< XIndexAccess > xShapes; + maViewSelection >>= xShapes; + + sal_Int32 nCount = xShapes->getCount(); + sal_Int32 nIndex; + for( nIndex = 0; nIndex < nCount; nIndex++ ) + { + Any aTarget( xShapes->getByIndex( nIndex ) ); + aTargets.push_back( aTarget ); + if( bHasText ) + { + Reference< XText > xText; + aTarget >>= xText; + if( !xText.is() || xText->getString().getLength() == 0 ) + bHasText = false; + } + } + } + else if ( maViewSelection.getValueType() == ::getCppuType((const Reference< XShape >*)0) ) + { + aTargets.push_back( maViewSelection ); + Reference< XText > xText; + maViewSelection >>= xText; + if( !xText.is() || xText->getString().getLength() == 0 ) + bHasText = false; + } + else if ( maViewSelection.getValueType() == ::getCppuType((const Reference< XTextCursor >*)0) ) + { + Reference< XShape > xShape; + std::list< sal_Int16 > aParaList; + if( getTextSelection( maViewSelection, xShape, aParaList ) ) + { + ParagraphTarget aParaTarget; + aParaTarget.Shape = xShape; + + std::list< sal_Int16 >::iterator aIter( aParaList.begin() ); + for( ; aIter != aParaList.end(); aIter++ ) + { + aParaTarget.Paragraph = (*aIter); + aTargets.push_back( makeAny( aParaTarget ) ); + } + } + } + else + { + DBG_ERROR("sd::CustomAnimationPane::onChange(), unknown view selection!" ); + return; + } + } + else + { + // get selected effect + EffectSequence::iterator aIter( maListSelection.begin() ); + const EffectSequence::iterator aEnd( maListSelection.end() ); + while( aIter != aEnd ) + { + if( !bHasText || !(*aIter)->hasText() ) + bHasText = false; + + if( sPresetId.getLength() == 0 ) + { + sPresetId = (*aIter)->getPresetId(); + fDuration = (*aIter)->getDuration(); + } + + aTargets.push_back( (*aIter++)->getTarget() ); + } + } + + CustomAnimationCreateDialog* pDlg = new CustomAnimationCreateDialog( this, this, aTargets, bHasText, sPresetId, fDuration ); + if( pDlg->Execute() ) + { + addUndo(); + fDuration = pDlg->getSelectedDuration(); + CustomAnimationPresetPtr pDescriptor = pDlg->getSelectedPreset(); + if( pDescriptor.get() ) + { + if( bCreate ) + { + mpCustomAnimationList->SelectAll( FALSE ); + + // gather shapes from the selection + std::vector< Any >::iterator aIter( aTargets.begin() ); + const std::vector< Any >::iterator aEnd( aTargets.end() ); + bool bFirst = true; + for( ; aIter != aEnd; aIter++ ) + { + CustomAnimationEffectPtr pCreated = mpMainSequence->append( pDescriptor, (*aIter), fDuration ); + + if( bFirst ) + bFirst = false; + else + pCreated->setNodeType( EffectNodeType::WITH_PREVIOUS ); + + if( pCreated.get() ) + { + mpCustomAnimationList->select( pCreated ); + } + } + } + else + { + MainSequenceRebuildGuard aGuard( mpMainSequence ); + + // get selected effect + EffectSequence::iterator aIter( maListSelection.begin() ); + const EffectSequence::iterator aEnd( maListSelection.end() ); + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + + EffectSequenceHelper* pEffectSequence = pEffect->getEffectSequence(); + if( !pEffectSequence ) + pEffectSequence = mpMainSequence.get(); + + pEffectSequence->replace( pEffect, pDescriptor, fDuration ); + } + } + } + else + { + PathKind eKind = pDlg->getCreatePathKind(); + if( eKind != NONE ) + createPath( eKind, aTargets, fDuration ); + } + mrBase.GetDocShell()->SetModified(); + } + + delete pDlg; + + updateControls(); + + // stop running preview from dialog + SlideShow::Stop( mrBase ); +} + +void CustomAnimationPane::createPath( PathKind eKind, std::vector< Any >& rTargets, double fDuration) +{ + USHORT nSID = 0; + + switch( eKind ) + { + case CURVE: nSID = SID_DRAW_BEZIER_NOFILL; break; + case POLYGON: nSID = SID_DRAW_POLYGON_NOFILL; break; + case FREEFORM: nSID = SID_DRAW_FREELINE_NOFILL; break; + default: break; + } + + if( nSID ) + { + DrawViewShell* pViewShell = dynamic_cast< DrawViewShell* >( + FrameworkHelper::Instance(mrBase)->GetViewShell(FrameworkHelper::msCenterPaneURL).get()); + + if( pViewShell ) + { + DrawView* pView = pViewShell->GetDrawView(); + if( pView ) + pView->UnmarkAllObj(); + + std::vector< Any > aTargets( 1, Any( fDuration ) ); + aTargets.insert( aTargets.end(), rTargets.begin(), rTargets.end() ); + Sequence< Any > aTargetSequence( comphelper::containerToSequence( aTargets ) ); + const SfxUnoAnyItem aItem( SID_ADD_MOTION_PATH, Any( aTargetSequence ) ); + pViewShell->GetViewFrame()->GetDispatcher()->Execute( nSID, SFX_CALLMODE_ASYNCHRON, &aItem, 0 ); + } + } +} + +void CustomAnimationPane::onRemove() +{ + if( maListSelection.size() ) + { + addUndo(); + + MainSequenceRebuildGuard aGuard( mpMainSequence ); + + EffectSequence aList( maListSelection ); + + EffectSequence::iterator aIter( aList.begin() ); + const EffectSequence::iterator aEnd( aList.end() ); + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + if( pEffect->getEffectSequence() ) + pEffect->getEffectSequence()->remove( pEffect ); + } + + maListSelection.clear(); + mrBase.GetDocShell()->SetModified(); + } +} + +void CustomAnimationPane::remove( CustomAnimationEffectPtr& pEffect ) +{ + if( pEffect->getEffectSequence() ) + { + addUndo(); + pEffect->getEffectSequence()->remove( pEffect ); + mrBase.GetDocShell()->SetModified(); + } +} + +void CustomAnimationPane::onChangeStart() +{ + if( mpLBStart->GetSelectEntryCount() == 1 ) + { + sal_Int16 nNodeType; + USHORT nPos= mpLBStart->GetSelectEntryPos(); + switch( nPos ) + { + case 0: nNodeType = EffectNodeType::ON_CLICK; break; + case 1: nNodeType = EffectNodeType::WITH_PREVIOUS; break; + case 2: nNodeType = EffectNodeType::AFTER_PREVIOUS; break; + default: + return; + } + + onChangeStart( nNodeType ); + } +} + +void CustomAnimationPane::onChangeStart( sal_Int16 nNodeType ) +{ + addUndo(); + + MainSequenceRebuildGuard aGuard( mpMainSequence ); + + bool bNeedRebuild = false; + + EffectSequence::iterator aIter( maListSelection.begin() ); + const EffectSequence::iterator aEnd( maListSelection.end() ); + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + if( pEffect->getNodeType() != nNodeType ) + { + pEffect->setNodeType( nNodeType ); + bNeedRebuild = true; + } + } + + if( bNeedRebuild ) + { + mpMainSequence->rebuild(); + updateControls(); + mrBase.GetDocShell()->SetModified(); + } +} + +void CustomAnimationPane::onChangeProperty() +{ + if( mpLBProperty->getSubControl() ) + { + addUndo(); + + MainSequenceRebuildGuard aGuard( mpMainSequence ); + + const Any aValue( mpLBProperty->getSubControl()->getValue() ); + + bool bNeedUpdate = false; + + // change selected effect + EffectSequence::iterator aIter( maListSelection.begin() ); + const EffectSequence::iterator aEnd( maListSelection.end() ); + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + + if( setProperty1Value( mnPropertyType, pEffect, aValue ) ) + bNeedUpdate = true; + } + + if( bNeedUpdate ) + { + mpMainSequence->rebuild(); + updateControls(); + mrBase.GetDocShell()->SetModified(); + } + + onPreview( false ); + } +} + +void CustomAnimationPane::onChangeSpeed() +{ + if( mpCBSpeed->GetSelectEntryCount() == 1 ) + { + addUndo(); + + MainSequenceRebuildGuard aGuard( mpMainSequence ); + + double fDuration; + + USHORT nPos= mpCBSpeed->GetSelectEntryPos(); + + switch( nPos ) + { + case 0: fDuration = 5.0; break; + case 1: fDuration = 3.0; break; + case 2: fDuration = 2.0; break; + case 3: fDuration = 1.0; break; + case 4: fDuration = 0.5; break; + default: + return; + } + + // change selected effect + EffectSequence::iterator aIter( maListSelection.begin() ); + const EffectSequence::iterator aEnd( maListSelection.end() ); + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + pEffect->setDuration( fDuration ); + } + + mpMainSequence->rebuild(); + updateControls(); + mrBase.GetDocShell()->SetModified(); + + onPreview( false ); + } +} + +/// this link is called when the property box is modified by the user +IMPL_LINK( CustomAnimationPane, implPropertyHdl, Control*, EMPTYARG ) +{ + onChangeProperty(); + return 0; +} + +/// this link is called when one of the controls is modified +IMPL_LINK( CustomAnimationPane, implControlHdl, Control*, pControl ) +{ + if( pControl == mpPBAddEffect ) + onChange(true); + else if( pControl == mpPBChangeEffect ) + onChange(false); + else if( pControl == mpPBRemoveEffect ) + onRemove(); + else if( pControl == mpLBStart ) + onChangeStart(); + else if( pControl == mpCBSpeed ) + onChangeSpeed(); + else if( pControl == mpPBPropertyMore ) + showOptions(); + else if( pControl == mpPBMoveUp ) + moveSelection( true ); + else if( pControl == mpPBMoveDown ) + moveSelection( false ); + else if( pControl == mpPBPlay ) + onPreview( true ); + else if( pControl == mpPBSlideShow ) + { + mrBase.StartPresentation(); + } + else if( pControl == mpCBAutoPreview ) + { + SdOptions* pOptions = SD_MOD()->GetSdOptions(DOCUMENT_TYPE_IMPRESS); + pOptions->SetPreviewChangedEffects( mpCBAutoPreview->IsChecked() ? sal_True : sal_False ); + } + + updateControls(); + + return 0; +} + +IMPL_LINK(CustomAnimationPane, lateInitCallback, Timer*, EMPTYARG ) +{ + // Call getPresets() to initiate the (expensive) construction of the + // presets list. + getPresets(); + + // update selection and control states + onSelectionChanged(); + + return 0; +} + +void CustomAnimationPane::moveSelection( bool bUp ) +{ + if( maListSelection.empty() ) + return; + + EffectSequenceHelper* pSequence = maListSelection.front()->getEffectSequence(); + if( pSequence == 0 ) + return; + + addUndo(); + + bool bChanged = false; + + MainSequenceRebuildGuard aGuard( mpMainSequence ); + EffectSequence& rEffectSequence = pSequence->getSequence(); + + if( bUp ) + { + EffectSequence::iterator aIter( maListSelection.begin() ); + const EffectSequence::iterator aEnd( maListSelection.end() ); + + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + + EffectSequence::iterator aEffectPos( pSequence->find( pEffect ) ); + if( aEffectPos != rEffectSequence.end() ) + { + EffectSequence::iterator aInsertPos( rEffectSequence.erase( aEffectPos ) ); + + if( aInsertPos != rEffectSequence.begin() ) + { + aInsertPos--; + while( (aInsertPos != rEffectSequence.begin()) && !mpCustomAnimationList->isExpanded(*aInsertPos)) + aInsertPos--; + + rEffectSequence.insert( aInsertPos, pEffect ); + } + else + { + rEffectSequence.push_front( pEffect ); + } + bChanged = true; + } + } + } + else + { + EffectSequence::reverse_iterator aIter( maListSelection.rbegin() ); + const EffectSequence::reverse_iterator aEnd( maListSelection.rend() ); + + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + + EffectSequence::iterator aEffectPos( pSequence->find( pEffect ) ); + if( aEffectPos != rEffectSequence.end() ) + { + EffectSequence::iterator aInsertPos( rEffectSequence.erase( aEffectPos ) ); + + if( aInsertPos != rEffectSequence.end() ) + { + aInsertPos++; + while( (aInsertPos != rEffectSequence.end()) && !mpCustomAnimationList->isExpanded(*aInsertPos)) + aInsertPos++; + + rEffectSequence.insert( aInsertPos, pEffect ); + } + else + { + rEffectSequence.push_back( pEffect ); + } + bChanged = true; + } + } + } + + if( bChanged ) + { + mpMainSequence->rebuild(); + updateControls(); + mrBase.GetDocShell()->SetModified(); + } +} + +void CustomAnimationPane::onPreview( bool bForcePreview ) +{ + if( !bForcePreview && !mpCBAutoPreview->IsChecked() ) + return; + + if( maListSelection.empty() ) + { + rtl::Reference< MotionPathTag > xMotionPathTag; + MotionPathTagVector::iterator aIter; + for( aIter = maMotionPathTags.begin(); aIter != maMotionPathTags.end(); aIter++ ) + { + if( (*aIter)->isSelected() ) + { + xMotionPathTag = (*aIter); + break; + } + } + + if( xMotionPathTag.is() ) + { + MainSequencePtr pSequence( new MainSequence() ); + pSequence->append( xMotionPathTag->getEffect()->clone() ); + preview( pSequence->getRootNode() ); + } + else + { + Reference< XAnimationNodeSupplier > xNodeSupplier( mxCurrentPage, UNO_QUERY ); + if( !xNodeSupplier.is() ) + return; + + preview( xNodeSupplier->getAnimationNode() ); + } + } + else + { + MainSequencePtr pSequence( new MainSequence() ); + + EffectSequence::iterator aIter( maListSelection.begin() ); + const EffectSequence::iterator aEnd( maListSelection.end() ); + + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + pSequence->append( pEffect->clone() ); + } + + preview( pSequence->getRootNode() ); + } +} + +void CustomAnimationPane::preview( const Reference< XAnimationNode >& xAnimationNode ) +{ + Reference< XTimeContainer > xRoot(::comphelper::getProcessServiceFactory()->createInstance(OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.animations.ParallelTimeContainer"))), UNO_QUERY); + if( xRoot.is() ) + { + Sequence< ::com::sun::star::beans::NamedValue > aUserData( 1 ); + aUserData[0].Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "node-type" ) ); + aUserData[0].Value <<= ::com::sun::star::presentation::EffectNodeType::TIMING_ROOT; + xRoot->setUserData( aUserData ); + xRoot->appendChild( xAnimationNode ); + + Reference< XAnimationNode > xNode( xRoot, UNO_QUERY ); + SlideShow::StartPreview( mrBase, mxCurrentPage, xNode ); + } +} + + +// ICustomAnimationListController +void CustomAnimationPane::onSelect() +{ + maListSelection = mpCustomAnimationList->getSelection(); + updateControls(); + markShapesFromSelectedEffects(); +} + + + + +const CustomAnimationPresets& CustomAnimationPane::getPresets (void) +{ + if (mpCustomAnimationPresets == NULL) + mpCustomAnimationPresets = &CustomAnimationPresets::getCustomAnimationPresets(); + return *mpCustomAnimationPresets; +} + + + +void CustomAnimationPane::markShapesFromSelectedEffects() +{ + if( !maSelectionLock.isLocked() ) + { + ScopeLockGuard aGuard( maSelectionLock ); + DrawViewShell* pViewShell = dynamic_cast< DrawViewShell* >( + FrameworkHelper::Instance(mrBase)->GetViewShell(FrameworkHelper::msCenterPaneURL).get()); + DrawView* pView = pViewShell ? pViewShell->GetDrawView() : NULL; + + if( pView ) + { + pView->UnmarkAllObj(); + EffectSequence::iterator aIter( maListSelection.begin() ); + const EffectSequence::iterator aEnd( maListSelection.end() ); + while( aIter != aEnd ) + { + CustomAnimationEffectPtr pEffect = (*aIter++); + + Reference< XShape > xShape( pEffect->getTargetShape() ); + SdrObject* pObj = GetSdrObjectFromXShape( xShape ); + if( pObj ) + pView->MarkObj(pObj, pView->GetSdrPageView(), FALSE, FALSE); + } + } + } +} + + +void CustomAnimationPane::updatePathFromMotionPathTag( const rtl::Reference< MotionPathTag >& xTag ) +{ + MainSequenceRebuildGuard aGuard( mpMainSequence ); + if( xTag.is() ) + { + SdrPathObj* pPathObj = xTag->getPathObj(); + CustomAnimationEffectPtr pEffect = xTag->getEffect(); + if( (pPathObj != 0) && pEffect.get() != 0 ) + { + SfxUndoManager* pManager = mrBase.GetDocShell()->GetUndoManager(); + if( pManager ) + { + SdPage* pPage = SdPage::getImplementation( mxCurrentPage ); + if( pPage ) + pManager->AddUndoAction( new UndoAnimationPath( mrBase.GetDocShell()->GetDoc(), pPage, pEffect->getNode() ) ); + } + + pEffect->updatePathFromSdrPathObj( *pPathObj ); + } + } +} + +// ==================================================================== + +::Window * createCustomAnimationPanel( ::Window* pParent, ViewShellBase& rBase ) +{ + DialogListBox* pWindow = 0; + + DrawDocShell* pDocSh = rBase.GetDocShell(); + if( pDocSh ) + { + pWindow = new DialogListBox( pParent, WB_CLIPCHILDREN|WB_TABSTOP|WB_AUTOHSCROLL ); + + Size aMinSize( pWindow->LogicToPixel( Size( 80, 256 ), MAP_APPFONT ) ); + ::Window* pPaneWindow = new CustomAnimationPane( pWindow, rBase, aMinSize ); + pWindow->SetChildWindow( pPaneWindow, aMinSize ); + pWindow->SetText( pPaneWindow->GetText() ); + } + + return pWindow; +} + + + +} |