LCOV - code coverage report
Current view: top level - sd/source/core - CustomAnimationEffect.cxx (source / functions) Hit Total Coverage
Test: commit c8344322a7af75b84dd3ca8f78b05543a976dfd5 Lines: 78 1800 4.3 %
Date: 2015-06-13 12:38:46 Functions: 19 140 13.6 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include <tools/debug.hxx>
      21             : #include <com/sun/star/animations/AnimationNodeType.hpp>
      22             : #include <com/sun/star/animations/AnimateColor.hpp>
      23             : #include <com/sun/star/animations/AnimateMotion.hpp>
      24             : #include <com/sun/star/animations/AnimateSet.hpp>
      25             : #include <com/sun/star/animations/AnimationFill.hpp>
      26             : #include <com/sun/star/animations/AnimationTransformType.hpp>
      27             : #include <com/sun/star/animations/Audio.hpp>
      28             : #include <com/sun/star/animations/Command.hpp>
      29             : #include <com/sun/star/animations/Event.hpp>
      30             : #include <com/sun/star/animations/EventTrigger.hpp>
      31             : #include <com/sun/star/animations/IterateContainer.hpp>
      32             : #include <com/sun/star/animations/ParallelTimeContainer.hpp>
      33             : #include <com/sun/star/animations/SequenceTimeContainer.hpp>
      34             : #include <com/sun/star/animations/Timing.hpp>
      35             : #include <com/sun/star/animations/XCommand.hpp>
      36             : #include <com/sun/star/animations/XIterateContainer.hpp>
      37             : #include <com/sun/star/animations/XAnimateTransform.hpp>
      38             : #include <com/sun/star/animations/XAnimateMotion.hpp>
      39             : #include <com/sun/star/animations/XAnimate.hpp>
      40             : #include <com/sun/star/beans/NamedValue.hpp>
      41             : #include <com/sun/star/beans/XPropertySet.hpp>
      42             : #include <com/sun/star/container/XEnumerationAccess.hpp>
      43             : #include <com/sun/star/drawing/XDrawPage.hpp>
      44             : #include <com/sun/star/lang/XInitialization.hpp>
      45             : #include <com/sun/star/presentation/EffectNodeType.hpp>
      46             : #include <com/sun/star/presentation/EffectCommands.hpp>
      47             : #include <com/sun/star/presentation/EffectPresetClass.hpp>
      48             : #include <com/sun/star/presentation/ParagraphTarget.hpp>
      49             : #include <com/sun/star/presentation/ShapeAnimationSubType.hpp>
      50             : #include <com/sun/star/text/XText.hpp>
      51             : #include <com/sun/star/util/XCloneable.hpp>
      52             : #include <com/sun/star/util/XChangesNotifier.hpp>
      53             : #include <comphelper/processfactory.hxx>
      54             : #include <comphelper/sequence.hxx>
      55             : #include <com/sun/star/lang/Locale.hpp>
      56             : #include <com/sun/star/i18n/BreakIterator.hpp>
      57             : #include <com/sun/star/i18n/CharacterIteratorMode.hpp>
      58             : #include <com/sun/star/i18n/WordType.hpp>
      59             : #include <com/sun/star/presentation/TextAnimationType.hpp>
      60             : 
      61             : #include <basegfx/polygon/b2dpolypolygon.hxx>
      62             : #include <basegfx/polygon/b2dpolypolygontools.hxx>
      63             : #include <basegfx/matrix/b2dhommatrix.hxx>
      64             : #include <basegfx/range/b2drange.hxx>
      65             : #include <basegfx/matrix/b2dhommatrixtools.hxx>
      66             : 
      67             : #include <algorithm>
      68             : 
      69             : #include <cppuhelper/implbase1.hxx>
      70             : 
      71             : #include <drawinglayer/geometry/viewinformation2d.hxx>
      72             : #include <svx/sdr/contact/viewcontact.hxx>
      73             : #include <svx/svdopath.hxx>
      74             : #include <svx/svdpage.hxx>
      75             : #include <svx/unoapi.hxx>
      76             : #include "CustomAnimationEffect.hxx"
      77             : #include <CustomAnimationPreset.hxx>
      78             : #include "animations.hxx"
      79             : 
      80             : using namespace ::com::sun::star;
      81             : using namespace ::com::sun::star::uno;
      82             : using namespace ::com::sun::star::presentation;
      83             : using namespace ::com::sun::star::animations;
      84             : 
      85             : using ::com::sun::star::container::XEnumerationAccess;
      86             : using ::com::sun::star::container::XEnumeration;
      87             : using ::com::sun::star::beans::NamedValue;
      88             : using ::com::sun::star::container::XChild;
      89             : using ::com::sun::star::container::XElementAccess;
      90             : using ::com::sun::star::drawing::XShape;
      91             : using ::com::sun::star::lang::XInitialization;
      92             : using ::com::sun::star::drawing::XShapes;
      93             : using ::com::sun::star::drawing::XDrawPage;
      94             : using ::com::sun::star::text::XText;
      95             : using ::com::sun::star::text::XTextRange;
      96             : using ::com::sun::star::beans::XPropertySet;
      97             : using ::com::sun::star::lang::XMultiServiceFactory;
      98             : using ::com::sun::star::util::XCloneable;
      99             : using ::com::sun::star::lang::Locale;
     100             : using ::com::sun::star::util::XChangesNotifier;
     101             : using ::com::sun::star::util::XChangesListener;
     102             : 
     103             : namespace sd
     104             : {
     105             : class MainSequenceChangeGuard
     106             : {
     107             : public:
     108           0 :     MainSequenceChangeGuard( EffectSequenceHelper* pSequence )
     109             :     {
     110           0 :         mpMainSequence = dynamic_cast< MainSequence* >( pSequence );
     111           0 :         if( mpMainSequence == 0 )
     112             :         {
     113           0 :             InteractiveSequence* pI = dynamic_cast< InteractiveSequence* >( pSequence );
     114           0 :             if( pI )
     115           0 :                 mpMainSequence = pI->mpMainSequence;
     116             :         }
     117             :         DBG_ASSERT( mpMainSequence, "sd::MainSequenceChangeGuard::MainSequenceChangeGuard(), no main sequence to guard!" );
     118             : 
     119           0 :         if( mpMainSequence )
     120           0 :             mpMainSequence->mbIgnoreChanges++;
     121           0 :     }
     122             : 
     123           0 :     ~MainSequenceChangeGuard()
     124             :     {
     125           0 :         if( mpMainSequence )
     126           0 :             mpMainSequence->mbIgnoreChanges++;
     127           0 :     }
     128             : 
     129             : private:
     130             :     MainSequence* mpMainSequence;
     131             : };
     132             : 
     133           0 : CustomAnimationEffect::CustomAnimationEffect( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xNode )
     134             : :   mnNodeType(-1),
     135             :     mnPresetClass(-1),
     136             :     mfBegin(-1.0),
     137             :     mfDuration(-1.0),
     138             :     mfAbsoluteDuration(-1.0),
     139             :     mnGroupId(-1),
     140             :     mnIterateType(0),
     141             :     mfIterateInterval(0.0),
     142             :     mnParaDepth( -1 ),
     143             :     mbHasText(false),
     144             :     mfAcceleration( 1.0 ),
     145             :     mfDecelerate( 1.0 ),
     146             :     mbAutoReverse(false),
     147             :     mnTargetSubItem(0),
     148             :     mnCommand(0),
     149             :     mpEffectSequence( 0 ),
     150             :     mbHasAfterEffect(false),
     151           0 :     mbAfterEffectOnNextEffect(false)
     152             : {
     153           0 :     setNode( xNode );
     154           0 : }
     155             : 
     156           0 : void CustomAnimationEffect::setNode( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xNode )
     157             : {
     158           0 :     mxNode = xNode;
     159           0 :     mxAudio.clear();
     160             : 
     161           0 :     Sequence< NamedValue > aUserData( mxNode->getUserData() );
     162           0 :     sal_Int32 nLength = aUserData.getLength();
     163           0 :     const NamedValue* p = aUserData.getConstArray();
     164             : 
     165           0 :     while( nLength-- )
     166             :     {
     167           0 :         if ( p->Name == "node-type" )
     168             :         {
     169           0 :             p->Value >>= mnNodeType;
     170             :         }
     171           0 :         else if ( p->Name == "preset-id" )
     172             :         {
     173           0 :             p->Value >>= maPresetId;
     174             :         }
     175           0 :         else if ( p->Name == "preset-sub-type" )
     176             :         {
     177           0 :             p->Value >>= maPresetSubType;
     178             :         }
     179           0 :         else if ( p->Name == "preset-class" )
     180             :         {
     181           0 :             p->Value >>= mnPresetClass;
     182             :         }
     183           0 :         else if ( p->Name == "preset-property" )
     184             :         {
     185           0 :             p->Value >>= maProperty;
     186             :         }
     187           0 :         else if ( p->Name == "group-id" )
     188             :         {
     189           0 :             p->Value >>= mnGroupId;
     190             :         }
     191             : 
     192           0 :         p++;
     193             :     }
     194             : 
     195             :     // get effect start time
     196           0 :     mxNode->getBegin() >>= mfBegin;
     197             : 
     198           0 :     mfAcceleration = mxNode->getAcceleration();
     199           0 :     mfDecelerate = mxNode->getDecelerate();
     200           0 :     mbAutoReverse = mxNode->getAutoReverse();
     201             : 
     202             :     // get iteration data
     203           0 :     Reference< XIterateContainer > xIter( mxNode, UNO_QUERY );
     204           0 :     if( xIter.is() )
     205             :     {
     206           0 :         mfIterateInterval = xIter->getIterateInterval();
     207           0 :         mnIterateType = xIter->getIterateType();
     208           0 :         maTarget = xIter->getTarget();
     209           0 :         mnTargetSubItem = xIter->getSubItem();
     210             :     }
     211             :     else
     212             :     {
     213           0 :         mfIterateInterval = 0.0f;
     214           0 :         mnIterateType = 0;
     215             :     }
     216             : 
     217             :     // calculate effect duration and get target shape
     218           0 :     Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY );
     219           0 :     if( xEnumerationAccess.is() )
     220             :     {
     221           0 :         Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY );
     222           0 :         if( xEnumeration.is() )
     223             :         {
     224           0 :             while( xEnumeration->hasMoreElements() )
     225             :             {
     226           0 :                 Reference< XAnimationNode > xChildNode( xEnumeration->nextElement(), UNO_QUERY );
     227           0 :                 if( !xChildNode.is() )
     228           0 :                     continue;
     229             : 
     230           0 :                 if( xChildNode->getType() == AnimationNodeType::AUDIO )
     231             :                 {
     232           0 :                     mxAudio.set( xChildNode, UNO_QUERY );
     233             :                 }
     234           0 :                 else if( xChildNode->getType() == AnimationNodeType::COMMAND )
     235             :                 {
     236           0 :                     Reference< XCommand > xCommand( xChildNode, UNO_QUERY );
     237           0 :                     if( xCommand.is() )
     238             :                     {
     239           0 :                         mnCommand = xCommand->getCommand();
     240           0 :                         if( !maTarget.hasValue() )
     241           0 :                             maTarget = xCommand->getTarget();
     242           0 :                     }
     243             :                 }
     244             :                 else
     245             :                 {
     246           0 :                     double fBegin = 0.0;
     247           0 :                     double fDuration = 0.0;
     248           0 :                     xChildNode->getBegin() >>= fBegin;
     249           0 :                     xChildNode->getDuration() >>= fDuration;
     250             : 
     251           0 :                     fDuration += fBegin;
     252           0 :                     if( fDuration > mfDuration )
     253           0 :                         mfDuration = fDuration;
     254             : 
     255             :                     // no target shape yet?
     256           0 :                     if( !maTarget.hasValue() )
     257             :                     {
     258             :                         // go get it boys!
     259           0 :                         Reference< XAnimate > xAnimate( xChildNode, UNO_QUERY );
     260           0 :                         if( xAnimate.is() )
     261             :                         {
     262           0 :                             maTarget = xAnimate->getTarget();
     263           0 :                             mnTargetSubItem = xAnimate->getSubItem();
     264           0 :                         }
     265             :                     }
     266             :                 }
     267           0 :             }
     268           0 :         }
     269             :     }
     270             : 
     271           0 :     mfAbsoluteDuration = mfDuration;
     272           0 :     double fRepeatCount = 1.0;
     273           0 :     if( (mxNode->getRepeatCount()) >>= fRepeatCount )
     274           0 :         mfAbsoluteDuration *= fRepeatCount;
     275             : 
     276           0 :     checkForText();
     277           0 : }
     278             : 
     279           0 : sal_Int32 CustomAnimationEffect::getNumberOfSubitems( const Any& aTarget, sal_Int16 nIterateType )
     280             : {
     281           0 :     sal_Int32 nSubItems = 0;
     282             : 
     283             :     try
     284             :     {
     285             :         // first get target text
     286           0 :         sal_Int32 nOnlyPara = -1;
     287             : 
     288           0 :         Reference< XText > xShape;
     289           0 :         aTarget >>= xShape;
     290           0 :         if( !xShape.is() )
     291             :         {
     292           0 :             ParagraphTarget aParaTarget;
     293           0 :             if( aTarget >>= aParaTarget )
     294             :             {
     295           0 :                 xShape.set( aParaTarget.Shape, UNO_QUERY );
     296           0 :                 nOnlyPara = aParaTarget.Paragraph;
     297           0 :             }
     298             :         }
     299             : 
     300             :         // now use the break iterator to iterate over the given text
     301             :         // and count the sub items
     302             : 
     303           0 :         if( xShape.is() )
     304             :         {
     305             :             // TODO/LATER: Optimize this, don't create a break iterator each time
     306           0 :             Reference< uno::XComponentContext > xContext( ::comphelper::getProcessComponentContext() );
     307           0 :             Reference < i18n::XBreakIterator > xBI = i18n::BreakIterator::create(xContext);
     308             : 
     309           0 :             Reference< XEnumerationAccess > xEA( xShape, UNO_QUERY_THROW );
     310           0 :             Reference< XEnumeration > xEnumeration( xEA->createEnumeration(), UNO_QUERY_THROW );
     311           0 :             Locale aLocale;
     312           0 :             const OUString aStrLocaleName( "CharLocale" );
     313           0 :             Reference< XTextRange > xParagraph;
     314             : 
     315           0 :             sal_Int32 nPara = 0;
     316           0 :             while( xEnumeration->hasMoreElements() )
     317             :             {
     318           0 :                 xEnumeration->nextElement() >>= xParagraph;
     319             : 
     320             :                 // skip this if its not the only paragraph we want to count
     321           0 :                 if( (nOnlyPara != -1) && (nOnlyPara != nPara ) )
     322           0 :                     continue;
     323             : 
     324           0 :                 if( nIterateType == TextAnimationType::BY_PARAGRAPH )
     325             :                 {
     326           0 :                     nSubItems++;
     327             :                 }
     328             :                 else
     329             :                 {
     330           0 :                     const OUString aText( xParagraph->getString() );
     331           0 :                     Reference< XPropertySet > xSet( xParagraph, UNO_QUERY_THROW );
     332           0 :                     xSet->getPropertyValue( aStrLocaleName ) >>= aLocale;
     333             : 
     334             :                     sal_Int32 nPos;
     335           0 :                     const sal_Int32 nEndPos = aText.getLength();
     336             : 
     337           0 :                     if( nIterateType == TextAnimationType::BY_WORD )
     338             :                     {
     339           0 :                         for( nPos = 0; nPos < nEndPos; nPos++ )
     340             :                         {
     341           0 :                             nPos = xBI->getWordBoundary(aText, nPos, aLocale, i18n::WordType::ANY_WORD, sal_True).endPos;
     342           0 :                             nSubItems++;
     343             :                         }
     344           0 :                         break;
     345             :                     }
     346             :                     else
     347             :                     {
     348             :                         sal_Int32 nDone;
     349           0 :                         for( nPos = 0; nPos < nEndPos; nPos++ )
     350             :                         {
     351           0 :                             nPos = xBI->nextCharacters(aText, nPos, aLocale, i18n::CharacterIteratorMode::SKIPCELL, 0, nDone);
     352           0 :                             nSubItems++;
     353             :                         }
     354           0 :                     }
     355             :                 }
     356             : 
     357           0 :                 if( nPara == nOnlyPara )
     358           0 :                     break;
     359             : 
     360           0 :                 nPara++;
     361           0 :             }
     362           0 :         }
     363             :     }
     364           0 :     catch( Exception& )
     365             :     {
     366           0 :         nSubItems = 0;
     367             :         OSL_FAIL( "sd::CustomAnimationEffect::getNumberOfSubitems(), exception caught!" );
     368             :     }
     369             : 
     370           0 :     return nSubItems;
     371             : }
     372             : 
     373           0 : CustomAnimationEffect::~CustomAnimationEffect()
     374             : {
     375           0 : }
     376             : 
     377           0 : CustomAnimationEffectPtr CustomAnimationEffect::clone() const
     378             : {
     379           0 :     Reference< XCloneable > xCloneable( mxNode, UNO_QUERY_THROW );
     380           0 :     Reference< XAnimationNode > xNode( xCloneable->createClone(), UNO_QUERY_THROW );
     381           0 :     CustomAnimationEffectPtr pEffect( new CustomAnimationEffect( xNode ) );
     382           0 :     pEffect->setEffectSequence( getEffectSequence() );
     383           0 :     return pEffect;
     384             : }
     385             : 
     386           0 : sal_Int32 CustomAnimationEffect::get_node_type( const Reference< XAnimationNode >& xNode )
     387             : {
     388           0 :     sal_Int16 nNodeType = -1;
     389             : 
     390           0 :     if( xNode.is() )
     391             :     {
     392           0 :         Sequence< NamedValue > aUserData( xNode->getUserData() );
     393           0 :         sal_Int32 nLength = aUserData.getLength();
     394           0 :         if( nLength )
     395             :         {
     396           0 :             const NamedValue* p = aUserData.getConstArray();
     397           0 :             while( nLength-- )
     398             :             {
     399           0 :                 if ( p->Name == "node-type" )
     400             :                 {
     401           0 :                     p->Value >>= nNodeType;
     402           0 :                     break;
     403             :                 }
     404           0 :                 p++;
     405             :             }
     406           0 :         }
     407             :     }
     408             : 
     409           0 :     return nNodeType;
     410             : }
     411             : 
     412           0 : void CustomAnimationEffect::setPresetClass( sal_Int16 nPresetClass )
     413             : {
     414           0 :     if( mnPresetClass != nPresetClass )
     415             :     {
     416           0 :         mnPresetClass = nPresetClass;
     417           0 :         if( mxNode.is() )
     418             :         {
     419             :             // first try to find a "preset-class" entry in the user data
     420             :             // and change it
     421           0 :             Sequence< NamedValue > aUserData( mxNode->getUserData() );
     422           0 :             sal_Int32 nLength = aUserData.getLength();
     423           0 :             bool bFound = false;
     424           0 :             if( nLength )
     425             :             {
     426           0 :                 NamedValue* p = aUserData.getArray();
     427           0 :                 while( nLength-- )
     428             :                 {
     429           0 :                     if ( p->Name == "preset-class" )
     430             :                     {
     431           0 :                         p->Value <<= mnPresetClass;
     432           0 :                         bFound = true;
     433           0 :                         break;
     434             :                     }
     435           0 :                     p++;
     436             :                 }
     437             :             }
     438             : 
     439             :             // no "node-type" entry inside user data, so add it
     440           0 :             if( !bFound )
     441             :             {
     442           0 :                 nLength = aUserData.getLength();
     443           0 :                 aUserData.realloc( nLength + 1);
     444           0 :                 aUserData[nLength].Name = "preset-class";
     445           0 :                 aUserData[nLength].Value <<= mnPresetClass;
     446             :             }
     447             : 
     448           0 :             mxNode->setUserData( aUserData );
     449             :         }
     450             :     }
     451           0 : }
     452             : 
     453           0 : void CustomAnimationEffect::setNodeType( sal_Int16 nNodeType )
     454             : {
     455           0 :     if( mnNodeType != nNodeType )
     456             :     {
     457           0 :         mnNodeType = nNodeType;
     458           0 :         if( mxNode.is() )
     459             :         {
     460             :             // first try to find a "node-type" entry in the user data
     461             :             // and change it
     462           0 :             Sequence< NamedValue > aUserData( mxNode->getUserData() );
     463           0 :             sal_Int32 nLength = aUserData.getLength();
     464           0 :             bool bFound = false;
     465           0 :             if( nLength )
     466             :             {
     467           0 :                 NamedValue* p = aUserData.getArray();
     468           0 :                 while( nLength-- )
     469             :                 {
     470           0 :                     if ( p->Name == "node-type" )
     471             :                     {
     472           0 :                         p->Value <<= mnNodeType;
     473           0 :                         bFound = true;
     474           0 :                         break;
     475             :                     }
     476           0 :                     p++;
     477             :                 }
     478             :             }
     479             : 
     480             :             // no "node-type" entry inside user data, so add it
     481           0 :             if( !bFound )
     482             :             {
     483           0 :                 nLength = aUserData.getLength();
     484           0 :                 aUserData.realloc( nLength + 1);
     485           0 :                 aUserData[nLength].Name = "node-type";
     486           0 :                 aUserData[nLength].Value <<= mnNodeType;
     487             :             }
     488             : 
     489           0 :             mxNode->setUserData( aUserData );
     490             :         }
     491             :     }
     492           0 : }
     493             : 
     494           0 : void CustomAnimationEffect::setGroupId( sal_Int32 nGroupId )
     495             : {
     496           0 :     mnGroupId = nGroupId;
     497           0 :     if( mxNode.is() )
     498             :     {
     499             :         // first try to find a "group-id" entry in the user data
     500             :         // and change it
     501           0 :         Sequence< NamedValue > aUserData( mxNode->getUserData() );
     502           0 :         sal_Int32 nLength = aUserData.getLength();
     503           0 :         bool bFound = false;
     504           0 :         if( nLength )
     505             :         {
     506           0 :             NamedValue* p = aUserData.getArray();
     507           0 :             while( nLength-- )
     508             :             {
     509           0 :                 if ( p->Name == "group-id" )
     510             :                 {
     511           0 :                     p->Value <<= mnGroupId;
     512           0 :                     bFound = true;
     513           0 :                     break;
     514             :                 }
     515           0 :                 p++;
     516             :             }
     517             :         }
     518             : 
     519             :         // no "node-type" entry inside user data, so add it
     520           0 :         if( !bFound )
     521             :         {
     522           0 :             nLength = aUserData.getLength();
     523           0 :             aUserData.realloc( nLength + 1);
     524           0 :             aUserData[nLength].Name = "group-id";
     525           0 :             aUserData[nLength].Value <<= mnGroupId;
     526             :         }
     527             : 
     528           0 :         mxNode->setUserData( aUserData );
     529             :     }
     530           0 : }
     531             : 
     532             : /** checks if the text for this effect has changed and updates internal flags.
     533             :     returns true if something changed.
     534             : */
     535           0 : bool CustomAnimationEffect::checkForText()
     536             : {
     537           0 :     bool bChange = false;
     538             : 
     539           0 :     Reference< XText > xText;
     540             : 
     541           0 :     if( maTarget.getValueType() == ::cppu::UnoType<ParagraphTarget>::get() )
     542             :     {
     543             :         // calc para depth
     544           0 :         ParagraphTarget aParaTarget;
     545           0 :         maTarget >>= aParaTarget;
     546             : 
     547           0 :         xText = Reference< XText >::query( aParaTarget.Shape );
     548             : 
     549             :         // get paragraph
     550           0 :         if( xText.is() )
     551             :         {
     552           0 :             Reference< XEnumerationAccess > xEA( xText, UNO_QUERY );
     553           0 :             if( xEA.is() )
     554             :             {
     555           0 :                 Reference< XEnumeration > xEnumeration( xEA->createEnumeration(), UNO_QUERY );
     556           0 :                 if( xEnumeration.is() )
     557             :                 {
     558           0 :                     bool bHasText = xEnumeration->hasMoreElements();
     559           0 :                     bChange |= bHasText != mbHasText;
     560           0 :                     mbHasText = bHasText;
     561             : 
     562           0 :                     sal_Int32 nPara = aParaTarget.Paragraph;
     563             : 
     564           0 :                     while( xEnumeration->hasMoreElements() && nPara-- )
     565           0 :                         xEnumeration->nextElement();
     566             : 
     567           0 :                     if( xEnumeration->hasMoreElements() )
     568             :                     {
     569           0 :                         Reference< XPropertySet > xParaSet;
     570           0 :                         xEnumeration->nextElement() >>= xParaSet;
     571           0 :                         if( xParaSet.is() )
     572             :                         {
     573           0 :                             sal_Int32 nParaDepth = 0;
     574           0 :                             const OUString strNumberingLevel( "NumberingLevel" );
     575           0 :                             xParaSet->getPropertyValue( strNumberingLevel ) >>= nParaDepth;
     576           0 :                             bChange |= nParaDepth != mnParaDepth;
     577           0 :                             mnParaDepth = nParaDepth;
     578           0 :                         }
     579             :                     }
     580           0 :                 }
     581           0 :             }
     582           0 :         }
     583             :     }
     584             :     else
     585             :     {
     586           0 :         maTarget >>= xText;
     587           0 :         bool bHasText = xText.is() && !xText->getString().isEmpty();
     588           0 :         bChange |= bHasText != mbHasText;
     589           0 :         mbHasText = bHasText;
     590             :     }
     591             : 
     592           0 :     bChange |= calculateIterateDuration();
     593           0 :     return bChange;
     594             : }
     595             : 
     596           0 : bool CustomAnimationEffect::calculateIterateDuration()
     597             : {
     598           0 :     bool bChange = false;
     599             : 
     600             :     // if we have an iteration, we must also calculate the
     601             :     // 'true' container duration, that is
     602             :     // ( ( is form animated ) ? [contained effects duration] : 0 ) +
     603             :     // ( [number of animated children] - 1 ) * [interval-delay] + [contained effects duration]
     604           0 :     Reference< XIterateContainer > xIter( mxNode, UNO_QUERY );
     605           0 :     if( xIter.is() )
     606             :     {
     607           0 :         double fDuration = mfDuration;
     608           0 :         const double fSubEffectDuration = mfDuration;
     609             : 
     610           0 :         if( mnTargetSubItem != ShapeAnimationSubType::ONLY_BACKGROUND ) // does not make sense for iterate container but better check
     611             :         {
     612           0 :             const sal_Int32 nSubItems = getNumberOfSubitems( maTarget, mnIterateType );
     613           0 :             if( nSubItems )
     614             :             {
     615           0 :                 const double f = (nSubItems-1) * mfIterateInterval;
     616           0 :                 fDuration += f;
     617             :             }
     618             :         }
     619             : 
     620             :         // if we also animate the form first, we have to add the
     621             :         // sub effect duration to the whole effect duration
     622           0 :         if( mnTargetSubItem == ShapeAnimationSubType::AS_WHOLE )
     623           0 :             fDuration += fSubEffectDuration;
     624             : 
     625           0 :         bChange |= fDuration != mfAbsoluteDuration;
     626           0 :         mfAbsoluteDuration = fDuration;
     627             :     }
     628             : 
     629           0 :     return bChange;
     630             : }
     631             : 
     632           0 : void CustomAnimationEffect::setTarget( const ::com::sun::star::uno::Any& rTarget )
     633             : {
     634             :     try
     635             :     {
     636           0 :         maTarget = rTarget;
     637             : 
     638             :         // first, check special case for random node
     639           0 :         Reference< XInitialization > xInit( mxNode, UNO_QUERY );
     640           0 :         if( xInit.is() )
     641             :         {
     642           0 :             const Sequence< Any > aArgs( &maTarget, 1 );
     643           0 :             xInit->initialize( aArgs );
     644             :         }
     645             :         else
     646             :         {
     647           0 :             Reference< XIterateContainer > xIter( mxNode, UNO_QUERY );
     648           0 :             if( xIter.is() )
     649             :             {
     650           0 :                 xIter->setTarget(maTarget);
     651             :             }
     652             :             else
     653             :             {
     654           0 :                 Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY );
     655           0 :                 if( xEnumerationAccess.is() )
     656             :                 {
     657           0 :                     Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY );
     658           0 :                     if( xEnumeration.is() )
     659             :                     {
     660           0 :                         while( xEnumeration->hasMoreElements() )
     661             :                         {
     662           0 :                             const Any aElem( xEnumeration->nextElement() );
     663           0 :                             Reference< XAnimate > xAnimate( aElem, UNO_QUERY );
     664           0 :                             if( xAnimate.is() )
     665           0 :                                 xAnimate->setTarget( rTarget );
     666             :                             else
     667             :                             {
     668           0 :                                 Reference< XCommand > xCommand( aElem, UNO_QUERY );
     669           0 :                                 if( xCommand.is() )
     670           0 :                                     xCommand->setTarget( rTarget );
     671             :                             }
     672           0 :                         }
     673           0 :                     }
     674           0 :                 }
     675           0 :             }
     676             :         }
     677           0 :         checkForText();
     678             :     }
     679           0 :     catch( Exception& )
     680             :     {
     681             :         OSL_FAIL( "sd::CustomAnimationEffect::setTarget(), exception caught!" );
     682             :     }
     683           0 : }
     684             : 
     685           0 : void CustomAnimationEffect::setTargetSubItem( sal_Int16 nSubItem )
     686             : {
     687             :     try
     688             :     {
     689           0 :         mnTargetSubItem = nSubItem;
     690             : 
     691           0 :         Reference< XIterateContainer > xIter( mxNode, UNO_QUERY );
     692           0 :         if( xIter.is() )
     693             :         {
     694           0 :             xIter->setSubItem(mnTargetSubItem);
     695             :         }
     696             :         else
     697             :         {
     698           0 :             Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY );
     699           0 :             if( xEnumerationAccess.is() )
     700             :             {
     701           0 :                 Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY );
     702           0 :                 if( xEnumeration.is() )
     703             :                 {
     704           0 :                     while( xEnumeration->hasMoreElements() )
     705             :                     {
     706           0 :                         Reference< XAnimate > xAnimate( xEnumeration->nextElement(), UNO_QUERY );
     707           0 :                         if( xAnimate.is() )
     708           0 :                             xAnimate->setSubItem( mnTargetSubItem );
     709           0 :                     }
     710           0 :                 }
     711           0 :             }
     712           0 :         }
     713             :     }
     714           0 :     catch( Exception& )
     715             :     {
     716             :         OSL_FAIL( "sd::CustomAnimationEffect::setTargetSubItem(), exception caught!" );
     717             :     }
     718           0 : }
     719             : 
     720           0 : void CustomAnimationEffect::setDuration( double fDuration )
     721             : {
     722           0 :     if( (mfDuration != -1.0) && (mfDuration != fDuration) ) try
     723             :     {
     724           0 :         double fScale = fDuration / mfDuration;
     725           0 :         mfDuration = fDuration;
     726           0 :         double fRepeatCount = 1.0;
     727           0 :         getRepeatCount() >>= fRepeatCount;
     728           0 :         mfAbsoluteDuration = mfDuration * fRepeatCount;
     729             : 
     730             :         // calculate effect duration and get target shape
     731           0 :         Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY );
     732           0 :         if( xEnumerationAccess.is() )
     733             :         {
     734           0 :             Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY );
     735           0 :             if( xEnumeration.is() )
     736             :             {
     737           0 :                 while( xEnumeration->hasMoreElements() )
     738             :                 {
     739           0 :                     Reference< XAnimationNode > xChildNode( xEnumeration->nextElement(), UNO_QUERY );
     740           0 :                     if( !xChildNode.is() )
     741           0 :                         continue;
     742             : 
     743           0 :                     double fChildBegin = 0.0;
     744           0 :                     xChildNode->getBegin() >>= fChildBegin;
     745           0 :                     if(  fChildBegin != 0.0 )
     746             :                     {
     747           0 :                         fChildBegin *= fScale;
     748           0 :                         xChildNode->setBegin( makeAny( fChildBegin ) );
     749             :                     }
     750             : 
     751           0 :                     double fChildDuration = 0.0;
     752           0 :                     xChildNode->getDuration() >>= fChildDuration;
     753           0 :                     if( fChildDuration != 0.0 )
     754             :                     {
     755           0 :                         fChildDuration *= fScale;
     756           0 :                         xChildNode->setDuration( makeAny( fChildDuration ) );
     757             :                     }
     758           0 :                 }
     759           0 :             }
     760             :         }
     761           0 :         calculateIterateDuration();
     762             :     }
     763           0 :     catch( Exception& )
     764             :     {
     765             :         OSL_FAIL( "sd::CustomAnimationEffect::setDuration(), exception caught!" );
     766             :     }
     767           0 : }
     768             : 
     769           0 : void CustomAnimationEffect::setBegin( double fBegin )
     770             : {
     771           0 :     if( mxNode.is() ) try
     772             :     {
     773           0 :         mfBegin = fBegin;
     774           0 :         mxNode->setBegin( makeAny( fBegin ) );
     775             :     }
     776           0 :     catch( Exception& )
     777             :     {
     778             :         OSL_FAIL( "sd::CustomAnimationEffect::setBegin(), exception caught!" );
     779             :     }
     780           0 : }
     781             : 
     782           0 : void CustomAnimationEffect::setAcceleration( double fAcceleration )
     783             : {
     784           0 :     if( mxNode.is() ) try
     785             :     {
     786           0 :         mfAcceleration = fAcceleration;
     787           0 :         mxNode->setAcceleration( fAcceleration );
     788             :     }
     789           0 :     catch( Exception& )
     790             :     {
     791             :         OSL_FAIL( "sd::CustomAnimationEffect::setAcceleration(), exception caught!" );
     792             :     }
     793           0 : }
     794             : 
     795           0 : void CustomAnimationEffect::setDecelerate( double fDecelerate )
     796             : {
     797           0 :     if( mxNode.is() ) try
     798             :     {
     799           0 :         mfDecelerate = fDecelerate;
     800           0 :         mxNode->setDecelerate( fDecelerate );
     801             :     }
     802           0 :     catch( Exception& )
     803             :     {
     804             :         OSL_FAIL( "sd::CustomAnimationEffect::setDecelerate(), exception caught!" );
     805             :     }
     806           0 : }
     807             : 
     808           0 : void CustomAnimationEffect::setAutoReverse( bool bAutoReverse )
     809             : {
     810           0 :     if( mxNode.is() ) try
     811             :     {
     812           0 :         mbAutoReverse = bAutoReverse;
     813           0 :         mxNode->setAutoReverse( bAutoReverse );
     814             :     }
     815           0 :     catch( Exception& )
     816             :     {
     817             :         OSL_FAIL( "sd::CustomAnimationEffect::setAutoReverse(), exception caught!" );
     818             :     }
     819           0 : }
     820             : 
     821           0 : void CustomAnimationEffect::replaceNode( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xNode )
     822             : {
     823           0 :     sal_Int16 nNodeType = mnNodeType;
     824           0 :     Any aTarget = maTarget;
     825             : 
     826           0 :     double fBegin = mfBegin;
     827           0 :     double fDuration = mfDuration;
     828           0 :     double fAcceleration = mfAcceleration;
     829           0 :     double fDecelerate = mfDecelerate ;
     830           0 :     bool bAutoReverse = mbAutoReverse;
     831           0 :     Reference< XAudio > xAudio( mxAudio );
     832           0 :     sal_Int16 nIterateType = mnIterateType;
     833           0 :     double fIterateInterval = mfIterateInterval;
     834           0 :     sal_Int16 nSubItem = mnTargetSubItem;
     835             : 
     836           0 :     setNode( xNode );
     837             : 
     838           0 :     setAudio( xAudio );
     839           0 :     setNodeType( nNodeType );
     840           0 :     setTarget( aTarget );
     841           0 :     setTargetSubItem( nSubItem );
     842           0 :     setDuration( fDuration );
     843           0 :     setBegin( fBegin );
     844             : 
     845           0 :     setAcceleration( fAcceleration );
     846           0 :     setDecelerate( fDecelerate );
     847           0 :     setAutoReverse( bAutoReverse );
     848             : 
     849           0 :     if( nIterateType != mnIterateType )
     850           0 :         setIterateType( nIterateType );
     851             : 
     852           0 :     if( mnIterateType && ( fIterateInterval != mfIterateInterval ) )
     853           0 :         setIterateInterval( fIterateInterval );
     854           0 : }
     855             : 
     856           0 : Reference< XShape > CustomAnimationEffect::getTargetShape() const
     857             : {
     858           0 :     Reference< XShape > xShape;
     859           0 :     maTarget >>= xShape;
     860           0 :     if( !xShape.is() )
     861             :     {
     862           0 :         ParagraphTarget aParaTarget;
     863           0 :         if( maTarget >>= aParaTarget )
     864           0 :             xShape = aParaTarget.Shape;
     865             :     }
     866             : 
     867           0 :     return xShape;
     868             : }
     869             : 
     870           0 : Any CustomAnimationEffect::getRepeatCount() const
     871             : {
     872           0 :     if( mxNode.is() )
     873             :     {
     874           0 :         return mxNode->getRepeatCount();
     875             :     }
     876             :     else
     877             :     {
     878           0 :         Any aAny;
     879           0 :         return aAny;
     880             :     }
     881             : }
     882             : 
     883           0 : Any CustomAnimationEffect::getEnd() const
     884             : {
     885           0 :     if( mxNode.is() )
     886             :     {
     887           0 :         return mxNode->getEnd();
     888             :     }
     889             :     else
     890             :     {
     891           0 :         Any aAny;
     892           0 :         return aAny;
     893             :     }
     894             : }
     895             : 
     896           0 : sal_Int16 CustomAnimationEffect::getFill() const
     897             : {
     898           0 :     if( mxNode.is() )
     899           0 :         return mxNode->getFill();
     900             :     else
     901           0 :         return 0;
     902             : }
     903             : 
     904           0 : void CustomAnimationEffect::setRepeatCount( const Any& rRepeatCount )
     905             : {
     906           0 :     if( mxNode.is() )
     907             :     {
     908           0 :         mxNode->setRepeatCount( rRepeatCount );
     909           0 :         double fRepeatCount = 1.0;
     910           0 :         rRepeatCount >>= fRepeatCount;
     911           0 :         mfAbsoluteDuration = mfDuration * fRepeatCount;
     912             :     }
     913           0 : }
     914             : 
     915           0 : void CustomAnimationEffect::setEnd( const Any& rEnd )
     916             : {
     917           0 :     if( mxNode.is() )
     918           0 :         mxNode->setEnd( rEnd );
     919           0 : }
     920             : 
     921           0 : void CustomAnimationEffect::setFill( sal_Int16 nFill )
     922             : {
     923           0 :     if( mxNode.is() )
     924           0 :         mxNode->setFill( nFill );
     925           0 : }
     926             : 
     927           0 : Reference< XAnimationNode > CustomAnimationEffect::createAfterEffectNode() const throw (Exception)
     928             : {
     929             :     DBG_ASSERT( mbHasAfterEffect, "sd::CustomAnimationEffect::createAfterEffectNode(), this node has no after effect!" );
     930             : 
     931           0 :     Reference< XComponentContext > xContext( ::comphelper::getProcessComponentContext() );
     932             : 
     933           0 :     Reference< XAnimate > xAnimate;
     934           0 :     if( maDimColor.hasValue() )
     935           0 :         xAnimate = AnimateColor::create( xContext );
     936             :     else
     937           0 :         xAnimate = AnimateSet::create( xContext );
     938             : 
     939           0 :     Any aTo;
     940           0 :     OUString aAttributeName;
     941             : 
     942           0 :     if( maDimColor.hasValue() )
     943             :     {
     944           0 :         aTo = maDimColor;
     945           0 :         aAttributeName = "DimColor";
     946             :     }
     947             :     else
     948             :     {
     949           0 :         aTo = makeAny( false );
     950           0 :         aAttributeName = "Visibility";
     951             :     }
     952             : 
     953           0 :     Any aBegin;
     954           0 :     if( !mbAfterEffectOnNextEffect ) // sameClick
     955             :     {
     956           0 :         Event aEvent;
     957             : 
     958           0 :         aEvent.Source <<= getNode();
     959           0 :         aEvent.Trigger = EventTrigger::END_EVENT;
     960           0 :         aEvent.Repeat = 0;
     961             : 
     962           0 :         aBegin <<= aEvent;
     963             :     }
     964             :     else
     965             :     {
     966           0 :         aBegin <<= (double)0.0;
     967             :     }
     968             : 
     969           0 :     xAnimate->setBegin( aBegin );
     970           0 :     xAnimate->setTo( aTo );
     971           0 :     xAnimate->setAttributeName( aAttributeName );
     972             : 
     973           0 :     xAnimate->setDuration( makeAny( (double)0.001 ) );
     974           0 :     xAnimate->setFill( AnimationFill::HOLD );
     975           0 :     xAnimate->setTarget( maTarget );
     976             : 
     977           0 :     return xAnimate;
     978             : }
     979             : 
     980           0 : void CustomAnimationEffect::setIterateType( sal_Int16 nIterateType )
     981             : {
     982           0 :     if( mnIterateType != nIterateType ) try
     983             :     {
     984             :         // do we need to exchange the container node?
     985           0 :         if( (mnIterateType == 0) || (nIterateType == 0) )
     986             :         {
     987           0 :             sal_Int16 nTargetSubItem = mnTargetSubItem;
     988             : 
     989           0 :             Reference< XComponentContext > xContext( ::comphelper::getProcessComponentContext() );
     990           0 :             Reference< XTimeContainer > xNewContainer;
     991           0 :             if(nIterateType)
     992             :             {
     993           0 :                 xNewContainer.set( IterateContainer::create( xContext ) );
     994             :             }
     995             :             else
     996           0 :                 xNewContainer.set( ParallelTimeContainer::create( xContext ), UNO_QUERY_THROW );
     997             : 
     998           0 :             Reference< XTimeContainer > xOldContainer( mxNode, UNO_QUERY_THROW );
     999           0 :             Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY_THROW );
    1000           0 :             Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    1001           0 :             while( xEnumeration->hasMoreElements() )
    1002             :             {
    1003           0 :                 Reference< XAnimationNode > xChildNode( xEnumeration->nextElement(), UNO_QUERY_THROW );
    1004           0 :                 xOldContainer->removeChild( xChildNode );
    1005           0 :                 xNewContainer->appendChild( xChildNode );
    1006           0 :             }
    1007             : 
    1008           0 :             xNewContainer->setBegin( mxNode->getBegin() );
    1009           0 :             xNewContainer->setDuration( mxNode->getDuration() );
    1010           0 :             xNewContainer->setEnd( mxNode->getEnd() );
    1011           0 :             xNewContainer->setEndSync( mxNode->getEndSync() );
    1012           0 :             xNewContainer->setRepeatCount( mxNode->getRepeatCount() );
    1013           0 :             xNewContainer->setFill( mxNode->getFill() );
    1014           0 :             xNewContainer->setFillDefault( mxNode->getFillDefault() );
    1015           0 :             xNewContainer->setRestart( mxNode->getRestart() );
    1016           0 :             xNewContainer->setRestartDefault( mxNode->getRestartDefault() );
    1017           0 :             xNewContainer->setAcceleration( mxNode->getAcceleration() );
    1018           0 :             xNewContainer->setDecelerate( mxNode->getDecelerate() );
    1019           0 :             xNewContainer->setAutoReverse( mxNode->getAutoReverse() );
    1020           0 :             xNewContainer->setRepeatDuration( mxNode->getRepeatDuration() );
    1021           0 :             xNewContainer->setEndSync( mxNode->getEndSync() );
    1022           0 :             xNewContainer->setRepeatCount( mxNode->getRepeatCount() );
    1023           0 :             xNewContainer->setUserData( mxNode->getUserData() );
    1024             : 
    1025           0 :             mxNode = xNewContainer;
    1026             : 
    1027           0 :             Any aTarget;
    1028           0 :             if( nIterateType )
    1029             :             {
    1030           0 :                 Reference< XIterateContainer > xIter( mxNode, UNO_QUERY_THROW );
    1031           0 :                 xIter->setTarget(maTarget);
    1032           0 :                 xIter->setSubItem( nTargetSubItem );
    1033             :             }
    1034             :             else
    1035             :             {
    1036           0 :                 aTarget = maTarget;
    1037             :             }
    1038             : 
    1039           0 :             Reference< XEnumerationAccess > xEA( mxNode, UNO_QUERY_THROW );
    1040           0 :             Reference< XEnumeration > xE( xEA->createEnumeration(), UNO_QUERY_THROW );
    1041           0 :             while( xE->hasMoreElements() )
    1042             :             {
    1043           0 :                 Reference< XAnimate > xAnimate( xE->nextElement(), UNO_QUERY );
    1044           0 :                 if( xAnimate.is() )
    1045             :                 {
    1046           0 :                     xAnimate->setTarget( aTarget );
    1047           0 :                     xAnimate->setSubItem( nTargetSubItem );
    1048             :                 }
    1049           0 :             }
    1050             :         }
    1051             : 
    1052           0 :         mnIterateType = nIterateType;
    1053             : 
    1054             :         // if we have an iteration container, we must set its type
    1055           0 :         if( mnIterateType )
    1056             :         {
    1057           0 :             Reference< XIterateContainer > xIter( mxNode, UNO_QUERY_THROW );
    1058           0 :             xIter->setIterateType( nIterateType );
    1059             :         }
    1060             : 
    1061           0 :         checkForText();
    1062             :     }
    1063           0 :     catch( Exception& )
    1064             :     {
    1065             :         OSL_FAIL( "sd::CustomAnimationEffect::setIterateType(), Exception caught!" );
    1066             :     }
    1067           0 : }
    1068             : 
    1069           0 : void CustomAnimationEffect::setIterateInterval( double fIterateInterval )
    1070             : {
    1071           0 :     if( mfIterateInterval != fIterateInterval )
    1072             :     {
    1073           0 :         Reference< XIterateContainer > xIter( mxNode, UNO_QUERY );
    1074             : 
    1075             :         DBG_ASSERT( xIter.is(), "sd::CustomAnimationEffect::setIterateInterval(), not an iteration node" );
    1076           0 :         if( xIter.is() )
    1077             :         {
    1078           0 :             mfIterateInterval = fIterateInterval;
    1079           0 :             xIter->setIterateInterval( fIterateInterval );
    1080             :         }
    1081             : 
    1082           0 :         calculateIterateDuration();
    1083             :     }
    1084           0 : }
    1085             : 
    1086           0 : OUString CustomAnimationEffect::getPath() const
    1087             : {
    1088           0 :     OUString aPath;
    1089             : 
    1090           0 :     if( mxNode.is() ) try
    1091             :     {
    1092           0 :         Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY_THROW );
    1093           0 :         Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    1094           0 :         while( xEnumeration->hasMoreElements() )
    1095             :         {
    1096           0 :             Reference< XAnimateMotion > xMotion( xEnumeration->nextElement(), UNO_QUERY );
    1097           0 :             if( xMotion.is() )
    1098             :             {
    1099           0 :                 xMotion->getPath() >>= aPath;
    1100           0 :                 break;
    1101             :             }
    1102           0 :         }
    1103             :     }
    1104           0 :     catch( Exception& )
    1105             :     {
    1106             :         OSL_FAIL("sd::CustomAnimationEffect::getPath(), exception caught!" );
    1107             :     }
    1108             : 
    1109           0 :     return aPath;
    1110             : }
    1111             : 
    1112           0 : void CustomAnimationEffect::setPath( const OUString& rPath )
    1113             : {
    1114           0 :     if( mxNode.is() ) try
    1115             :     {
    1116           0 :         Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY_THROW );
    1117           0 :         Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    1118           0 :         while( xEnumeration->hasMoreElements() )
    1119             :         {
    1120           0 :             Reference< XAnimateMotion > xMotion( xEnumeration->nextElement(), UNO_QUERY );
    1121           0 :             if( xMotion.is() )
    1122             :             {
    1123             : 
    1124           0 :                 MainSequenceChangeGuard aGuard( mpEffectSequence );
    1125           0 :                 xMotion->setPath( Any( rPath ) );
    1126           0 :                 break;
    1127             :             }
    1128           0 :         }
    1129             :     }
    1130           0 :     catch( Exception& )
    1131             :     {
    1132             :         OSL_FAIL("sd::CustomAnimationEffect::setPath(), exception caught!" );
    1133             :     }
    1134           0 : }
    1135             : 
    1136           0 : Any CustomAnimationEffect::getProperty( sal_Int32 nNodeType, const OUString& rAttributeName, EValue eValue )
    1137             : {
    1138           0 :     Any aProperty;
    1139           0 :     if( mxNode.is() ) try
    1140             :     {
    1141           0 :         Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY );
    1142           0 :         if( xEnumerationAccess.is() )
    1143             :         {
    1144           0 :             Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY );
    1145           0 :             if( xEnumeration.is() )
    1146             :             {
    1147           0 :                 while( xEnumeration->hasMoreElements() && !aProperty.hasValue() )
    1148             :                 {
    1149           0 :                     Reference< XAnimate > xAnimate( xEnumeration->nextElement(), UNO_QUERY );
    1150           0 :                     if( !xAnimate.is() )
    1151           0 :                         continue;
    1152             : 
    1153           0 :                     if( xAnimate->getType() == nNodeType )
    1154             :                     {
    1155           0 :                         if( xAnimate->getAttributeName() == rAttributeName )
    1156             :                         {
    1157           0 :                             switch( eValue )
    1158             :                             {
    1159           0 :                             case VALUE_FROM: aProperty = xAnimate->getFrom(); break;
    1160           0 :                             case VALUE_TO:   aProperty = xAnimate->getTo(); break;
    1161           0 :                             case VALUE_BY:   aProperty = xAnimate->getBy(); break;
    1162             :                             case VALUE_FIRST:
    1163             :                             case VALUE_LAST:
    1164             :                                 {
    1165           0 :                                     Sequence<Any> aValues( xAnimate->getValues() );
    1166           0 :                                     if( aValues.hasElements() )
    1167           0 :                                         aProperty =  aValues[ eValue == VALUE_FIRST ? 0 : aValues.getLength() - 1 ];
    1168             :                                 }
    1169           0 :                                 break;
    1170             :                             }
    1171             :                         }
    1172             :                     }
    1173           0 :                 }
    1174           0 :             }
    1175           0 :         }
    1176             :     }
    1177           0 :     catch( Exception& )
    1178             :     {
    1179             :         OSL_FAIL("sd::CustomAnimationEffect::getProperty(), exception caught!" );
    1180             :     }
    1181             : 
    1182           0 :     return aProperty;
    1183             : }
    1184             : 
    1185           0 : bool CustomAnimationEffect::setProperty( sal_Int32 nNodeType, const OUString& rAttributeName, EValue eValue, const Any& rValue )
    1186             : {
    1187           0 :     bool bChanged = false;
    1188           0 :     if( mxNode.is() ) try
    1189             :     {
    1190           0 :         Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY );
    1191           0 :         if( xEnumerationAccess.is() )
    1192             :         {
    1193           0 :             Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY );
    1194           0 :             if( xEnumeration.is() )
    1195             :             {
    1196           0 :                 while( xEnumeration->hasMoreElements() )
    1197             :                 {
    1198           0 :                     Reference< XAnimate > xAnimate( xEnumeration->nextElement(), UNO_QUERY );
    1199           0 :                     if( !xAnimate.is() )
    1200           0 :                         continue;
    1201             : 
    1202           0 :                     if( xAnimate->getType() == nNodeType )
    1203             :                     {
    1204           0 :                         if( xAnimate->getAttributeName() == rAttributeName )
    1205             :                         {
    1206           0 :                             switch( eValue )
    1207             :                             {
    1208             :                             case VALUE_FROM:
    1209           0 :                                 if( xAnimate->getFrom() != rValue )
    1210             :                                 {
    1211           0 :                                     xAnimate->setFrom( rValue );
    1212           0 :                                     bChanged = true;
    1213             :                                 }
    1214           0 :                                 break;
    1215             :                             case VALUE_TO:
    1216           0 :                                 if( xAnimate->getTo() != rValue )
    1217             :                                 {
    1218           0 :                                     xAnimate->setTo( rValue );
    1219           0 :                                     bChanged = true;
    1220             :                                 }
    1221           0 :                                 break;
    1222             :                             case VALUE_BY:
    1223           0 :                                 if( xAnimate->getTo() != rValue )
    1224             :                                 {
    1225           0 :                                     xAnimate->setBy( rValue );
    1226           0 :                                     bChanged = true;
    1227             :                                 }
    1228           0 :                                 break;
    1229             :                             case VALUE_FIRST:
    1230             :                             case VALUE_LAST:
    1231             :                                 {
    1232           0 :                                     Sequence<Any> aValues( xAnimate->getValues() );
    1233           0 :                                     if( !aValues.hasElements() )
    1234           0 :                                         aValues.realloc(1);
    1235             : 
    1236           0 :                                     sal_Int32 nIndex = eValue == VALUE_FIRST ? 0 : aValues.getLength() - 1;
    1237             : 
    1238           0 :                                     if( aValues[ nIndex ] != rValue )
    1239             :                                     {
    1240           0 :                                         aValues[ nIndex ] = rValue;
    1241           0 :                                         xAnimate->setValues( aValues );
    1242           0 :                                         bChanged = true;
    1243           0 :                                     }
    1244             :                                 }
    1245             :                             }
    1246             :                         }
    1247             :                     }
    1248           0 :                 }
    1249           0 :             }
    1250           0 :         }
    1251             :     }
    1252           0 :     catch( Exception& )
    1253             :     {
    1254             :         OSL_FAIL("sd::CustomAnimationEffect::setProperty(), exception caught!" );
    1255             :     }
    1256             : 
    1257           0 :     return bChanged;
    1258             : }
    1259             : 
    1260           0 : static bool implIsColorAttribute( const OUString& rAttributeName )
    1261             : {
    1262           0 :     return rAttributeName == "FillColor" || rAttributeName == "LineColor" || rAttributeName == "CharColor";
    1263             : }
    1264             : 
    1265           0 : Any CustomAnimationEffect::getColor( sal_Int32 nIndex )
    1266             : {
    1267           0 :     Any aColor;
    1268           0 :     if( mxNode.is() ) try
    1269             :     {
    1270           0 :         Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY );
    1271           0 :         if( xEnumerationAccess.is() )
    1272             :         {
    1273           0 :             Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY );
    1274           0 :             if( xEnumeration.is() )
    1275             :             {
    1276           0 :                 while( xEnumeration->hasMoreElements() && !aColor.hasValue() )
    1277             :                 {
    1278           0 :                     Reference< XAnimate > xAnimate( xEnumeration->nextElement(), UNO_QUERY );
    1279           0 :                     if( !xAnimate.is() )
    1280           0 :                         continue;
    1281             : 
    1282           0 :                     switch( xAnimate->getType() )
    1283             :                     {
    1284             :                     case AnimationNodeType::SET:
    1285             :                     case AnimationNodeType::ANIMATE:
    1286           0 :                         if( !implIsColorAttribute( xAnimate->getAttributeName() ) )
    1287           0 :                             break;
    1288             :                     case AnimationNodeType::ANIMATECOLOR:
    1289           0 :                         Sequence<Any> aValues( xAnimate->getValues() );
    1290           0 :                         if( aValues.hasElements() )
    1291             :                         {
    1292           0 :                             if( aValues.getLength() > nIndex )
    1293           0 :                                 aColor = aValues[nIndex];
    1294             :                         }
    1295           0 :                         else if( nIndex == 0 )
    1296           0 :                             aColor = xAnimate->getFrom();
    1297             :                         else
    1298           0 :                             aColor = xAnimate->getTo();
    1299             :                     }
    1300           0 :                 }
    1301           0 :             }
    1302           0 :         }
    1303             :     }
    1304           0 :     catch( Exception& )
    1305             :     {
    1306             :         OSL_FAIL("sd::CustomAnimationEffect::getColor(), exception caught!" );
    1307             :     }
    1308             : 
    1309           0 :     return aColor;
    1310             : }
    1311             : 
    1312           0 : void CustomAnimationEffect::setColor( sal_Int32 nIndex, const Any& rColor )
    1313             : {
    1314           0 :     if( mxNode.is() ) try
    1315             :     {
    1316           0 :         Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY );
    1317           0 :         if( xEnumerationAccess.is() )
    1318             :         {
    1319           0 :             Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY );
    1320           0 :             if( xEnumeration.is() )
    1321             :             {
    1322           0 :                 while( xEnumeration->hasMoreElements() )
    1323             :                 {
    1324           0 :                     Reference< XAnimate > xAnimate( xEnumeration->nextElement(), UNO_QUERY );
    1325           0 :                     if( !xAnimate.is() )
    1326           0 :                         continue;
    1327             : 
    1328           0 :                     switch( xAnimate->getType() )
    1329             :                     {
    1330             :                     case AnimationNodeType::SET:
    1331             :                     case AnimationNodeType::ANIMATE:
    1332           0 :                         if( !implIsColorAttribute( xAnimate->getAttributeName() ) )
    1333           0 :                             break;
    1334             :                     case AnimationNodeType::ANIMATECOLOR:
    1335             :                     {
    1336           0 :                         Sequence<Any> aValues( xAnimate->getValues() );
    1337           0 :                         if( aValues.hasElements() )
    1338             :                         {
    1339           0 :                             if( aValues.getLength() > nIndex )
    1340             :                             {
    1341           0 :                                 aValues[nIndex] = rColor;
    1342           0 :                                 xAnimate->setValues( aValues );
    1343             :                             }
    1344             :                         }
    1345           0 :                         else if( (nIndex == 0) && xAnimate->getFrom().hasValue() )
    1346           0 :                             xAnimate->setFrom(rColor);
    1347           0 :                         else if( (nIndex == 1) && xAnimate->getTo().hasValue() )
    1348           0 :                             xAnimate->setTo(rColor);
    1349             :                     }
    1350           0 :                     break;
    1351             : 
    1352             :                     }
    1353           0 :                 }
    1354           0 :             }
    1355           0 :         }
    1356             :     }
    1357           0 :     catch( Exception& )
    1358             :     {
    1359             :         OSL_FAIL("sd::CustomAnimationEffect::setColor(), exception caught!" );
    1360             :     }
    1361           0 : }
    1362             : 
    1363           0 : Any CustomAnimationEffect::getTransformationProperty( sal_Int32 nTransformType, EValue eValue )
    1364             : {
    1365           0 :     Any aProperty;
    1366           0 :     if( mxNode.is() ) try
    1367             :     {
    1368           0 :         Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY );
    1369           0 :         if( xEnumerationAccess.is() )
    1370             :         {
    1371           0 :             Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY );
    1372           0 :             if( xEnumeration.is() )
    1373             :             {
    1374           0 :                 while( xEnumeration->hasMoreElements() && !aProperty.hasValue() )
    1375             :                 {
    1376           0 :                     Reference< XAnimateTransform > xTransform( xEnumeration->nextElement(), UNO_QUERY );
    1377           0 :                     if( !xTransform.is() )
    1378           0 :                         continue;
    1379             : 
    1380           0 :                     if( xTransform->getTransformType() == nTransformType )
    1381             :                     {
    1382           0 :                         switch( eValue )
    1383             :                         {
    1384           0 :                         case VALUE_FROM: aProperty = xTransform->getFrom(); break;
    1385           0 :                         case VALUE_TO:   aProperty = xTransform->getTo(); break;
    1386           0 :                         case VALUE_BY:   aProperty = xTransform->getBy(); break;
    1387             :                         case VALUE_FIRST:
    1388             :                         case VALUE_LAST:
    1389             :                             {
    1390           0 :                                 Sequence<Any> aValues( xTransform->getValues() );
    1391           0 :                                 if( aValues.hasElements() )
    1392           0 :                                     aProperty =  aValues[ eValue == VALUE_FIRST ? 0 : aValues.getLength() - 1 ];
    1393             :                             }
    1394           0 :                             break;
    1395             :                         }
    1396             :                     }
    1397           0 :                 }
    1398           0 :             }
    1399           0 :         }
    1400             :     }
    1401           0 :     catch( Exception& )
    1402             :     {
    1403             :         OSL_FAIL("sd::CustomAnimationEffect::getTransformationProperty(), exception caught!" );
    1404             :     }
    1405             : 
    1406           0 :     return aProperty;
    1407             : }
    1408             : 
    1409           0 : bool CustomAnimationEffect::setTransformationProperty( sal_Int32 nTransformType, EValue eValue, const Any& rValue )
    1410             : {
    1411           0 :     bool bChanged = false;
    1412           0 :     if( mxNode.is() ) try
    1413             :     {
    1414           0 :         Reference< XEnumerationAccess > xEnumerationAccess( mxNode, UNO_QUERY );
    1415           0 :         if( xEnumerationAccess.is() )
    1416             :         {
    1417           0 :             Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY );
    1418           0 :             if( xEnumeration.is() )
    1419             :             {
    1420           0 :                 while( xEnumeration->hasMoreElements() )
    1421             :                 {
    1422           0 :                     Reference< XAnimateTransform > xTransform( xEnumeration->nextElement(), UNO_QUERY );
    1423           0 :                     if( !xTransform.is() )
    1424           0 :                         continue;
    1425             : 
    1426           0 :                     if( xTransform->getTransformType() == nTransformType )
    1427             :                     {
    1428           0 :                         switch( eValue )
    1429             :                         {
    1430             :                         case VALUE_FROM:
    1431           0 :                             if( xTransform->getFrom() != rValue )
    1432             :                             {
    1433           0 :                                 xTransform->setFrom( rValue );
    1434           0 :                                 bChanged = true;
    1435             :                             }
    1436           0 :                             break;
    1437             :                         case VALUE_TO:
    1438           0 :                             if( xTransform->getTo() != rValue )
    1439             :                             {
    1440           0 :                                 xTransform->setTo( rValue );
    1441           0 :                                 bChanged = true;
    1442             :                             }
    1443           0 :                             break;
    1444             :                         case VALUE_BY:
    1445           0 :                             if( xTransform->getBy() != rValue )
    1446             :                             {
    1447           0 :                                 xTransform->setBy( rValue );
    1448           0 :                                 bChanged = true;
    1449             :                             }
    1450           0 :                             break;
    1451             :                         case VALUE_FIRST:
    1452             :                         case VALUE_LAST:
    1453             :                             {
    1454           0 :                                 Sequence<Any> aValues( xTransform->getValues() );
    1455           0 :                                 if( !aValues.hasElements() )
    1456           0 :                                     aValues.realloc(1);
    1457             : 
    1458           0 :                                 sal_Int32 nIndex = eValue == VALUE_FIRST ? 0 : aValues.getLength() - 1;
    1459           0 :                                 if( aValues[nIndex] != rValue )
    1460             :                                 {
    1461           0 :                                     aValues[nIndex] = rValue;
    1462           0 :                                     xTransform->setValues( aValues );
    1463           0 :                                     bChanged = true;
    1464           0 :                                 }
    1465             :                             }
    1466             :                         }
    1467             :                     }
    1468           0 :                 }
    1469           0 :             }
    1470           0 :         }
    1471             :     }
    1472           0 :     catch( Exception& )
    1473             :     {
    1474             :         OSL_FAIL("sd::CustomAnimationEffect::setTransformationProperty(), exception caught!" );
    1475             :     }
    1476             : 
    1477           0 :     return bChanged;
    1478             : }
    1479             : 
    1480           0 : void CustomAnimationEffect::createAudio( const ::com::sun::star::uno::Any& rSource, double fVolume /* = 1.0 */ )
    1481             : {
    1482             :     DBG_ASSERT( !mxAudio.is(), "sd::CustomAnimationEffect::createAudio(), node already has an audio!" );
    1483             : 
    1484           0 :     if( !mxAudio.is() ) try
    1485             :     {
    1486           0 :         Reference< XComponentContext > xContext( ::comphelper::getProcessComponentContext() );
    1487           0 :         Reference< XAudio > xAudio( Audio::create( xContext ) );
    1488           0 :         xAudio->setSource( rSource );
    1489           0 :         xAudio->setVolume( fVolume );
    1490           0 :         setAudio( xAudio );
    1491             :     }
    1492           0 :     catch( Exception& )
    1493             :     {
    1494             :         OSL_FAIL("sd::CustomAnimationEffect::createAudio(), exception caught!" );
    1495             :     }
    1496           0 : }
    1497             : 
    1498           0 : static Reference< XCommand > findCommandNode( const Reference< XAnimationNode >& xRootNode )
    1499             : {
    1500           0 :     Reference< XCommand > xCommand;
    1501             : 
    1502           0 :     if( xRootNode.is() ) try
    1503             :     {
    1504           0 :         Reference< XEnumerationAccess > xEnumerationAccess( xRootNode, UNO_QUERY_THROW );
    1505           0 :         Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    1506           0 :         while( !xCommand.is() && xEnumeration->hasMoreElements() )
    1507             :         {
    1508           0 :             Reference< XAnimationNode > xNode( xEnumeration->nextElement(), UNO_QUERY );
    1509           0 :             if( xNode.is() && (xNode->getType() == AnimationNodeType::COMMAND) )
    1510           0 :                 xCommand.set( xNode, UNO_QUERY_THROW );
    1511           0 :         }
    1512             :     }
    1513           0 :     catch( Exception& )
    1514             :     {
    1515             :         OSL_FAIL("sd::findCommandNode(), exception caught!" );
    1516             :     }
    1517             : 
    1518           0 :     return xCommand;
    1519             : }
    1520             : 
    1521           0 : void CustomAnimationEffect::removeAudio()
    1522             : {
    1523             :     try
    1524             :     {
    1525           0 :         Reference< XAnimationNode > xChild;
    1526             : 
    1527           0 :         if( mxAudio.is() )
    1528             :         {
    1529           0 :             xChild.set( mxAudio, UNO_QUERY );
    1530           0 :             mxAudio.clear();
    1531             :         }
    1532           0 :         else if( mnCommand == EffectCommands::STOPAUDIO )
    1533             :         {
    1534           0 :             xChild.set( findCommandNode( mxNode ), UNO_QUERY );
    1535           0 :             mnCommand = 0;
    1536             :         }
    1537             : 
    1538           0 :         if( xChild.is() )
    1539             :         {
    1540           0 :             Reference< XTimeContainer > xContainer( mxNode, UNO_QUERY );
    1541           0 :             if( xContainer.is() )
    1542           0 :                 xContainer->removeChild( xChild );
    1543           0 :         }
    1544             :     }
    1545           0 :     catch( Exception& )
    1546             :     {
    1547             :         OSL_FAIL("sd::CustomAnimationEffect::removeAudio(), exception caught!" );
    1548             :     }
    1549             : 
    1550           0 : }
    1551             : 
    1552           0 : void CustomAnimationEffect::setAudio( const Reference< ::com::sun::star::animations::XAudio >& xAudio )
    1553             : {
    1554           0 :     if( mxAudio != xAudio ) try
    1555             :     {
    1556           0 :         removeAudio();
    1557           0 :         mxAudio = xAudio;
    1558           0 :         Reference< XTimeContainer > xContainer( mxNode, UNO_QUERY );
    1559           0 :         Reference< XAnimationNode > xChild( mxAudio, UNO_QUERY );
    1560           0 :         if( xContainer.is() && xChild.is() )
    1561           0 :             xContainer->appendChild( xChild );
    1562             :     }
    1563           0 :     catch( Exception& )
    1564             :     {
    1565             :         OSL_FAIL("sd::CustomAnimationEffect::setAudio(), exception caught!" );
    1566             :     }
    1567           0 : }
    1568             : 
    1569           0 : void CustomAnimationEffect::setStopAudio()
    1570             : {
    1571           0 :     if( mnCommand != EffectCommands::STOPAUDIO ) try
    1572             :     {
    1573           0 :         if( mxAudio.is() )
    1574           0 :             removeAudio();
    1575             : 
    1576           0 :         Reference< XComponentContext > xContext( ::comphelper::getProcessComponentContext() );
    1577           0 :         Reference< XCommand > xCommand( Command::create( xContext ) );
    1578             : 
    1579           0 :         xCommand->setCommand( EffectCommands::STOPAUDIO );
    1580             : 
    1581           0 :         Reference< XTimeContainer > xContainer( mxNode, UNO_QUERY_THROW );
    1582           0 :         xContainer->appendChild( xCommand );
    1583             : 
    1584           0 :         mnCommand = EffectCommands::STOPAUDIO;
    1585             :     }
    1586           0 :     catch( Exception& )
    1587             :     {
    1588             :         OSL_FAIL("sd::CustomAnimationEffect::setStopAudio(), exception caught!" );
    1589             :     }
    1590           0 : }
    1591             : 
    1592           0 : bool CustomAnimationEffect::getStopAudio() const
    1593             : {
    1594           0 :     return mnCommand == EffectCommands::STOPAUDIO;
    1595             : }
    1596             : 
    1597           0 : SdrPathObj* CustomAnimationEffect::createSdrPathObjFromPath()
    1598             : {
    1599           0 :     SdrPathObj * pPathObj = new SdrPathObj( OBJ_PATHLINE );
    1600           0 :     updateSdrPathObjFromPath( *pPathObj );
    1601           0 :     return pPathObj;
    1602             : }
    1603             : 
    1604           0 : void CustomAnimationEffect::updateSdrPathObjFromPath( SdrPathObj& rPathObj )
    1605             : {
    1606           0 :     ::basegfx::B2DPolyPolygon xPolyPoly;
    1607           0 :     if( ::basegfx::tools::importFromSvgD( xPolyPoly, getPath(), true, 0 ) )
    1608             :     {
    1609           0 :         SdrObject* pObj = GetSdrObjectFromXShape( getTargetShape() );
    1610           0 :         if( pObj )
    1611             :         {
    1612           0 :             SdrPage* pPage = pObj->GetPage();
    1613           0 :             if( pPage )
    1614             :             {
    1615           0 :                 const Size aPageSize( pPage->GetSize() );
    1616           0 :                 xPolyPoly.transform(basegfx::tools::createScaleB2DHomMatrix((double)aPageSize.Width(), (double)aPageSize.Height()));
    1617             :             }
    1618             : 
    1619           0 :             const Rectangle aBoundRect( pObj->GetCurrentBoundRect() );
    1620           0 :             const Point aCenter( aBoundRect.Center() );
    1621           0 :             xPolyPoly.transform(basegfx::tools::createTranslateB2DHomMatrix(aCenter.X(), aCenter.Y()));
    1622             :         }
    1623             :     }
    1624             : 
    1625           0 :     rPathObj.SetPathPoly( xPolyPoly );
    1626           0 : }
    1627             : 
    1628           0 : void CustomAnimationEffect::updatePathFromSdrPathObj( const SdrPathObj& rPathObj )
    1629             : {
    1630           0 :     ::basegfx::B2DPolyPolygon xPolyPoly( rPathObj.GetPathPoly() );
    1631             : 
    1632           0 :     SdrObject* pObj = GetSdrObjectFromXShape( getTargetShape() );
    1633           0 :     if( pObj )
    1634             :     {
    1635           0 :         Rectangle aBoundRect(0,0,0,0);
    1636             : 
    1637           0 :         const drawinglayer::primitive2d::Primitive2DSequence xPrimitives(pObj->GetViewContact().getViewIndependentPrimitive2DSequence());
    1638           0 :         const drawinglayer::geometry::ViewInformation2D aViewInformation2D;
    1639           0 :             const basegfx::B2DRange aRange(drawinglayer::primitive2d::getB2DRangeFromPrimitive2DSequence(xPrimitives, aViewInformation2D));
    1640             : 
    1641           0 :         if(!aRange.isEmpty())
    1642             :         {
    1643             :             aBoundRect = Rectangle(
    1644           0 :                     (sal_Int32)floor(aRange.getMinX()), (sal_Int32)floor(aRange.getMinY()),
    1645           0 :                     (sal_Int32)ceil(aRange.getMaxX()), (sal_Int32)ceil(aRange.getMaxY()));
    1646             :         }
    1647             : 
    1648           0 :         const Point aCenter( aBoundRect.Center() );
    1649             : 
    1650           0 :         xPolyPoly.transform(basegfx::tools::createTranslateB2DHomMatrix(-aCenter.X(), -aCenter.Y()));
    1651             : 
    1652           0 :         SdrPage* pPage = pObj->GetPage();
    1653           0 :         if( pPage )
    1654             :         {
    1655           0 :             const Size aPageSize( pPage->GetSize() );
    1656             :             xPolyPoly.transform(basegfx::tools::createScaleB2DHomMatrix(
    1657           0 :                 1.0 / (double)aPageSize.Width(), 1.0 / (double)aPageSize.Height()));
    1658           0 :         }
    1659             :     }
    1660             : 
    1661           0 :     setPath( ::basegfx::tools::exportToSvgD( xPolyPoly, true, true, true) );
    1662           0 : }
    1663             : 
    1664          93 : EffectSequenceHelper::EffectSequenceHelper()
    1665          93 : : mnSequenceType( EffectNodeType::DEFAULT )
    1666             : {
    1667          93 : }
    1668             : 
    1669           0 : EffectSequenceHelper::EffectSequenceHelper( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XTimeContainer >& xSequenceRoot )
    1670           0 : : mxSequenceRoot( xSequenceRoot ), mnSequenceType( EffectNodeType::DEFAULT )
    1671             : {
    1672           0 :     Reference< XAnimationNode > xNode( mxSequenceRoot, UNO_QUERY_THROW );
    1673           0 :     create( xNode );
    1674           0 : }
    1675             : 
    1676         178 : EffectSequenceHelper::~EffectSequenceHelper()
    1677             : {
    1678          89 :     reset();
    1679          89 : }
    1680             : 
    1681         178 : void EffectSequenceHelper::reset()
    1682             : {
    1683         178 :     EffectSequence::iterator aIter( maEffects.begin() );
    1684         178 :     EffectSequence::iterator aEnd( maEffects.end() );
    1685         178 :     if( aIter != aEnd )
    1686             :     {
    1687           0 :         CustomAnimationEffectPtr pEffect = (*aIter++);
    1688           0 :         pEffect->setEffectSequence(0);
    1689             :     }
    1690         178 :     maEffects.clear();
    1691         178 : }
    1692             : 
    1693           0 : Reference< XAnimationNode > EffectSequenceHelper::getRootNode()
    1694             : {
    1695           0 :     Reference< XAnimationNode > xRoot( mxSequenceRoot, UNO_QUERY );
    1696           0 :     return xRoot;
    1697             : }
    1698             : 
    1699           0 : void EffectSequenceHelper::append( const CustomAnimationEffectPtr& pEffect )
    1700             : {
    1701           0 :     pEffect->setEffectSequence( this );
    1702           0 :     maEffects.push_back(pEffect);
    1703           0 :     rebuild();
    1704           0 : }
    1705             : 
    1706           0 : CustomAnimationEffectPtr EffectSequenceHelper::append( const CustomAnimationPresetPtr& pPreset, const Any& rTarget, double fDuration /* = -1.0 */ )
    1707             : {
    1708           0 :     CustomAnimationEffectPtr pEffect;
    1709             : 
    1710           0 :     if( pPreset.get() )
    1711             :     {
    1712           0 :         OUString strEmpty;
    1713           0 :         Reference< XAnimationNode > xNode( pPreset->create( strEmpty ) );
    1714           0 :         if( xNode.is() )
    1715             :         {
    1716             :             // first, filter all only ui relevant user data
    1717           0 :             std::vector< NamedValue > aNewUserData;
    1718           0 :             Sequence< NamedValue > aUserData( xNode->getUserData() );
    1719           0 :             sal_Int32 nLength = aUserData.getLength();
    1720           0 :             const NamedValue* p = aUserData.getConstArray();
    1721           0 :             bool bFilter = false;
    1722             : 
    1723           0 :             while( nLength-- )
    1724             :             {
    1725           0 :                 if( p->Name != "text-only" && p->Name != "preset-property" )
    1726             :                 {
    1727           0 :                     aNewUserData.push_back( *p );
    1728           0 :                     bFilter = true;
    1729             :                 }
    1730           0 :                 p++;
    1731             :             }
    1732             : 
    1733           0 :             if( bFilter )
    1734             :             {
    1735           0 :                 aUserData = ::comphelper::containerToSequence< NamedValue, std::vector< NamedValue > >( aNewUserData );
    1736           0 :                 xNode->setUserData( aUserData );
    1737             :             }
    1738             : 
    1739             :             // check target, maybe we need to force it to text
    1740           0 :             Any aTarget( rTarget );
    1741           0 :             sal_Int16 nSubItem = ShapeAnimationSubType::AS_WHOLE;
    1742             : 
    1743           0 :             if( aTarget.getValueType() == ::cppu::UnoType<ParagraphTarget>::get() )
    1744             :             {
    1745           0 :                 nSubItem = ShapeAnimationSubType::ONLY_TEXT;
    1746             :             }
    1747           0 :             else if( pPreset->isTextOnly() )
    1748             :             {
    1749           0 :                 Reference< XShape > xShape;
    1750           0 :                 aTarget >>= xShape;
    1751           0 :                 if( xShape.is() )
    1752             :                 {
    1753             :                     // thats bad, we target a shape here but the effect is only for text
    1754             :                     // so change subitem
    1755           0 :                     nSubItem = ShapeAnimationSubType::ONLY_TEXT;
    1756           0 :                 }
    1757             :             }
    1758             : 
    1759             :             // now create effect from preset
    1760           0 :             pEffect.reset( new CustomAnimationEffect( xNode ) );
    1761           0 :             pEffect->setEffectSequence( this );
    1762           0 :             pEffect->setTarget( aTarget );
    1763           0 :             pEffect->setTargetSubItem( nSubItem );
    1764           0 :             if( fDuration != -1.0 )
    1765           0 :                 pEffect->setDuration( fDuration );
    1766             : 
    1767           0 :             maEffects.push_back(pEffect);
    1768             : 
    1769           0 :             rebuild();
    1770           0 :         }
    1771             :     }
    1772             : 
    1773             :     DBG_ASSERT( pEffect.get(), "sd::EffectSequenceHelper::append(), failed!" );
    1774           0 :     return pEffect;
    1775             : }
    1776             : 
    1777           0 : CustomAnimationEffectPtr EffectSequenceHelper::append( const SdrPathObj& rPathObj, const Any& rTarget, double fDuration /* = -1.0 */ )
    1778             : {
    1779           0 :     CustomAnimationEffectPtr pEffect;
    1780             : 
    1781           0 :     if( fDuration <= 0.0 )
    1782           0 :         fDuration = 2.0;
    1783             : 
    1784             :     try
    1785             :     {
    1786           0 :         Reference< XTimeContainer > xEffectContainer( ParallelTimeContainer::create( ::comphelper::getProcessComponentContext() ), UNO_QUERY_THROW );
    1787           0 :         Reference< XAnimationNode > xAnimateMotion( AnimateMotion::create( ::comphelper::getProcessComponentContext() ) );
    1788             : 
    1789           0 :         xAnimateMotion->setDuration( Any( fDuration ) );
    1790           0 :         xAnimateMotion->setFill( AnimationFill::HOLD );
    1791           0 :         xEffectContainer->appendChild( xAnimateMotion );
    1792             : 
    1793           0 :         sal_Int16 nSubItem = ShapeAnimationSubType::AS_WHOLE;
    1794             : 
    1795           0 :         if( rTarget.getValueType() == ::cppu::UnoType<ParagraphTarget>::get() )
    1796           0 :             nSubItem = ShapeAnimationSubType::ONLY_TEXT;
    1797             : 
    1798           0 :         pEffect.reset( new CustomAnimationEffect( xEffectContainer ) );
    1799           0 :         pEffect->setEffectSequence( this );
    1800           0 :         pEffect->setTarget( rTarget );
    1801           0 :         pEffect->setTargetSubItem( nSubItem );
    1802           0 :         pEffect->setNodeType( ::com::sun::star::presentation::EffectNodeType::ON_CLICK );
    1803           0 :         pEffect->setPresetClass( ::com::sun::star::presentation::EffectPresetClass::MOTIONPATH );
    1804           0 :         pEffect->setAcceleration( 0.5 );
    1805           0 :         pEffect->setDecelerate( 0.5 );
    1806           0 :         pEffect->setFill( AnimationFill::HOLD );
    1807           0 :         pEffect->setBegin( 0.0 );
    1808           0 :         pEffect->updatePathFromSdrPathObj( rPathObj );
    1809           0 :         if( fDuration != -1.0 )
    1810           0 :             pEffect->setDuration( fDuration );
    1811             : 
    1812           0 :         maEffects.push_back(pEffect);
    1813             : 
    1814           0 :         rebuild();
    1815             :     }
    1816           0 :     catch( Exception& )
    1817             :     {
    1818             :         OSL_FAIL( "sd::EffectSequenceHelper::append(), exception caught!" );
    1819             :     }
    1820             : 
    1821           0 :     return pEffect;
    1822             : }
    1823             : 
    1824           0 : void EffectSequenceHelper::replace( const CustomAnimationEffectPtr& pEffect, const CustomAnimationPresetPtr& pPreset, const OUString& rPresetSubType, double fDuration /* = -1.0 */ )
    1825             : {
    1826           0 :     if( pEffect.get() && pPreset.get() ) try
    1827             :     {
    1828           0 :         Reference< XAnimationNode > xNewNode( pPreset->create( rPresetSubType ) );
    1829           0 :         if( xNewNode.is() )
    1830             :         {
    1831           0 :             pEffect->replaceNode( xNewNode );
    1832           0 :             if( fDuration != -1.0 )
    1833           0 :                 pEffect->setDuration( fDuration );
    1834             :         }
    1835             : 
    1836           0 :         rebuild();
    1837             :     }
    1838           0 :     catch( Exception& )
    1839             :     {
    1840             :         OSL_FAIL( "sd::EffectSequenceHelper::replace(), exception caught!" );
    1841             :     }
    1842           0 : }
    1843             : 
    1844           0 : void EffectSequenceHelper::replace( const CustomAnimationEffectPtr& pEffect, const CustomAnimationPresetPtr& pPreset, double fDuration /* = -1.0 */ )
    1845             : {
    1846           0 :     OUString strEmpty;
    1847           0 :     replace( pEffect, pPreset, strEmpty, fDuration );
    1848           0 : }
    1849             : 
    1850           0 : void EffectSequenceHelper::remove( const CustomAnimationEffectPtr& pEffect )
    1851             : {
    1852           0 :     if( pEffect.get() )
    1853             :     {
    1854           0 :         pEffect->setEffectSequence( 0 );
    1855           0 :         maEffects.remove( pEffect );
    1856             :     }
    1857             : 
    1858           0 :     rebuild();
    1859           0 : }
    1860             : 
    1861           0 : void EffectSequenceHelper::rebuild()
    1862             : {
    1863           0 :     implRebuild();
    1864           0 : }
    1865             : 
    1866           0 : void EffectSequenceHelper::implRebuild()
    1867             : {
    1868             :     try
    1869             :     {
    1870             :         // first we delete all time containers on the first two levels
    1871           0 :         Reference< XEnumerationAccess > xEnumerationAccess( mxSequenceRoot, UNO_QUERY_THROW );
    1872           0 :         Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    1873           0 :         while( xEnumeration->hasMoreElements() )
    1874             :         {
    1875           0 :             Reference< XAnimationNode > xChildNode( xEnumeration->nextElement(), UNO_QUERY_THROW );
    1876           0 :             Reference< XTimeContainer > xChildContainer( xChildNode, UNO_QUERY_THROW );
    1877             : 
    1878           0 :             Reference< XEnumerationAccess > xChildEnumerationAccess( xChildNode, UNO_QUERY_THROW );
    1879           0 :             Reference< XEnumeration > xChildEnumeration( xChildEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    1880           0 :             while( xChildEnumeration->hasMoreElements() )
    1881             :             {
    1882           0 :                 Reference< XAnimationNode > xNode( xChildEnumeration->nextElement(), UNO_QUERY_THROW );
    1883           0 :                 xChildContainer->removeChild( xNode );
    1884           0 :             }
    1885             : 
    1886           0 :             mxSequenceRoot->removeChild( xChildNode );
    1887           0 :         }
    1888             : 
    1889             :         // second, rebuild main sequence
    1890           0 :         EffectSequence::iterator aIter( maEffects.begin() );
    1891           0 :         EffectSequence::iterator aEnd( maEffects.end() );
    1892           0 :         if( aIter != aEnd )
    1893             :         {
    1894           0 :             AfterEffectNodeList aAfterEffects;
    1895             : 
    1896           0 :             CustomAnimationEffectPtr pEffect = (*aIter++);
    1897             : 
    1898           0 :             bool bFirst = true;
    1899           0 :             do
    1900             :             {
    1901             :                 // create a par container for the next click node and all following with and after effects
    1902           0 :                 Reference< XTimeContainer > xOnClickContainer( ParallelTimeContainer::create( ::comphelper::getProcessComponentContext() ), UNO_QUERY_THROW );
    1903             : 
    1904           0 :                 Event aEvent;
    1905           0 :                 if( mxEventSource.is() )
    1906             :                 {
    1907           0 :                     aEvent.Source <<= mxEventSource;
    1908           0 :                     aEvent.Trigger = EventTrigger::ON_CLICK;
    1909             :                 }
    1910             :                 else
    1911             :                 {
    1912           0 :                     aEvent.Trigger = EventTrigger::ON_NEXT;
    1913             :                 }
    1914           0 :                 aEvent.Repeat = 0;
    1915             : 
    1916           0 :                 Any aBegin( makeAny( aEvent ) );
    1917           0 :                 if( bFirst )
    1918             :                 {
    1919             :                     // if the first node is not a click action, this click container
    1920             :                     // must not have INDEFINITE begin but start at 0s
    1921           0 :                     bFirst = false;
    1922           0 :                     if( pEffect->getNodeType() != EffectNodeType::ON_CLICK )
    1923           0 :                         aBegin <<= (double)0.0;
    1924             :                 }
    1925             : 
    1926           0 :                 xOnClickContainer->setBegin( aBegin );
    1927             : 
    1928           0 :                 mxSequenceRoot->appendChild( xOnClickContainer );
    1929             : 
    1930           0 :                 double fBegin = 0.0;
    1931             : 
    1932           0 :                 do
    1933             :                 {
    1934             :                     // create a par container for the current click or after effect node and all following with effects
    1935           0 :                     Reference< XTimeContainer > xWithContainer( ParallelTimeContainer::create( ::comphelper::getProcessComponentContext() ), UNO_QUERY_THROW );
    1936           0 :                     xWithContainer->setBegin( makeAny( fBegin ) );
    1937           0 :                     xOnClickContainer->appendChild( xWithContainer );
    1938             : 
    1939           0 :                     double fDuration = 0.0;
    1940           0 :                     do
    1941             :                     {
    1942           0 :                         Reference< XAnimationNode > xEffectNode( pEffect->getNode() );
    1943           0 :                         xWithContainer->appendChild( xEffectNode );
    1944             : 
    1945           0 :                         if( pEffect->hasAfterEffect() )
    1946             :                         {
    1947           0 :                             Reference< XAnimationNode > xAfterEffect( pEffect->createAfterEffectNode() );
    1948           0 :                             AfterEffectNode a( xAfterEffect, xEffectNode, pEffect->IsAfterEffectOnNext() );
    1949           0 :                             aAfterEffects.push_back( a );
    1950             :                         }
    1951             : 
    1952           0 :                         double fTemp = pEffect->getBegin() + pEffect->getAbsoluteDuration();
    1953           0 :                         if( fTemp > fDuration )
    1954           0 :                             fDuration = fTemp;
    1955             : 
    1956           0 :                         if( aIter != aEnd )
    1957           0 :                             pEffect = (*aIter++);
    1958             :                         else
    1959           0 :                             pEffect.reset();
    1960             :                     }
    1961           0 :                     while( pEffect.get() && (pEffect->getNodeType() == EffectNodeType::WITH_PREVIOUS) );
    1962             : 
    1963           0 :                     fBegin += fDuration;
    1964             :                 }
    1965           0 :                 while( pEffect.get() && (pEffect->getNodeType() != EffectNodeType::ON_CLICK) );
    1966             :             }
    1967           0 :             while( pEffect.get() );
    1968             : 
    1969             :             // process after effect nodes
    1970           0 :             std::for_each( aAfterEffects.begin(), aAfterEffects.end(), stl_process_after_effect_node_func );
    1971             : 
    1972           0 :             updateTextGroups();
    1973             : 
    1974             :             // reset duration, might have been altered (see below)
    1975           0 :             mxSequenceRoot->setDuration( Any() );
    1976             :         }
    1977             :         else
    1978             :         {
    1979             :             // empty sequence, set duration to 0.0 explicitly
    1980             :             // (otherwise, this sequence will never end)
    1981           0 :             mxSequenceRoot->setDuration( makeAny((double)0.0) );
    1982           0 :         }
    1983             :     }
    1984           0 :     catch( Exception& )
    1985             :     {
    1986             :         OSL_FAIL( "sd::EffectSequenceHelper::rebuild(), exception caught!" );
    1987             :     }
    1988           0 : }
    1989             : 
    1990           0 : stl_CustomAnimationEffect_search_node_predict::stl_CustomAnimationEffect_search_node_predict( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xSearchNode )
    1991           0 : : mxSearchNode( xSearchNode )
    1992             : {
    1993           0 : }
    1994             : 
    1995           0 : bool stl_CustomAnimationEffect_search_node_predict::operator()( CustomAnimationEffectPtr pEffect ) const
    1996             : {
    1997           0 :     return pEffect->getNode() == mxSearchNode;
    1998             : }
    1999             : 
    2000           0 : static bool implFindNextContainer( Reference< XTimeContainer >& xParent, Reference< XTimeContainer >& xCurrent, Reference< XTimeContainer >& xNext )
    2001             :  throw(Exception)
    2002             : {
    2003           0 :     Reference< XEnumerationAccess > xEnumerationAccess( xParent, UNO_QUERY_THROW );
    2004           0 :     Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration() );
    2005           0 :     if( xEnumeration.is() )
    2006             :     {
    2007           0 :         Reference< XInterface > x;
    2008           0 :         while( xEnumeration->hasMoreElements() && !xNext.is() )
    2009             :         {
    2010           0 :             if( (xEnumeration->nextElement() >>= x) && (x == xCurrent) )
    2011             :             {
    2012           0 :                 if( xEnumeration->hasMoreElements() )
    2013           0 :                     xEnumeration->nextElement() >>= xNext;
    2014             :             }
    2015           0 :         }
    2016             :     }
    2017           0 :     return xNext.is();
    2018             : }
    2019             : 
    2020           0 : void stl_process_after_effect_node_func(AfterEffectNode& rNode)
    2021             : {
    2022             :     try
    2023             :     {
    2024           0 :         if( rNode.mxNode.is() && rNode.mxMaster.is() )
    2025             :         {
    2026             :             // set master node
    2027           0 :             Reference< XAnimationNode > xMasterNode( rNode.mxMaster, UNO_QUERY_THROW );
    2028           0 :             Sequence< NamedValue > aUserData( rNode.mxNode->getUserData() );
    2029           0 :             sal_Int32 nSize = aUserData.getLength();
    2030           0 :             aUserData.realloc(nSize+1);
    2031           0 :             aUserData[nSize].Name = "master-element";
    2032           0 :             aUserData[nSize].Value <<= xMasterNode;
    2033           0 :             rNode.mxNode->setUserData( aUserData );
    2034             : 
    2035             :             // insert after effect node into timeline
    2036           0 :             Reference< XTimeContainer > xContainer( rNode.mxMaster->getParent(), UNO_QUERY_THROW );
    2037             : 
    2038           0 :             if( !rNode.mbOnNextEffect ) // sameClick
    2039             :             {
    2040             :                 // insert the aftereffect after its effect is animated
    2041           0 :                 xContainer->insertAfter( rNode.mxNode, rNode.mxMaster );
    2042             :             }
    2043             :             else // nextClick
    2044             :             {
    2045           0 :                 Reference< XComponentContext > xContext( ::comphelper::getProcessComponentContext() );
    2046             :                 // insert the aftereffect in the next group
    2047             : 
    2048           0 :                 Reference< XTimeContainer > xClickContainer( xContainer->getParent(), UNO_QUERY_THROW );
    2049           0 :                 Reference< XTimeContainer > xSequenceContainer( xClickContainer->getParent(), UNO_QUERY_THROW );
    2050             : 
    2051           0 :                 Reference< XTimeContainer > xNextContainer;
    2052             : 
    2053             :                 // first try if we have an after effect container
    2054           0 :                 if( !implFindNextContainer( xClickContainer, xContainer, xNextContainer ) )
    2055             :                 {
    2056           0 :                     Reference< XTimeContainer > xNextClickContainer;
    2057             :                     // if not, try to find the next click effect container
    2058           0 :                     if( implFindNextContainer( xSequenceContainer, xClickContainer, xNextClickContainer ) )
    2059             :                     {
    2060           0 :                         Reference< XEnumerationAccess > xEnumerationAccess( xNextClickContainer, UNO_QUERY_THROW );
    2061           0 :                         Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    2062           0 :                         if( xEnumeration->hasMoreElements() )
    2063             :                         {
    2064             :                             // the next container is the first child container
    2065           0 :                             xEnumeration->nextElement() >>= xNextContainer;
    2066             :                         }
    2067             :                         else
    2068             :                         {
    2069             :                             // this does not yet have a child container, create one
    2070           0 :                             xNextContainer.set( ParallelTimeContainer::create(xContext), UNO_QUERY_THROW );
    2071             : 
    2072           0 :                             xNextContainer->setBegin( makeAny( (double)0.0 ) );
    2073           0 :                             xNextClickContainer->appendChild( xNextContainer );
    2074             :                         }
    2075           0 :                         DBG_ASSERT( xNextContainer.is(), "ppt::stl_process_after_effect_node_func::operator(), could not find/create container!" );
    2076           0 :                     }
    2077             :                 }
    2078             : 
    2079             :                 // if we don't have a next container, we add one to the sequence container
    2080           0 :                 if( !xNextContainer.is() )
    2081             :                 {
    2082           0 :                     Reference< XTimeContainer > xNewClickContainer( ParallelTimeContainer::create( xContext ), UNO_QUERY_THROW );
    2083             : 
    2084           0 :                     Event aEvent;
    2085           0 :                     aEvent.Trigger = EventTrigger::ON_NEXT;
    2086           0 :                     aEvent.Repeat = 0;
    2087           0 :                     xNewClickContainer->setBegin( makeAny( aEvent ) );
    2088             : 
    2089           0 :                     xSequenceContainer->insertAfter( xNewClickContainer, xClickContainer );
    2090             : 
    2091           0 :                     xNextContainer.set( ParallelTimeContainer::create( xContext ), UNO_QUERY_THROW );
    2092             : 
    2093             :                     DBG_ASSERT( xNextContainer.is(), "ppt::stl_process_after_effect_node_func::operator(), could not create container!" );
    2094           0 :                     if( xNextContainer.is() )
    2095             :                     {
    2096           0 :                         xNextContainer->setBegin( makeAny( (double)0.0 ) );
    2097           0 :                         xNewClickContainer->appendChild( xNextContainer );
    2098           0 :                     }
    2099             :                 }
    2100             : 
    2101           0 :                 if( xNextContainer.is() )
    2102             :                 {
    2103             :                     // find begin time of first element
    2104           0 :                     Reference< XEnumerationAccess > xEnumerationAccess( xNextContainer, UNO_QUERY_THROW );
    2105           0 :                     Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    2106           0 :                     if( xEnumeration->hasMoreElements() )
    2107             :                     {
    2108           0 :                         Reference< XAnimationNode > xChild;
    2109             :                         // the next container is the first child container
    2110           0 :                         xEnumeration->nextElement() >>= xChild;
    2111           0 :                         if( xChild.is() )
    2112             :                         {
    2113           0 :                             Any aBegin( xChild->getBegin() );
    2114           0 :                             double fBegin = 0.0;
    2115           0 :                             if( (aBegin >>= fBegin) && (fBegin >= 0.0))
    2116           0 :                                 rNode.mxNode->setBegin( aBegin );
    2117           0 :                         }
    2118             :                     }
    2119             : 
    2120           0 :                     xNextContainer->appendChild( rNode.mxNode );
    2121           0 :                 }
    2122           0 :             }
    2123             :         }
    2124             :     }
    2125           0 :     catch( Exception& )
    2126             :     {
    2127             :         OSL_FAIL( "ppt::stl_process_after_effect_node_func::operator(), exception caught!" );
    2128             :     }
    2129           0 : }
    2130             : 
    2131           0 : EffectSequence::iterator EffectSequenceHelper::find( const CustomAnimationEffectPtr& pEffect )
    2132             : {
    2133           0 :     return std::find( maEffects.begin(), maEffects.end(), pEffect );
    2134             : }
    2135             : 
    2136           0 : CustomAnimationEffectPtr EffectSequenceHelper::findEffect( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xNode ) const
    2137             : {
    2138           0 :     CustomAnimationEffectPtr pEffect;
    2139             : 
    2140           0 :     EffectSequence::const_iterator aIter( maEffects.begin() );
    2141           0 :     for( ; aIter != maEffects.end(); ++aIter )
    2142             :     {
    2143           0 :         if( (*aIter)->getNode() == xNode )
    2144             :         {
    2145           0 :             pEffect = (*aIter);
    2146           0 :             break;
    2147             :         }
    2148             :     }
    2149             : 
    2150           0 :     return pEffect;
    2151             : }
    2152             : 
    2153           0 : sal_Int32 EffectSequenceHelper::getOffsetFromEffect( const CustomAnimationEffectPtr& xEffect ) const
    2154             : {
    2155           0 :     sal_Int32 nOffset = 0;
    2156             : 
    2157           0 :     EffectSequence::const_iterator aIter( maEffects.begin() );
    2158           0 :     for( ; aIter != maEffects.end(); ++aIter, nOffset++ )
    2159             :     {
    2160           0 :         if( (*aIter) == xEffect )
    2161           0 :             return nOffset;
    2162             :     }
    2163             : 
    2164           0 :     return -1;
    2165             : }
    2166             : 
    2167           0 : CustomAnimationEffectPtr EffectSequenceHelper::getEffectFromOffset( sal_Int32 nOffset ) const
    2168             : {
    2169           0 :     EffectSequence::const_iterator aIter( maEffects.begin() );
    2170           0 :     while( nOffset-- && aIter != maEffects.end() )
    2171           0 :         ++aIter;
    2172             : 
    2173           0 :     CustomAnimationEffectPtr pEffect;
    2174           0 :     if( aIter != maEffects.end() )
    2175           0 :         pEffect = (*aIter);
    2176             : 
    2177           0 :     return pEffect;
    2178             : }
    2179             : 
    2180           0 : bool EffectSequenceHelper::disposeShape( const Reference< XShape >& xShape )
    2181             : {
    2182           0 :     bool bChanges = false;
    2183             : 
    2184           0 :     EffectSequence::iterator aIter( maEffects.begin() );
    2185           0 :     while( aIter != maEffects.end() )
    2186             :     {
    2187           0 :         if( (*aIter)->getTargetShape() == xShape )
    2188             :         {
    2189           0 :             (*aIter)->setEffectSequence( 0 );
    2190           0 :             bChanges = true;
    2191           0 :             aIter = maEffects.erase( aIter );
    2192             :         }
    2193             :         else
    2194             :         {
    2195           0 :             ++aIter;
    2196             :         }
    2197             :     }
    2198             : 
    2199           0 :     return bChanges;
    2200             : }
    2201             : 
    2202         168 : bool EffectSequenceHelper::hasEffect( const com::sun::star::uno::Reference< com::sun::star::drawing::XShape >& xShape )
    2203             : {
    2204         168 :     EffectSequence::iterator aIter( maEffects.begin() );
    2205         336 :     while( aIter != maEffects.end() )
    2206             :     {
    2207           0 :         if( (*aIter)->getTargetShape() == xShape )
    2208           0 :             return true;
    2209           0 :         ++aIter;
    2210             :     }
    2211             : 
    2212         168 :     return false;
    2213             : }
    2214             : 
    2215           0 : void EffectSequenceHelper::insertTextRange( const com::sun::star::uno::Any& aTarget )
    2216             : {
    2217           0 :     bool bChanges = false;
    2218             : 
    2219           0 :     ParagraphTarget aParaTarget;
    2220           0 :     if( !(aTarget >>= aParaTarget ) )
    2221           0 :         return;
    2222             : 
    2223           0 :     EffectSequence::iterator aIter( maEffects.begin() );
    2224           0 :     while( aIter != maEffects.end() )
    2225             :     {
    2226           0 :         if( (*aIter)->getTargetShape() == aParaTarget.Shape )
    2227           0 :             bChanges |= (*aIter)->checkForText();
    2228           0 :         ++aIter;
    2229             :     }
    2230             : 
    2231           0 :     if( bChanges )
    2232           0 :         rebuild();
    2233             : }
    2234             : 
    2235           0 : void EffectSequenceHelper::disposeTextRange( const com::sun::star::uno::Any& aTarget )
    2236             : {
    2237           0 :     ParagraphTarget aParaTarget;
    2238           0 :     if( !(aTarget >>= aParaTarget ) )
    2239           0 :         return;
    2240             : 
    2241           0 :     bool bChanges = false;
    2242           0 :     bool bErased = false;
    2243             : 
    2244           0 :     EffectSequence::iterator aIter( maEffects.begin() );
    2245           0 :     while( aIter != maEffects.end() )
    2246             :     {
    2247           0 :         Any aIterTarget( (*aIter)->getTarget() );
    2248           0 :         if( aIterTarget.getValueType() == ::cppu::UnoType<ParagraphTarget>::get() )
    2249             :         {
    2250           0 :             ParagraphTarget aIterParaTarget;
    2251           0 :             if( (aIterTarget >>= aIterParaTarget) && (aIterParaTarget.Shape == aParaTarget.Shape) )
    2252             :             {
    2253           0 :                 if( aIterParaTarget.Paragraph == aParaTarget.Paragraph )
    2254             :                 {
    2255             :                     // delete this effect if it targets the disposed paragraph directly
    2256           0 :                     (*aIter)->setEffectSequence( 0 );
    2257           0 :                     aIter = maEffects.erase( aIter );
    2258           0 :                     bChanges = true;
    2259           0 :                     bErased = true;
    2260             :                 }
    2261             :                 else
    2262             :                 {
    2263           0 :                     if( aIterParaTarget.Paragraph > aParaTarget.Paragraph )
    2264             :                     {
    2265             :                         // shift all paragraphs after disposed paragraph
    2266           0 :                         aIterParaTarget.Paragraph--;
    2267           0 :                         (*aIter)->setTarget( makeAny( aIterParaTarget ) );
    2268             :                     }
    2269             :                 }
    2270           0 :             }
    2271             :         }
    2272           0 :         else if( (*aIter)->getTargetShape() == aParaTarget.Shape )
    2273             :         {
    2274           0 :             bChanges |= (*aIter)->checkForText();
    2275             :         }
    2276             : 
    2277           0 :         if( bErased )
    2278           0 :             bErased = false;
    2279             :         else
    2280           0 :             ++aIter;
    2281           0 :     }
    2282             : 
    2283           0 :     if( bChanges )
    2284           0 :         rebuild();
    2285             : }
    2286             : 
    2287           0 : CustomAnimationTextGroup::CustomAnimationTextGroup( const Reference< XShape >& rTarget, sal_Int32 nGroupId )
    2288             : :   maTarget( rTarget ),
    2289           0 :     mnGroupId( nGroupId )
    2290             : {
    2291           0 :     reset();
    2292           0 : }
    2293             : 
    2294           0 : void CustomAnimationTextGroup::reset()
    2295             : {
    2296           0 :     mnTextGrouping = -1;
    2297           0 :     mbAnimateForm = false;
    2298           0 :     mbTextReverse = false;
    2299           0 :     mfGroupingAuto = -1.0;
    2300           0 :     mnLastPara = -1; // used to check for TextReverse
    2301             : 
    2302           0 :     for (int i = 0; i < PARA_LEVELS; ++i)
    2303             :     {
    2304           0 :         mnDepthFlags[i] = 0;
    2305             :     }
    2306             : 
    2307           0 :     maEffects.clear();
    2308           0 : }
    2309             : 
    2310           0 : void CustomAnimationTextGroup::addEffect( CustomAnimationEffectPtr& pEffect )
    2311             : {
    2312           0 :     maEffects.push_back( pEffect );
    2313             : 
    2314           0 :     Any aTarget( pEffect->getTarget() );
    2315           0 :     if( aTarget.getValueType() == ::cppu::UnoType<ParagraphTarget>::get() )
    2316             :     {
    2317             :         // now look at the paragraph
    2318           0 :         ParagraphTarget aParaTarget;
    2319           0 :         aTarget >>= aParaTarget;
    2320             : 
    2321           0 :         if( mnLastPara != -1 )
    2322           0 :             mbTextReverse = mnLastPara > aParaTarget.Paragraph;
    2323             : 
    2324           0 :         mnLastPara = aParaTarget.Paragraph;
    2325             : 
    2326           0 :         const sal_Int32 nParaDepth = pEffect->getParaDepth();
    2327             : 
    2328             :         // only look at the first PARA_LEVELS levels
    2329           0 :         if( nParaDepth < PARA_LEVELS )
    2330             :         {
    2331             :             // our first paragraph with this level?
    2332           0 :             if( mnDepthFlags[nParaDepth] == 0 )
    2333             :             {
    2334             :                 // so set it to the first found
    2335           0 :                 mnDepthFlags[nParaDepth] = (sal_Int8)pEffect->getNodeType();
    2336             :             }
    2337           0 :             else if( mnDepthFlags[nParaDepth] != pEffect->getNodeType() )
    2338             :             {
    2339           0 :                 mnDepthFlags[nParaDepth] = -1;
    2340             :             }
    2341             : 
    2342           0 :             if( pEffect->getNodeType() == EffectNodeType::AFTER_PREVIOUS )
    2343           0 :                 mfGroupingAuto = pEffect->getBegin();
    2344             : 
    2345           0 :             mnTextGrouping = PARA_LEVELS;
    2346           0 :             while( (mnTextGrouping > 0)
    2347           0 :                    && (mnDepthFlags[mnTextGrouping - 1] <= 0) )
    2348           0 :                 --mnTextGrouping;
    2349           0 :         }
    2350             :     }
    2351             :     else
    2352             :     {
    2353             :         // if we have an effect with the shape as a target, we animate the background
    2354           0 :         mbAnimateForm = pEffect->getTargetSubItem() != ShapeAnimationSubType::ONLY_TEXT;
    2355           0 :     }
    2356           0 : }
    2357             : 
    2358           0 : CustomAnimationTextGroupPtr EffectSequenceHelper::findGroup( sal_Int32 nGroupId )
    2359             : {
    2360           0 :     CustomAnimationTextGroupPtr aPtr;
    2361             : 
    2362           0 :     CustomAnimationTextGroupMap::iterator aIter( maGroupMap.find( nGroupId ) );
    2363           0 :     if( aIter != maGroupMap.end() )
    2364           0 :         aPtr = (*aIter).second;
    2365             : 
    2366           0 :     return aPtr;
    2367             : }
    2368             : 
    2369          93 : void EffectSequenceHelper::updateTextGroups()
    2370             : {
    2371          93 :     maGroupMap.clear();
    2372             : 
    2373             :     // first create all the groups
    2374          93 :     EffectSequence::iterator aIter( maEffects.begin() );
    2375          93 :     const EffectSequence::iterator aEnd( maEffects.end() );
    2376         186 :     while( aIter != aEnd )
    2377             :     {
    2378           0 :         CustomAnimationEffectPtr pEffect( (*aIter++) );
    2379             : 
    2380           0 :         const sal_Int32 nGroupId = pEffect->getGroupId();
    2381             : 
    2382           0 :         if( nGroupId == -1 )
    2383           0 :             continue; // trivial case, no group
    2384             : 
    2385           0 :         CustomAnimationTextGroupPtr pGroup = findGroup( nGroupId );
    2386           0 :         if( !pGroup.get() )
    2387             :         {
    2388           0 :             pGroup.reset( new CustomAnimationTextGroup( pEffect->getTargetShape(), nGroupId ) );
    2389           0 :             maGroupMap[nGroupId] = pGroup;
    2390             :         }
    2391             : 
    2392           0 :         pGroup->addEffect( pEffect );
    2393           0 :     }
    2394          93 : }
    2395             : 
    2396           0 : CustomAnimationTextGroupPtr EffectSequenceHelper::createTextGroup( CustomAnimationEffectPtr pEffect, sal_Int32 nTextGrouping, double fTextGroupingAuto, bool bAnimateForm, bool bTextReverse )
    2397             : {
    2398             :     // first finde a free group-id
    2399           0 :     sal_Int32 nGroupId = 0;
    2400             : 
    2401           0 :     CustomAnimationTextGroupMap::iterator aIter( maGroupMap.begin() );
    2402           0 :     const CustomAnimationTextGroupMap::iterator aEnd( maGroupMap.end() );
    2403           0 :     while( aIter != aEnd )
    2404             :     {
    2405           0 :         if( (*aIter).first == nGroupId )
    2406             :         {
    2407           0 :             nGroupId++;
    2408           0 :             aIter = maGroupMap.begin();
    2409             :         }
    2410             :         else
    2411             :         {
    2412           0 :             ++aIter;
    2413             :         }
    2414             :     }
    2415             : 
    2416           0 :     Reference< XShape > xTarget( pEffect->getTargetShape() );
    2417             : 
    2418           0 :     CustomAnimationTextGroupPtr pTextGroup( new CustomAnimationTextGroup( xTarget, nGroupId ) );
    2419           0 :     maGroupMap[nGroupId] = pTextGroup;
    2420             : 
    2421           0 :     bool bUsed = false;
    2422             : 
    2423             :     // do we need to target the shape?
    2424           0 :     if( (nTextGrouping == 0) || bAnimateForm )
    2425             :     {
    2426             :         sal_Int16 nSubItem;
    2427           0 :         if( nTextGrouping == 0)
    2428           0 :             nSubItem = bAnimateForm ? ShapeAnimationSubType::AS_WHOLE : ShapeAnimationSubType::ONLY_TEXT;
    2429             :         else
    2430           0 :             nSubItem = ShapeAnimationSubType::ONLY_BACKGROUND;
    2431             : 
    2432           0 :         pEffect->setTarget( makeAny( xTarget ) );
    2433           0 :         pEffect->setTargetSubItem( nSubItem );
    2434           0 :         pEffect->setEffectSequence( this );
    2435           0 :         pEffect->setGroupId( nGroupId );
    2436             : 
    2437           0 :         pTextGroup->addEffect( pEffect );
    2438           0 :         bUsed = true;
    2439             :     }
    2440             : 
    2441           0 :     pTextGroup->mnTextGrouping = nTextGrouping;
    2442           0 :     pTextGroup->mfGroupingAuto = fTextGroupingAuto;
    2443           0 :     pTextGroup->mbTextReverse = bTextReverse;
    2444             : 
    2445             :     // now add an effect for each paragraph
    2446           0 :     createTextGroupParagraphEffects( pTextGroup, pEffect, bUsed );
    2447             : 
    2448           0 :     notify_listeners();
    2449             : 
    2450           0 :     return pTextGroup;
    2451             : }
    2452             : 
    2453           0 : void EffectSequenceHelper::createTextGroupParagraphEffects( CustomAnimationTextGroupPtr pTextGroup, CustomAnimationEffectPtr pEffect, bool bUsed )
    2454             : {
    2455           0 :     Reference< XShape > xTarget( pTextGroup->maTarget );
    2456             : 
    2457           0 :     sal_Int32 nTextGrouping = pTextGroup->mnTextGrouping;
    2458           0 :     double fTextGroupingAuto = pTextGroup->mfGroupingAuto;
    2459           0 :     bool bTextReverse = pTextGroup->mbTextReverse;
    2460             : 
    2461             :     // now add an effect for each paragraph
    2462           0 :     if( nTextGrouping >= 0 ) try
    2463             :     {
    2464           0 :         EffectSequence::iterator aInsertIter( find( pEffect ) );
    2465             : 
    2466           0 :         Reference< XEnumerationAccess > xText( xTarget, UNO_QUERY_THROW );
    2467           0 :         Reference< XEnumeration > xEnumeration( xText->createEnumeration(), UNO_QUERY_THROW );
    2468             : 
    2469           0 :         std::list< sal_Int16 > aParaList;
    2470             :         sal_Int16 nPara;
    2471             : 
    2472             :         // fill the list with all valid paragraphs
    2473           0 :         for( nPara = 0; xEnumeration->hasMoreElements(); nPara++ )
    2474             :         {
    2475           0 :             Reference< XTextRange > xRange( xEnumeration->nextElement(), UNO_QUERY );
    2476           0 :             if( xRange.is() && !xRange->getString().isEmpty() )
    2477             :             {
    2478           0 :                 if( bTextReverse ) // sort them
    2479           0 :                     aParaList.push_front( nPara );
    2480             :                 else
    2481           0 :                     aParaList.push_back( nPara );
    2482             :             }
    2483           0 :         }
    2484             : 
    2485           0 :         ParagraphTarget aTarget;
    2486           0 :         aTarget.Shape = xTarget;
    2487             : 
    2488           0 :         std::list< sal_Int16 >::iterator aIter( aParaList.begin() );
    2489           0 :         std::list< sal_Int16 >::iterator aEnd( aParaList.end() );
    2490           0 :         while( aIter != aEnd )
    2491             :         {
    2492           0 :             aTarget.Paragraph = (*aIter++);
    2493             : 
    2494           0 :             CustomAnimationEffectPtr pNewEffect;
    2495           0 :             if( bUsed )
    2496             :             {
    2497             :                 // clone a new effect from first effect
    2498           0 :                 pNewEffect = pEffect->clone();
    2499           0 :                 ++aInsertIter;
    2500           0 :                 aInsertIter = maEffects.insert( aInsertIter, pNewEffect );
    2501             :             }
    2502             :             else
    2503             :             {
    2504             :                 // reuse first effect if its not yet used
    2505           0 :                 pNewEffect = pEffect;
    2506           0 :                 bUsed = true;
    2507           0 :                 aInsertIter = find( pNewEffect );
    2508             :             }
    2509             : 
    2510             :             // set target and group-id
    2511           0 :             pNewEffect->setTarget( makeAny( aTarget ) );
    2512           0 :             pNewEffect->setTargetSubItem( ShapeAnimationSubType::ONLY_TEXT );
    2513           0 :             pNewEffect->setGroupId( pTextGroup->mnGroupId );
    2514           0 :             pNewEffect->setEffectSequence( this );
    2515             : 
    2516             :             // set correct node type
    2517           0 :             if( pNewEffect->getParaDepth() < nTextGrouping )
    2518             :             {
    2519           0 :                 if( fTextGroupingAuto == -1.0 )
    2520             :                 {
    2521           0 :                     pNewEffect->setNodeType( EffectNodeType::ON_CLICK );
    2522           0 :                     pNewEffect->setBegin( 0.0 );
    2523             :                 }
    2524             :                 else
    2525             :                 {
    2526           0 :                     pNewEffect->setNodeType( EffectNodeType::AFTER_PREVIOUS );
    2527           0 :                     pNewEffect->setBegin( fTextGroupingAuto );
    2528             :                 }
    2529             :             }
    2530             :             else
    2531             :             {
    2532           0 :                 pNewEffect->setNodeType( EffectNodeType::WITH_PREVIOUS );
    2533           0 :                 pNewEffect->setBegin( 0.0 );
    2534             :             }
    2535             : 
    2536           0 :             pTextGroup->addEffect( pNewEffect );
    2537           0 :         }
    2538           0 :         notify_listeners();
    2539             :     }
    2540           0 :     catch( Exception& )
    2541             :     {
    2542             :         OSL_FAIL("sd::EffectSequenceHelper::createTextGroup(), exception caught!" );
    2543           0 :     }
    2544           0 : }
    2545             : 
    2546           0 : void EffectSequenceHelper::setTextGrouping( CustomAnimationTextGroupPtr pTextGroup, sal_Int32 nTextGrouping )
    2547             : {
    2548           0 :     if( pTextGroup->mnTextGrouping == nTextGrouping )
    2549             :     {
    2550             :         // first case, trivial case, do nothing
    2551             :     }
    2552           0 :     else if( (pTextGroup->mnTextGrouping == -1) && (nTextGrouping >= 0) )
    2553             :     {
    2554             :         // second case, we need to add new effects for each paragraph
    2555             : 
    2556           0 :         CustomAnimationEffectPtr pEffect( pTextGroup->maEffects.front() );
    2557             : 
    2558           0 :         pTextGroup->mnTextGrouping = nTextGrouping;
    2559           0 :         createTextGroupParagraphEffects( pTextGroup, pEffect, true );
    2560           0 :         notify_listeners();
    2561             :     }
    2562           0 :     else if( (pTextGroup->mnTextGrouping >= 0) && (nTextGrouping == -1 ) )
    2563             :     {
    2564             :         // third case, we need to remove effects for each paragraph
    2565             : 
    2566           0 :         EffectSequence aEffects( pTextGroup->maEffects );
    2567           0 :         pTextGroup->reset();
    2568             : 
    2569           0 :         EffectSequence::iterator aIter( aEffects.begin() );
    2570           0 :         const EffectSequence::iterator aEnd( aEffects.end() );
    2571           0 :         while( aIter != aEnd )
    2572             :         {
    2573           0 :             CustomAnimationEffectPtr pEffect( (*aIter++) );
    2574             : 
    2575           0 :             if( pEffect->getTarget().getValueType() == ::cppu::UnoType<ParagraphTarget>::get() )
    2576           0 :                 remove( pEffect );
    2577             :             else
    2578           0 :                 pTextGroup->addEffect( pEffect );
    2579           0 :         }
    2580           0 :         notify_listeners();
    2581             :     }
    2582             :     else
    2583             :     {
    2584             :         // fourth case, we need to change the node types for the text nodes
    2585           0 :         double fTextGroupingAuto = pTextGroup->mfGroupingAuto;
    2586             : 
    2587           0 :         EffectSequence aEffects( pTextGroup->maEffects );
    2588           0 :         pTextGroup->reset();
    2589             : 
    2590           0 :         EffectSequence::iterator aIter( aEffects.begin() );
    2591           0 :         const EffectSequence::iterator aEnd( aEffects.end() );
    2592           0 :         while( aIter != aEnd )
    2593             :         {
    2594           0 :             CustomAnimationEffectPtr pEffect( (*aIter++) );
    2595             : 
    2596           0 :             if( pEffect->getTarget().getValueType() == ::cppu::UnoType<ParagraphTarget>::get() )
    2597             :             {
    2598             :                 // set correct node type
    2599           0 :                 if( pEffect->getParaDepth() < nTextGrouping )
    2600             :                 {
    2601           0 :                     if( fTextGroupingAuto == -1.0 )
    2602             :                     {
    2603           0 :                         pEffect->setNodeType( EffectNodeType::ON_CLICK );
    2604           0 :                         pEffect->setBegin( 0.0 );
    2605             :                     }
    2606             :                     else
    2607             :                     {
    2608           0 :                         pEffect->setNodeType( EffectNodeType::AFTER_PREVIOUS );
    2609           0 :                         pEffect->setBegin( fTextGroupingAuto );
    2610             :                     }
    2611             :                 }
    2612             :                 else
    2613             :                 {
    2614           0 :                     pEffect->setNodeType( EffectNodeType::WITH_PREVIOUS );
    2615           0 :                     pEffect->setBegin( 0.0 );
    2616             :                 }
    2617             :             }
    2618             : 
    2619           0 :             pTextGroup->addEffect( pEffect );
    2620             : 
    2621           0 :         }
    2622           0 :         notify_listeners();
    2623             :     }
    2624           0 : }
    2625             : 
    2626           0 : void EffectSequenceHelper::setAnimateForm( CustomAnimationTextGroupPtr pTextGroup, bool bAnimateForm )
    2627             : {
    2628           0 :     if( pTextGroup->mbAnimateForm == bAnimateForm )
    2629             :     {
    2630             :         // trivial case, do nothing
    2631             :     }
    2632             :     else
    2633             :     {
    2634           0 :         EffectSequence aEffects( pTextGroup->maEffects );
    2635           0 :         pTextGroup->reset();
    2636             : 
    2637             :         SAL_WARN_IF(aEffects.empty(), "sd", "EffectSequenceHelper::setAnimateForm effects empty" );
    2638             : 
    2639           0 :         if (aEffects.empty())
    2640           0 :             return;
    2641             : 
    2642           0 :         EffectSequence::iterator aIter( aEffects.begin() );
    2643           0 :         const EffectSequence::iterator aEnd( aEffects.end() );
    2644             : 
    2645             :         // first insert if we have to
    2646           0 :         if( bAnimateForm )
    2647             :         {
    2648           0 :             EffectSequence::iterator aInsertIter( find( (*aIter) ) );
    2649             : 
    2650           0 :             CustomAnimationEffectPtr pEffect;
    2651           0 :             if( (aEffects.size() == 1) && ((*aIter)->getTarget().getValueType() != ::cppu::UnoType<ParagraphTarget>::get() ) )
    2652             :             {
    2653             :                 // special case, only one effect and that targets whole text,
    2654             :                 // convert this to target whole shape
    2655           0 :                 pEffect = (*aIter++);
    2656           0 :                 pEffect->setTargetSubItem( ShapeAnimationSubType::AS_WHOLE );
    2657             :             }
    2658             :             else
    2659             :             {
    2660           0 :                 pEffect = (*aIter)->clone();
    2661           0 :                 pEffect->setTarget( makeAny( (*aIter)->getTargetShape() ) );
    2662           0 :                 pEffect->setTargetSubItem( ShapeAnimationSubType::ONLY_BACKGROUND );
    2663           0 :                 maEffects.insert( aInsertIter, pEffect );
    2664             :             }
    2665             : 
    2666           0 :             pTextGroup->addEffect( pEffect );
    2667             :         }
    2668             : 
    2669           0 :         if( !bAnimateForm && (aEffects.size() == 1) )
    2670             :         {
    2671           0 :             CustomAnimationEffectPtr pEffect( (*aIter) );
    2672           0 :             pEffect->setTarget( makeAny( (*aIter)->getTargetShape() ) );
    2673           0 :             pEffect->setTargetSubItem( ShapeAnimationSubType::ONLY_TEXT );
    2674           0 :             pTextGroup->addEffect( pEffect );
    2675             :         }
    2676             :         else
    2677             :         {
    2678             :             // read the rest to the group again
    2679           0 :             while( aIter != aEnd )
    2680             :             {
    2681           0 :                 CustomAnimationEffectPtr pEffect( (*aIter++) );
    2682             : 
    2683           0 :                 if( pEffect->getTarget().getValueType() == ::cppu::UnoType<ParagraphTarget>::get() )
    2684             :                 {
    2685           0 :                     pTextGroup->addEffect( pEffect );
    2686             :                 }
    2687             :                 else
    2688             :                 {
    2689             :                     DBG_ASSERT( !bAnimateForm, "sd::EffectSequenceHelper::setAnimateForm(), something is wrong here!" );
    2690           0 :                     remove( pEffect );
    2691             :                 }
    2692           0 :             }
    2693             :         }
    2694           0 :         notify_listeners();
    2695             :     }
    2696             : }
    2697             : 
    2698           0 : void EffectSequenceHelper::setTextGroupingAuto( CustomAnimationTextGroupPtr pTextGroup, double fTextGroupingAuto )
    2699             : {
    2700           0 :     sal_Int32 nTextGrouping = pTextGroup->mnTextGrouping;
    2701             : 
    2702           0 :     EffectSequence aEffects( pTextGroup->maEffects );
    2703           0 :     pTextGroup->reset();
    2704             : 
    2705           0 :     EffectSequence::iterator aIter( aEffects.begin() );
    2706           0 :     const EffectSequence::iterator aEnd( aEffects.end() );
    2707           0 :     while( aIter != aEnd )
    2708             :     {
    2709           0 :         CustomAnimationEffectPtr pEffect( (*aIter++) );
    2710             : 
    2711           0 :         if( pEffect->getTarget().getValueType() == ::cppu::UnoType<ParagraphTarget>::get() )
    2712             :         {
    2713             :             // set correct node type
    2714           0 :             if( pEffect->getParaDepth() < nTextGrouping )
    2715             :             {
    2716           0 :                 if( fTextGroupingAuto == -1.0 )
    2717             :                 {
    2718           0 :                     pEffect->setNodeType( EffectNodeType::ON_CLICK );
    2719           0 :                     pEffect->setBegin( 0.0 );
    2720             :                 }
    2721             :                 else
    2722             :                 {
    2723           0 :                     pEffect->setNodeType( EffectNodeType::AFTER_PREVIOUS );
    2724           0 :                     pEffect->setBegin( fTextGroupingAuto );
    2725             :                 }
    2726             :             }
    2727             :             else
    2728             :             {
    2729           0 :                 pEffect->setNodeType( EffectNodeType::WITH_PREVIOUS );
    2730           0 :                 pEffect->setBegin( 0.0 );
    2731             :             }
    2732             :         }
    2733             : 
    2734           0 :         pTextGroup->addEffect( pEffect );
    2735             : 
    2736           0 :     }
    2737           0 :     notify_listeners();
    2738           0 : }
    2739             : 
    2740             : struct ImplStlTextGroupSortHelper
    2741             : {
    2742           0 :     ImplStlTextGroupSortHelper( bool bReverse ) : mbReverse( bReverse ) {};
    2743             :     bool operator()( const CustomAnimationEffectPtr& p1, const CustomAnimationEffectPtr& p2 );
    2744             :     bool mbReverse;
    2745             :     sal_Int32 getTargetParagraph( const CustomAnimationEffectPtr& p1 );
    2746             : };
    2747             : 
    2748           0 : sal_Int32 ImplStlTextGroupSortHelper::getTargetParagraph( const CustomAnimationEffectPtr& p1 )
    2749             : {
    2750           0 :     const Any aTarget(p1->getTarget());
    2751           0 :     if( aTarget.hasValue() && aTarget.getValueType() == ::cppu::UnoType<ParagraphTarget>::get() )
    2752             :     {
    2753           0 :         ParagraphTarget aParaTarget;
    2754           0 :         aTarget >>= aParaTarget;
    2755           0 :         return aParaTarget.Paragraph;
    2756             :     }
    2757             :     else
    2758             :     {
    2759           0 :         return mbReverse ? 0x7fffffff : -1;
    2760           0 :     }
    2761             : }
    2762             : 
    2763           0 : bool ImplStlTextGroupSortHelper::operator()( const CustomAnimationEffectPtr& p1, const CustomAnimationEffectPtr& p2 )
    2764             : {
    2765           0 :     if( mbReverse )
    2766             :     {
    2767           0 :         return getTargetParagraph( p2 ) < getTargetParagraph( p1 );
    2768             :     }
    2769             :     else
    2770             :     {
    2771           0 :         return getTargetParagraph( p1 ) < getTargetParagraph( p2 );
    2772             :     }
    2773             : }
    2774             : 
    2775           0 : void EffectSequenceHelper::setTextReverse( CustomAnimationTextGroupPtr pTextGroup, bool bTextReverse )
    2776             : {
    2777           0 :     if( pTextGroup->mbTextReverse == bTextReverse )
    2778             :     {
    2779             :         // do nothing
    2780             :     }
    2781             :     else
    2782             :     {
    2783           0 :         std::vector< CustomAnimationEffectPtr > aSortedVector(pTextGroup->maEffects.size());
    2784           0 :         std::copy( pTextGroup->maEffects.begin(), pTextGroup->maEffects.end(), aSortedVector.begin() );
    2785           0 :         ImplStlTextGroupSortHelper aSortHelper( bTextReverse );
    2786           0 :         std::sort( aSortedVector.begin(), aSortedVector.end(), aSortHelper );
    2787             : 
    2788           0 :         pTextGroup->reset();
    2789             : 
    2790           0 :         std::vector< CustomAnimationEffectPtr >::iterator aIter( aSortedVector.begin() );
    2791           0 :         const std::vector< CustomAnimationEffectPtr >::iterator aEnd( aSortedVector.end() );
    2792             : 
    2793           0 :         if( aIter != aEnd )
    2794             :         {
    2795           0 :             pTextGroup->addEffect( (*aIter ) );
    2796           0 :             EffectSequence::iterator aInsertIter( find( (*aIter++) ) );
    2797           0 :             while( aIter != aEnd )
    2798             :             {
    2799           0 :                 CustomAnimationEffectPtr pEffect( (*aIter++) );
    2800           0 :                 maEffects.erase( find( pEffect ) );
    2801           0 :                 aInsertIter = maEffects.insert( ++aInsertIter, pEffect );
    2802           0 :                 pTextGroup->addEffect( pEffect );
    2803           0 :             }
    2804             :         }
    2805           0 :         notify_listeners();
    2806             :     }
    2807           0 : }
    2808             : 
    2809           0 : void EffectSequenceHelper::addListener( ISequenceListener* pListener )
    2810             : {
    2811           0 :     if( std::find( maListeners.begin(), maListeners.end(), pListener ) == maListeners.end() )
    2812           0 :         maListeners.push_back( pListener );
    2813           0 : }
    2814             : 
    2815           0 : void EffectSequenceHelper::removeListener( ISequenceListener* pListener )
    2816             : {
    2817           0 :     maListeners.remove( pListener );
    2818           0 : }
    2819             : 
    2820             : struct stl_notify_listeners_func : public std::unary_function<ISequenceListener*, void>
    2821             : {
    2822          93 :     stl_notify_listeners_func() {}
    2823           0 :     void operator()(ISequenceListener* pListener) { pListener->notify_change(); }
    2824             : };
    2825             : 
    2826          93 : void EffectSequenceHelper::notify_listeners()
    2827             : {
    2828          93 :     stl_notify_listeners_func aFunc;
    2829          93 :     std::for_each( maListeners.begin(), maListeners.end(), aFunc );
    2830          93 : }
    2831             : 
    2832           0 : void EffectSequenceHelper::create( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xNode )
    2833             : {
    2834             :     DBG_ASSERT( xNode.is(), "sd::EffectSequenceHelper::create(), illegal argument" );
    2835             : 
    2836           0 :     if( xNode.is() ) try
    2837             :     {
    2838           0 :         Reference< XEnumerationAccess > xEnumerationAccess( xNode, UNO_QUERY_THROW );
    2839           0 :         Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    2840           0 :         while( xEnumeration->hasMoreElements() )
    2841             :         {
    2842           0 :             Reference< XAnimationNode > xChildNode( xEnumeration->nextElement(), UNO_QUERY_THROW );
    2843           0 :             createEffectsequence( xChildNode );
    2844           0 :         }
    2845             :     }
    2846           0 :     catch( Exception& )
    2847             :     {
    2848             :         OSL_FAIL( "sd::EffectSequenceHelper::create(), exception caught!" );
    2849             :     }
    2850           0 : }
    2851             : 
    2852           0 : void EffectSequenceHelper::createEffectsequence( const Reference< XAnimationNode >& xNode )
    2853             : {
    2854             :     DBG_ASSERT( xNode.is(), "sd::EffectSequenceHelper::createEffectsequence(), illegal argument" );
    2855             : 
    2856           0 :     if( xNode.is() ) try
    2857             :     {
    2858           0 :         Reference< XEnumerationAccess > xEnumerationAccess( xNode, UNO_QUERY_THROW );
    2859           0 :         Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    2860           0 :         while( xEnumeration->hasMoreElements() )
    2861             :         {
    2862           0 :             Reference< XAnimationNode > xChildNode( xEnumeration->nextElement(), UNO_QUERY_THROW );
    2863             : 
    2864           0 :             createEffects( xChildNode );
    2865           0 :         }
    2866             :     }
    2867           0 :     catch( Exception& )
    2868             :     {
    2869             :         OSL_FAIL( "sd::EffectSequenceHelper::createEffectsequence(), exception caught!" );
    2870             :     }
    2871           0 : }
    2872             : 
    2873           0 : void EffectSequenceHelper::createEffects( const Reference< XAnimationNode >& xNode )
    2874             : {
    2875             :     DBG_ASSERT( xNode.is(), "sd::EffectSequenceHelper::createEffects(), illegal argument" );
    2876             : 
    2877           0 :     if( xNode.is() ) try
    2878             :     {
    2879           0 :         Reference< XEnumerationAccess > xEnumerationAccess( xNode, UNO_QUERY_THROW );
    2880           0 :         Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    2881           0 :         while( xEnumeration->hasMoreElements() )
    2882             :         {
    2883           0 :             Reference< XAnimationNode > xChildNode( xEnumeration->nextElement(), UNO_QUERY_THROW );
    2884             : 
    2885           0 :             switch( xChildNode->getType() )
    2886             :             {
    2887             :             // found an effect
    2888             :             case AnimationNodeType::PAR:
    2889             :             case AnimationNodeType::ITERATE:
    2890             :                 {
    2891           0 :                     CustomAnimationEffectPtr pEffect( new CustomAnimationEffect( xChildNode ) );
    2892             : 
    2893           0 :                     if( pEffect->mnNodeType != -1 )
    2894             :                     {
    2895           0 :                         pEffect->setEffectSequence( this );
    2896           0 :                         maEffects.push_back(pEffect);
    2897           0 :                     }
    2898             :                 }
    2899           0 :                 break;
    2900             : 
    2901             :             // found an after effect
    2902             :             case AnimationNodeType::SET:
    2903             :             case AnimationNodeType::ANIMATECOLOR:
    2904             :                 {
    2905           0 :                     processAfterEffect( xChildNode );
    2906             :                 }
    2907           0 :                 break;
    2908             :             }
    2909           0 :         }
    2910             :     }
    2911           0 :     catch( Exception& )
    2912             :     {
    2913             :         OSL_FAIL( "sd::EffectSequenceHelper::createEffects(), exception caught!" );
    2914             :     }
    2915           0 : }
    2916             : 
    2917           0 : void EffectSequenceHelper::processAfterEffect( const Reference< XAnimationNode >& xNode )
    2918             : {
    2919             :     try
    2920             :     {
    2921           0 :         Reference< XAnimationNode > xMaster;
    2922             : 
    2923           0 :         Sequence< NamedValue > aUserData( xNode->getUserData() );
    2924           0 :         sal_Int32 nLength = aUserData.getLength();
    2925           0 :         const NamedValue* p = aUserData.getConstArray();
    2926             : 
    2927           0 :         while( nLength-- )
    2928             :         {
    2929           0 :             if ( p->Name == "master-element" )
    2930             :             {
    2931           0 :                 p->Value >>= xMaster;
    2932           0 :                 break;
    2933             :             }
    2934           0 :             p++;
    2935             :         }
    2936             : 
    2937             :         // only process if this is a valid after effect
    2938           0 :         if( xMaster.is() )
    2939             :         {
    2940           0 :             CustomAnimationEffectPtr pMasterEffect;
    2941             : 
    2942             :             // find the master effect
    2943           0 :             stl_CustomAnimationEffect_search_node_predict aSearchPredict( xMaster );
    2944           0 :             EffectSequence::iterator aIter( std::find_if( maEffects.begin(), maEffects.end(), aSearchPredict ) );
    2945           0 :             if( aIter != maEffects.end() )
    2946           0 :                 pMasterEffect = (*aIter );
    2947             : 
    2948           0 :             if( pMasterEffect.get() )
    2949             :             {
    2950           0 :                 pMasterEffect->setHasAfterEffect( true );
    2951             : 
    2952             :                 // find out what kind of after effect this is
    2953           0 :                 if( xNode->getType() == AnimationNodeType::ANIMATECOLOR )
    2954             :                 {
    2955             :                     // it's a dim
    2956           0 :                     Reference< XAnimate > xAnimate( xNode, UNO_QUERY_THROW );
    2957           0 :                     pMasterEffect->setDimColor( xAnimate->getTo() );
    2958           0 :                     pMasterEffect->setAfterEffectOnNext( true );
    2959             :                 }
    2960             :                 else
    2961             :                 {
    2962             :                     // it's a hide
    2963           0 :                     pMasterEffect->setAfterEffectOnNext( xNode->getParent() != xMaster->getParent() );
    2964             :                 }
    2965           0 :             }
    2966           0 :         }
    2967             :     }
    2968           0 :     catch( Exception& )
    2969             :     {
    2970             :         OSL_FAIL( "sd::EffectSequenceHelper::processAfterEffect(), exception caught!" );
    2971             :     }
    2972           0 : }
    2973             : 
    2974         178 : class AnimationChangeListener : public cppu::WeakImplHelper1< XChangesListener >
    2975             : {
    2976             : public:
    2977          93 :     AnimationChangeListener( MainSequence* pMainSequence ) : mpMainSequence( pMainSequence ) {}
    2978             : 
    2979             :     virtual void SAL_CALL changesOccurred( const ::com::sun::star::util::ChangesEvent& Event ) throw (RuntimeException, std::exception) SAL_OVERRIDE;
    2980             :     virtual void SAL_CALL disposing( const ::com::sun::star::lang::EventObject& Source ) throw (RuntimeException, std::exception) SAL_OVERRIDE;
    2981             : private:
    2982             :     MainSequence* mpMainSequence;
    2983             : };
    2984             : 
    2985           0 : void SAL_CALL AnimationChangeListener::changesOccurred( const ::com::sun::star::util::ChangesEvent& ) throw (RuntimeException, std::exception)
    2986             : {
    2987           0 :     if( mpMainSequence )
    2988           0 :         mpMainSequence->startRecreateTimer();
    2989           0 : }
    2990             : 
    2991           0 : void SAL_CALL AnimationChangeListener::disposing( const ::com::sun::star::lang::EventObject& ) throw (RuntimeException, std::exception)
    2992             : {
    2993           0 : }
    2994             : 
    2995           0 : MainSequence::MainSequence()
    2996             :     : mxTimingRootNode(SequenceTimeContainer::create(::comphelper::getProcessComponentContext()))
    2997             :     , mbTimerMode(false)
    2998             :     , mbRebuilding( false )
    2999             :     , mnRebuildLockGuard( 0 )
    3000             :     , mbPendingRebuildRequest( false )
    3001           0 :     , mbIgnoreChanges( 0 )
    3002             : {
    3003           0 :     if( mxTimingRootNode.is() )
    3004             :     {
    3005           0 :         Sequence< ::com::sun::star::beans::NamedValue > aUserData( 1 );
    3006           0 :         aUserData[0].Name = "node-type";
    3007           0 :         aUserData[0].Value <<= ::com::sun::star::presentation::EffectNodeType::MAIN_SEQUENCE;
    3008           0 :         mxTimingRootNode->setUserData( aUserData );
    3009             :     }
    3010           0 :     init();
    3011           0 : }
    3012             : 
    3013          93 : MainSequence::MainSequence( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xNode )
    3014             :     : mxTimingRootNode( xNode, UNO_QUERY )
    3015             :     , mbTimerMode( false )
    3016             :     , mbRebuilding( false )
    3017             :     , mnRebuildLockGuard( 0 )
    3018             :     , mbPendingRebuildRequest( false )
    3019          93 :     , mbIgnoreChanges( 0 )
    3020             : {
    3021          93 :     init();
    3022          93 : }
    3023             : 
    3024         267 : MainSequence::~MainSequence()
    3025             : {
    3026          89 :     reset();
    3027         178 : }
    3028             : 
    3029          93 : void MainSequence::init()
    3030             : {
    3031          93 :     mnSequenceType = EffectNodeType::MAIN_SEQUENCE;
    3032             : 
    3033          93 :     maTimer.SetTimeoutHdl( LINK(this, MainSequence, onTimerHdl) );
    3034          93 :     maTimer.SetTimeout(500);
    3035             : 
    3036          93 :     mxChangesListener.set( new AnimationChangeListener( this ) );
    3037             : 
    3038          93 :     createMainSequence();
    3039          93 : }
    3040             : 
    3041           0 : void MainSequence::reset( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xTimingRootNode )
    3042             : {
    3043           0 :     reset();
    3044             : 
    3045           0 :     mxTimingRootNode.set( xTimingRootNode, UNO_QUERY );
    3046             : 
    3047           0 :     createMainSequence();
    3048           0 : }
    3049             : 
    3050           0 : Reference< ::com::sun::star::animations::XAnimationNode > MainSequence::getRootNode()
    3051             : {
    3052             :     DBG_ASSERT( mnRebuildLockGuard == 0, "MainSequence::getRootNode(), rebuild is locked, is this really what you want?" );
    3053             : 
    3054           0 :     if( maTimer.IsActive() && mbTimerMode )
    3055             :     {
    3056             :         // force a rebuild NOW if one is pending
    3057           0 :         maTimer.Stop();
    3058           0 :         implRebuild();
    3059             :     }
    3060             : 
    3061           0 :     return EffectSequenceHelper::getRootNode();
    3062             : }
    3063             : 
    3064          93 : void MainSequence::createMainSequence()
    3065             : {
    3066          93 :     if( mxTimingRootNode.is() ) try
    3067             :     {
    3068          93 :         Reference< XEnumerationAccess > xEnumerationAccess( mxTimingRootNode, UNO_QUERY_THROW );
    3069         186 :         Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    3070         186 :         while( xEnumeration->hasMoreElements() )
    3071             :         {
    3072           0 :             Reference< XAnimationNode > xChildNode( xEnumeration->nextElement(), UNO_QUERY_THROW );
    3073           0 :             sal_Int32 nNodeType = CustomAnimationEffect::get_node_type( xChildNode );
    3074           0 :             if( nNodeType == EffectNodeType::MAIN_SEQUENCE )
    3075             :             {
    3076           0 :                 mxSequenceRoot.set( xChildNode, UNO_QUERY );
    3077           0 :                 EffectSequenceHelper::create( xChildNode );
    3078             :             }
    3079           0 :             else if( nNodeType == EffectNodeType::INTERACTIVE_SEQUENCE )
    3080             :             {
    3081           0 :                 Reference< XTimeContainer > xInteractiveRoot( xChildNode, UNO_QUERY_THROW );
    3082           0 :                 InteractiveSequencePtr pIS( new InteractiveSequence( xInteractiveRoot, this ) );
    3083           0 :                 pIS->addListener( this );
    3084           0 :                 maInteractiveSequenceList.push_back( pIS );
    3085             :             }
    3086           0 :         }
    3087             : 
    3088             :         // see if we have a mainsequence at all. if not, create one...
    3089          93 :         if( !mxSequenceRoot.is() )
    3090             :         {
    3091          93 :             mxSequenceRoot = SequenceTimeContainer::create( ::comphelper::getProcessComponentContext() );
    3092             : 
    3093          93 :             uno::Sequence< ::com::sun::star::beans::NamedValue > aUserData( 1 );
    3094          93 :             aUserData[0].Name = "node-type";
    3095          93 :             aUserData[0].Value <<= ::com::sun::star::presentation::EffectNodeType::MAIN_SEQUENCE;
    3096          93 :             mxSequenceRoot->setUserData( aUserData );
    3097             : 
    3098             :             // empty sequence until now, set duration to 0.0
    3099             :             // explicitly (otherwise, this sequence will never
    3100             :             // end)
    3101          93 :             mxSequenceRoot->setDuration( makeAny((double)0.0) );
    3102             : 
    3103         186 :             Reference< XAnimationNode > xMainSequenceNode( mxSequenceRoot, UNO_QUERY_THROW );
    3104         186 :             mxTimingRootNode->appendChild( xMainSequenceNode );
    3105             :         }
    3106             : 
    3107          93 :         updateTextGroups();
    3108             : 
    3109          93 :         notify_listeners();
    3110             : 
    3111         186 :         Reference< XChangesNotifier > xNotifier( mxTimingRootNode, UNO_QUERY );
    3112          93 :         if( xNotifier.is() )
    3113         186 :             xNotifier->addChangesListener( mxChangesListener );
    3114             :     }
    3115           0 :     catch( Exception& )
    3116             :     {
    3117             :         OSL_FAIL( "sd::MainSequence::create(), exception caught!" );
    3118          93 :         return;
    3119             :     }
    3120             : 
    3121             :     DBG_ASSERT( mxSequenceRoot.is(), "sd::MainSequence::create(), found no main sequence!" );
    3122             : }
    3123             : 
    3124          89 : void MainSequence::reset()
    3125             : {
    3126          89 :     EffectSequenceHelper::reset();
    3127             : 
    3128          89 :     InteractiveSequenceList::iterator aIter;
    3129          89 :     for( aIter = maInteractiveSequenceList.begin(); aIter != maInteractiveSequenceList.end(); ++aIter )
    3130           0 :         (*aIter)->reset();
    3131          89 :     maInteractiveSequenceList.clear();
    3132             : 
    3133             :     try
    3134             :     {
    3135          89 :         Reference< XChangesNotifier > xNotifier( mxTimingRootNode, UNO_QUERY );
    3136          89 :         if( xNotifier.is() )
    3137          89 :             xNotifier->removeChangesListener( mxChangesListener );
    3138             :     }
    3139           0 :     catch( Exception& )
    3140             :     {
    3141             : 
    3142             :     }
    3143          89 : }
    3144             : 
    3145           0 : InteractiveSequencePtr MainSequence::createInteractiveSequence( const ::com::sun::star::uno::Reference< ::com::sun::star::drawing::XShape >& xShape )
    3146             : {
    3147           0 :     InteractiveSequencePtr pIS;
    3148             : 
    3149             :     // create a new interactive sequence container
    3150           0 :     Reference< XTimeContainer > xISRoot = SequenceTimeContainer::create( ::comphelper::getProcessComponentContext() );
    3151             : 
    3152           0 :     uno::Sequence< ::com::sun::star::beans::NamedValue > aUserData( 1 );
    3153           0 :     aUserData[0].Name = "node-type";
    3154           0 :     aUserData[0].Value <<= ::com::sun::star::presentation::EffectNodeType::INTERACTIVE_SEQUENCE ;
    3155           0 :     xISRoot->setUserData( aUserData );
    3156             : 
    3157           0 :     Reference< XChild > xChild( mxSequenceRoot, UNO_QUERY_THROW );
    3158           0 :     Reference< XTimeContainer > xParent( xChild->getParent(), UNO_QUERY_THROW );
    3159           0 :     xParent->appendChild( xISRoot );
    3160             : 
    3161           0 :     pIS.reset( new InteractiveSequence( xISRoot, this) );
    3162           0 :     pIS->setTriggerShape( xShape );
    3163           0 :     pIS->addListener( this );
    3164           0 :     maInteractiveSequenceList.push_back( pIS );
    3165           0 :     return pIS;
    3166             : }
    3167             : 
    3168           0 : CustomAnimationEffectPtr MainSequence::findEffect( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xNode ) const
    3169             : {
    3170           0 :     CustomAnimationEffectPtr pEffect = EffectSequenceHelper::findEffect( xNode );
    3171             : 
    3172           0 :     if( pEffect.get() == 0 )
    3173             :     {
    3174           0 :         InteractiveSequenceList::const_iterator aIter;
    3175           0 :         for( aIter = maInteractiveSequenceList.begin(); (aIter != maInteractiveSequenceList.end()) && (pEffect.get() == 0); ++aIter )
    3176             :         {
    3177           0 :             pEffect = (*aIter)->findEffect( xNode );
    3178             :         }
    3179             :     }
    3180           0 :     return pEffect;
    3181             : }
    3182             : 
    3183           0 : sal_Int32 MainSequence::getOffsetFromEffect( const CustomAnimationEffectPtr& pEffect ) const
    3184             : {
    3185           0 :     sal_Int32 nOffset = EffectSequenceHelper::getOffsetFromEffect( pEffect );
    3186             : 
    3187           0 :     if( nOffset != -1 )
    3188           0 :         return nOffset;
    3189             : 
    3190           0 :     nOffset = EffectSequenceHelper::getCount();
    3191             : 
    3192           0 :     InteractiveSequenceList::const_iterator aIter;
    3193           0 :     for( aIter = maInteractiveSequenceList.begin(); aIter != maInteractiveSequenceList.end(); ++aIter )
    3194             :     {
    3195           0 :         sal_Int32 nTemp = (*aIter)->getOffsetFromEffect( pEffect );
    3196           0 :         if( nTemp != -1 )
    3197           0 :             return nOffset + nTemp;
    3198             : 
    3199           0 :         nOffset += (*aIter)->getCount();
    3200             :     }
    3201             : 
    3202           0 :     return -1;
    3203             : }
    3204             : 
    3205           0 : CustomAnimationEffectPtr MainSequence::getEffectFromOffset( sal_Int32 nOffset ) const
    3206             : {
    3207           0 :     if( nOffset >= 0 )
    3208             :     {
    3209           0 :         if( nOffset < getCount() )
    3210           0 :             return EffectSequenceHelper::getEffectFromOffset( nOffset );
    3211             : 
    3212           0 :         nOffset -= getCount();
    3213             : 
    3214           0 :         InteractiveSequenceList::const_iterator aIter( maInteractiveSequenceList.begin() );
    3215             : 
    3216           0 :         while( (aIter != maInteractiveSequenceList.end()) && (nOffset > (*aIter)->getCount()) )
    3217           0 :             nOffset -= (*aIter++)->getCount();
    3218             : 
    3219           0 :         if( (aIter != maInteractiveSequenceList.end()) && (nOffset >= 0) )
    3220           0 :             return (*aIter)->getEffectFromOffset( nOffset );
    3221             :     }
    3222             : 
    3223           0 :     CustomAnimationEffectPtr pEffect;
    3224           0 :     return pEffect;
    3225             : }
    3226             : 
    3227           0 : bool MainSequence::disposeShape( const Reference< XShape >& xShape )
    3228             : {
    3229           0 :     bool bChanges = EffectSequenceHelper::disposeShape( xShape );
    3230             : 
    3231           0 :     InteractiveSequenceList::iterator aIter;
    3232           0 :     for( aIter = maInteractiveSequenceList.begin(); aIter != maInteractiveSequenceList.end();  )
    3233             :     {
    3234           0 :             bChanges |= (*aIter++)->disposeShape( xShape );
    3235             :     }
    3236             : 
    3237           0 :     if( bChanges )
    3238           0 :         startRebuildTimer();
    3239             : 
    3240           0 :     return bChanges;
    3241             : }
    3242             : 
    3243         168 : bool MainSequence::hasEffect( const com::sun::star::uno::Reference< com::sun::star::drawing::XShape >& xShape )
    3244             : {
    3245         168 :     if( EffectSequenceHelper::hasEffect( xShape ) )
    3246           0 :         return true;
    3247             : 
    3248         168 :     InteractiveSequenceList::iterator aIter;
    3249         336 :     for( aIter = maInteractiveSequenceList.begin(); aIter != maInteractiveSequenceList.end();  )
    3250             :     {
    3251           0 :         if( (*aIter)->getTriggerShape() == xShape )
    3252           0 :             return true;
    3253             : 
    3254           0 :         if( (*aIter++)->hasEffect( xShape ) )
    3255           0 :             return true;
    3256             :     }
    3257             : 
    3258         168 :     return false;
    3259             : }
    3260             : 
    3261           0 : void MainSequence::insertTextRange( const com::sun::star::uno::Any& aTarget )
    3262             : {
    3263           0 :     EffectSequenceHelper::insertTextRange( aTarget );
    3264             : 
    3265           0 :     InteractiveSequenceList::iterator aIter;
    3266           0 :     for( aIter = maInteractiveSequenceList.begin(); aIter != maInteractiveSequenceList.end(); ++aIter )
    3267             :     {
    3268           0 :         (*aIter)->insertTextRange( aTarget );
    3269             :     }
    3270           0 : }
    3271             : 
    3272           0 : void MainSequence::disposeTextRange( const com::sun::star::uno::Any& aTarget )
    3273             : {
    3274           0 :     EffectSequenceHelper::disposeTextRange( aTarget );
    3275             : 
    3276           0 :     InteractiveSequenceList::iterator aIter;
    3277           0 :     for( aIter = maInteractiveSequenceList.begin(); aIter != maInteractiveSequenceList.end(); ++aIter )
    3278             :     {
    3279           0 :         (*aIter)->disposeTextRange( aTarget );
    3280             :     }
    3281           0 : }
    3282             : 
    3283             : /** callback from the sd::View when an object just left text edit mode */
    3284           0 : void MainSequence::onTextChanged( const Reference< XShape >& xShape )
    3285             : {
    3286           0 :     EffectSequenceHelper::onTextChanged( xShape );
    3287             : 
    3288           0 :     InteractiveSequenceList::iterator aIter;
    3289           0 :     for( aIter = maInteractiveSequenceList.begin(); aIter != maInteractiveSequenceList.end(); ++aIter )
    3290             :     {
    3291           0 :         (*aIter)->onTextChanged( xShape );
    3292             :     }
    3293           0 : }
    3294             : 
    3295           0 : void EffectSequenceHelper::onTextChanged( const Reference< XShape >& xShape )
    3296             : {
    3297           0 :     bool bChanges = false;
    3298             : 
    3299           0 :     EffectSequence::iterator aIter;
    3300           0 :     for( aIter = maEffects.begin(); aIter != maEffects.end(); ++aIter )
    3301             :     {
    3302           0 :         if( (*aIter)->getTargetShape() == xShape )
    3303           0 :             bChanges |= (*aIter)->checkForText();
    3304             :     }
    3305             : 
    3306           0 :     if( bChanges )
    3307           0 :         EffectSequenceHelper::implRebuild();
    3308           0 : }
    3309             : 
    3310           0 : void MainSequence::rebuild()
    3311             : {
    3312           0 :     startRebuildTimer();
    3313           0 : }
    3314             : 
    3315           0 : void MainSequence::lockRebuilds()
    3316             : {
    3317           0 :     mnRebuildLockGuard++;
    3318           0 : }
    3319             : 
    3320           0 : void MainSequence::unlockRebuilds()
    3321             : {
    3322             :     DBG_ASSERT( mnRebuildLockGuard, "sd::MainSequence::unlockRebuilds(), no corresponding lockRebuilds() call!" );
    3323           0 :     if( mnRebuildLockGuard )
    3324           0 :         mnRebuildLockGuard--;
    3325             : 
    3326           0 :     if( (mnRebuildLockGuard == 0) && mbPendingRebuildRequest )
    3327             :     {
    3328           0 :         mbPendingRebuildRequest = false;
    3329           0 :         startRebuildTimer();
    3330             :     }
    3331           0 : }
    3332             : 
    3333           0 : void MainSequence::implRebuild()
    3334             : {
    3335           0 :     if( mnRebuildLockGuard )
    3336             :     {
    3337           0 :         mbPendingRebuildRequest = true;
    3338           0 :         return;
    3339             :     }
    3340             : 
    3341           0 :     mbRebuilding = true;
    3342             : 
    3343           0 :     EffectSequenceHelper::implRebuild();
    3344             : 
    3345           0 :     InteractiveSequenceList::iterator aIter( maInteractiveSequenceList.begin() );
    3346           0 :     const InteractiveSequenceList::iterator aEnd( maInteractiveSequenceList.end() );
    3347           0 :     while( aIter != aEnd )
    3348             :     {
    3349           0 :         InteractiveSequencePtr pIS( (*aIter) );
    3350           0 :         if( pIS->maEffects.empty() )
    3351             :         {
    3352             :             // remove empty interactive sequences
    3353           0 :             aIter = maInteractiveSequenceList.erase( aIter );
    3354             : 
    3355           0 :             Reference< XChild > xChild( mxSequenceRoot, UNO_QUERY_THROW );
    3356           0 :             Reference< XTimeContainer > xParent( xChild->getParent(), UNO_QUERY_THROW );
    3357           0 :             Reference< XAnimationNode > xISNode( pIS->mxSequenceRoot, UNO_QUERY_THROW );
    3358           0 :             xParent->removeChild( xISNode );
    3359             :         }
    3360             :         else
    3361             :         {
    3362           0 :             pIS->implRebuild();
    3363           0 :             ++aIter;
    3364             :         }
    3365           0 :     }
    3366             : 
    3367           0 :     notify_listeners();
    3368           0 :     mbRebuilding = false;
    3369             : }
    3370             : 
    3371           0 : void MainSequence::notify_change()
    3372             : {
    3373           0 :     notify_listeners();
    3374           0 : }
    3375             : 
    3376           0 : bool MainSequence::setTrigger( const CustomAnimationEffectPtr& pEffect, const ::com::sun::star::uno::Reference< ::com::sun::star::drawing::XShape >& xTriggerShape )
    3377             : {
    3378           0 :     EffectSequenceHelper* pOldSequence = pEffect->getEffectSequence();
    3379             : 
    3380           0 :     EffectSequenceHelper* pNewSequence = 0;
    3381           0 :     if( xTriggerShape.is() )
    3382             :     {
    3383           0 :         InteractiveSequenceList::iterator aIter( maInteractiveSequenceList.begin() );
    3384           0 :         const InteractiveSequenceList::iterator aEnd( maInteractiveSequenceList.end() );
    3385           0 :         while( aIter != aEnd )
    3386             :         {
    3387           0 :             InteractiveSequencePtr pIS( (*aIter++) );
    3388           0 :             if( pIS->getTriggerShape() == xTriggerShape )
    3389             :             {
    3390           0 :                 pNewSequence = pIS.get();
    3391           0 :                 break;
    3392             :             }
    3393           0 :         }
    3394             : 
    3395           0 :         if( !pNewSequence )
    3396           0 :             pNewSequence = createInteractiveSequence( xTriggerShape ).get();
    3397             :     }
    3398             :     else
    3399             :     {
    3400           0 :         pNewSequence = this;
    3401             :     }
    3402             : 
    3403           0 :     if( pOldSequence != pNewSequence )
    3404             :     {
    3405           0 :         if( pOldSequence )
    3406           0 :             pOldSequence->maEffects.remove( pEffect );
    3407           0 :         if( pNewSequence )
    3408           0 :             pNewSequence->maEffects.push_back( pEffect );
    3409           0 :         pEffect->setEffectSequence( pNewSequence );
    3410           0 :         return true;
    3411             :     }
    3412             :     else
    3413             :     {
    3414           0 :         return false;
    3415             :     }
    3416             : 
    3417             : }
    3418             : 
    3419           0 : IMPL_LINK_NOARG_TYPED(MainSequence, onTimerHdl, Timer *, void)
    3420             : {
    3421           0 :     if( mbTimerMode )
    3422             :     {
    3423           0 :         implRebuild();
    3424             :     }
    3425             :     else
    3426             :     {
    3427           0 :         reset();
    3428           0 :         createMainSequence();
    3429             :     }
    3430           0 : }
    3431             : 
    3432             : /** starts a timer that recreates the internal structure from the API core after 1 second */
    3433           0 : void MainSequence::startRecreateTimer()
    3434             : {
    3435           0 :     if( !mbRebuilding && (mbIgnoreChanges == 0) )
    3436             :     {
    3437           0 :         mbTimerMode = false;
    3438           0 :         maTimer.Start();
    3439             :     }
    3440           0 : }
    3441             : 
    3442             : /** starts a timer that rebuilds the API core from the internal structure after 1 second */
    3443           0 : void MainSequence::startRebuildTimer()
    3444             : {
    3445           0 :     mbTimerMode = true;
    3446           0 :     maTimer.Start();
    3447           0 : }
    3448             : 
    3449           0 : InteractiveSequence::InteractiveSequence( const Reference< XTimeContainer >& xSequenceRoot, MainSequence* pMainSequence )
    3450           0 : : EffectSequenceHelper( xSequenceRoot ), mpMainSequence( pMainSequence )
    3451             : {
    3452           0 :     mnSequenceType = EffectNodeType::INTERACTIVE_SEQUENCE;
    3453             : 
    3454             :     try
    3455             :     {
    3456           0 :         if( mxSequenceRoot.is() )
    3457             :         {
    3458           0 :             Reference< XEnumerationAccess > xEnumerationAccess( mxSequenceRoot, UNO_QUERY_THROW );
    3459           0 :             Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
    3460           0 :             while( !mxEventSource.is() && xEnumeration->hasMoreElements() )
    3461             :             {
    3462           0 :                 Reference< XAnimationNode > xChildNode( xEnumeration->nextElement(), UNO_QUERY_THROW );
    3463             : 
    3464           0 :                 Event aEvent;
    3465           0 :                 if( (xChildNode->getBegin() >>= aEvent) && (aEvent.Trigger == EventTrigger::ON_CLICK) )
    3466           0 :                     aEvent.Source >>= mxEventSource;
    3467           0 :             }
    3468             :         }
    3469             :     }
    3470           0 :     catch( Exception& )
    3471             :     {
    3472             :         OSL_FAIL( "sd::InteractiveSequence::InteractiveSequence(), exception caught!" );
    3473           0 :         return;
    3474             :     }
    3475             : }
    3476             : 
    3477           0 : void InteractiveSequence::rebuild()
    3478             : {
    3479           0 :     mpMainSequence->rebuild();
    3480           0 : }
    3481             : 
    3482           0 : void InteractiveSequence::implRebuild()
    3483             : {
    3484           0 :     EffectSequenceHelper::implRebuild();
    3485           0 : }
    3486             : 
    3487           0 : MainSequenceRebuildGuard::MainSequenceRebuildGuard( const MainSequencePtr& pMainSequence )
    3488           0 : : mpMainSequence( pMainSequence )
    3489             : {
    3490           0 :     if( mpMainSequence.get() )
    3491           0 :         mpMainSequence->lockRebuilds();
    3492           0 : }
    3493             : 
    3494           0 : MainSequenceRebuildGuard::~MainSequenceRebuildGuard()
    3495             : {
    3496           0 :     if( mpMainSequence.get() )
    3497           0 :         mpMainSequence->unlockRebuilds();
    3498           0 : }
    3499             : 
    3500          66 : }
    3501             : 
    3502             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.11