1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_sd.hxx"
26 #include <com/sun/star/animations/XAnimationNode.hpp>
27 #include <com/sun/star/animations/Event.hpp>
28 #ifndef _COM_SUN_STAR_ANIMATIONS_XAnimateColor_HPP_
29 #include <com/sun/star/animations/XAnimateColor.hpp>
30 #endif
31 #ifndef _COM_SUN_STAR_ANIMATIONS_XAnimateSet_HPP_
32 #include <com/sun/star/animations/XAnimateSet.hpp>
33 #endif
34 #include <com/sun/star/animations/XCommand.hpp>
35 #ifndef _COM_SUN_STAR_ANIMATIONS_XAnimateMotion_HPP_
36 #include <com/sun/star/animations/XAnimateMotion.hpp>
37 #endif
38 #ifndef _COM_SUN_STAR_ANIMATIONS_XAnimateTransform_HPP_
39 #include <com/sun/star/animations/XAnimateTransform.hpp>
40 #endif
41 #ifndef _COM_SUN_STAR_ANIMATIONS_XTransitionFilter_HPP_
42 #include <com/sun/star/animations/XTransitionFilter.hpp>
43 #endif
44 #include <com/sun/star/animations/XIterateContainer.hpp>
45 #include <com/sun/star/animations/XAudio.hpp>
46 #include <com/sun/star/animations/AnimationNodeType.hpp>
47 #include <com/sun/star/animations/ValuePair.hpp>
48 #include <com/sun/star/presentation/EffectNodeType.hpp>
49 #include <com/sun/star/util/XCloneable.hpp>
50 #include <com/sun/star/presentation/ParagraphTarget.hpp>
51 #include <com/sun/star/container/XEnumerationAccess.hpp>
52 #include <com/sun/star/beans/NamedValue.hpp>
53 
54 #include <map>
55 
56 #include "comphelper/anytostring.hxx"
57 #include "cppuhelper/exc_hlp.hxx"
58 #include "rtl/ref.hxx"
59 #include <animations/animationnodehelper.hxx>
60 
61 // header for class SdrObjListIter
62 #include <svx/svditer.hxx>
63 
64 #include "sdpage.hxx"
65 
66 using namespace ::com::sun::star::uno;
67 using namespace ::com::sun::star::animations;
68 using namespace ::com::sun::star::presentation;
69 using namespace ::com::sun::star::container;
70 
71 using ::rtl::OUString;
72 using ::rtl::OString;
73 using ::com::sun::star::drawing::XShape;
74 using ::com::sun::star::beans::NamedValue;
75 
76 namespace sd
77 {
78     class CustomAnimationClonerImpl
79     {
80 	public:
81         CustomAnimationClonerImpl();
82 		Reference< XAnimationNode > Clone( const Reference< XAnimationNode >& xSourceNode, const SdPage* pSource = 0, const SdPage* pTarget = 0 );
83 
84     private:
85 		void transformNode( const Reference< XAnimationNode >& xNode );
86 		Any transformValue( const Any& rValue );
87 
88         Reference< XShape > getClonedShape( const Reference< XShape >& xSource ) const;
89         Reference< XAnimationNode > getClonedNode( const Reference< XAnimationNode >& xSource ) const;
90 
91         mutable ::std::map< Reference< XShape >, Reference< XShape > > maShapeMap;
92 		std::vector< Reference< XAnimationNode > > maSourceNodeVector;
93 		std::vector< Reference< XAnimationNode > > maCloneNodeVector;
94     };
95 
96 	CustomAnimationClonerImpl::CustomAnimationClonerImpl()
97 	{
98 	}
99 
100 	Reference< XAnimationNode > Clone( const Reference< XAnimationNode >& xSourceNode, const SdPage* pSource, const SdPage* pTarget )
101 	{
102 		CustomAnimationClonerImpl aCloner;
103 		return aCloner.Clone( xSourceNode, pSource, pTarget );
104 	}
105 
106 	Reference< XAnimationNode > CustomAnimationClonerImpl::Clone( const Reference< XAnimationNode >& xSourceNode, const SdPage* pSourcePage, const SdPage* pTargetPage )
107 	{
108 		try
109 		{
110 			// clone animation hierarchie
111 			Reference< ::com::sun::star::util::XCloneable > xClonable( xSourceNode, UNO_QUERY_THROW );
112 			Reference< XAnimationNode > xCloneNode( xClonable->createClone(), UNO_QUERY_THROW );
113 
114 			// create a dictionary to map source to cloned shapes
115 			if( pSourcePage && pTargetPage )
116 			{
117 				SdrObjListIter aSourceIter( *pSourcePage, IM_DEEPWITHGROUPS );
118 				SdrObjListIter aTargetIter( *pTargetPage, IM_DEEPWITHGROUPS );
119 
120 				while( aSourceIter.IsMore() && aTargetIter.IsMore() )
121 				{
122 					SdrObject* pSource = aSourceIter.Next();
123 					SdrObject* pTarget = aTargetIter.Next();
124 
125 					if( pSource && pTarget)
126 					{
127 						Reference< XShape > xSource( pSource->getUnoShape(), UNO_QUERY );
128 						Reference< XShape > xTarget( pTarget->getUnoShape(), UNO_QUERY );
129 						if( xSource.is() && xTarget.is() )
130 						{
131 							maShapeMap[xSource] = xTarget;
132 						}
133 					}
134 				}
135 			}
136 
137 			// create a dictionary to map source to cloned nodes
138 			::anim::create_deep_vector( xSourceNode, maSourceNodeVector );
139 			::anim::create_deep_vector( xCloneNode, maCloneNodeVector );
140 
141 			transformNode( xCloneNode );
142 
143 			return xCloneNode;
144 		}
145 		catch( Exception& e )
146 		{
147 			(void)e;
148 			DBG_ERROR(
149 				(OString("sd::CustomAnimationClonerImpl::Clone(), "
150 						"exception caught: ") +
151 				rtl::OUStringToOString(
152 					comphelper::anyToString( cppu::getCaughtException() ),
153 					RTL_TEXTENCODING_UTF8 )).getStr() );
154 
155 			Reference< XAnimationNode > xEmpty;
156 			return xEmpty;
157 		}
158 	}
159 
160 	void CustomAnimationClonerImpl::transformNode( const Reference< XAnimationNode >& xNode )
161 	{
162 		try
163 		{
164 			xNode->setBegin( transformValue( xNode->getBegin() ) );
165 			xNode->setEnd( transformValue( xNode->getEnd() ) );
166 
167 			sal_Int16 nNodeType( xNode->getType() );
168 			switch( nNodeType )
169 			{
170 			case AnimationNodeType::ITERATE:
171 			{
172 				Reference< XIterateContainer > xIter( xNode, UNO_QUERY_THROW );
173 				xIter->setTarget( transformValue( xIter->getTarget() ) );
174 			}
175 			// its intended that here is no break!
176 			case AnimationNodeType::PAR:
177 			case AnimationNodeType::SEQ:
178 			{
179 				Reference< XEnumerationAccess > xEnumerationAccess( xNode, UNO_QUERY_THROW );
180 				Reference< XEnumeration > xEnumeration( xEnumerationAccess->createEnumeration(), UNO_QUERY_THROW );
181 				while( xEnumeration->hasMoreElements() )
182 				{
183 					Reference< XAnimationNode > xChildNode( xEnumeration->nextElement(), UNO_QUERY_THROW );
184 					transformNode( xChildNode );
185 				}
186 			}
187 			break;
188 
189 			case AnimationNodeType::ANIMATE:
190 			case AnimationNodeType::SET:
191 			case AnimationNodeType::ANIMATEMOTION:
192 			case AnimationNodeType::ANIMATECOLOR:
193 			case AnimationNodeType::ANIMATETRANSFORM:
194 			case AnimationNodeType::TRANSITIONFILTER:
195 			{
196 				Reference< XAnimate > xAnimate( xNode, UNO_QUERY_THROW );
197 				xAnimate->setTarget( transformValue( xAnimate->getTarget() ) );
198 			}
199 			break;
200 
201 			case AnimationNodeType::COMMAND:
202 			{
203 				Reference< XCommand > xCommand( xNode, UNO_QUERY_THROW );
204 				xCommand->setTarget( transformValue( xCommand->getTarget() ) );
205 			}
206 			break;
207 
208 			case AnimationNodeType::AUDIO:
209 			{
210 				Reference< XAudio > xAudio( xNode, UNO_QUERY_THROW );
211 				xAudio->setSource( transformValue( xAudio->getSource() ) );
212 			}
213 			break;
214 			}
215 
216 			Sequence< NamedValue > aUserData( xNode->getUserData() );
217 			if( aUserData.hasElements() )
218 			{
219 				NamedValue* pValue = aUserData.getArray();
220 				const sal_Int32 nLength = aUserData.getLength();
221 				sal_Int32 nElement;
222 				for( nElement = 0; nElement < nLength; nElement++, pValue++ )
223 				{
224 					pValue->Value = transformValue( pValue->Value );
225 				}
226 
227 				xNode->setUserData( aUserData );
228 			}
229 		}
230 		catch( Exception& e )
231 		{
232 			(void)e;
233 			DBG_ERROR(
234 				(OString("sd::CustomAnimationClonerImpl::transformNode(), "
235 						"exception caught: ") +
236 				rtl::OUStringToOString(
237 					comphelper::anyToString( cppu::getCaughtException() ),
238 					RTL_TEXTENCODING_UTF8 )).getStr() );
239 		}
240 	}
241 
242 	Any CustomAnimationClonerImpl::transformValue( const Any& rValue )
243 	{
244 		if( rValue.hasValue() ) try
245 		{
246 			if( rValue.getValueType() == ::getCppuType((const ValuePair*)0) )
247 			{
248 				ValuePair aValuePair;
249 				rValue >>= aValuePair;
250 
251 				aValuePair.First = transformValue( aValuePair.First );
252 				aValuePair.Second = transformValue( aValuePair.Second );
253 
254 				return makeAny( aValuePair );
255 			}
256 			else if( rValue.getValueType() == ::getCppuType((Sequence<Any>*)0) )
257 			{
258 				Sequence<Any> aSequence;
259 				rValue >>= aSequence;
260 
261 				const sal_Int32 nLength = aSequence.getLength();
262 				sal_Int32 nElement;
263 				Any* pAny = aSequence.getArray();
264 
265 				for( nElement = 0; nElement < nLength; nElement++, pAny++ )
266 					*pAny = transformValue( *pAny );
267 
268 				return makeAny( aSequence );
269 			}
270 			else if( rValue.getValueTypeClass() == TypeClass_INTERFACE )
271 			{
272 				Reference< XShape > xShape;
273 				rValue >>= xShape;
274 				if( xShape.is() )
275 				{
276 					return makeAny( getClonedShape( xShape ) );
277 				}
278 				else
279 				{
280 					Reference< XAnimationNode > xNode;
281 					rValue >>= xNode;
282 					if( xNode.is() )
283 						return makeAny( getClonedNode( xNode ) );
284 				}
285 			}
286 			else if( rValue.getValueType() == ::getCppuType((const ParagraphTarget*)0) )
287 			{
288 				ParagraphTarget aParaTarget;
289 				rValue >>= aParaTarget;
290 
291 				aParaTarget.Shape = getClonedShape( aParaTarget.Shape );
292 
293 				return makeAny( aParaTarget );
294 			}
295 			else if( rValue.getValueType() == ::getCppuType((const Event*)0) )
296 			{
297 				Event aEvent;
298 				rValue >>= aEvent;
299 
300 				aEvent.Source = transformValue( aEvent.Source );
301 
302 				return makeAny( aEvent );
303 			}
304 		}
305 		catch( Exception& e )
306 		{
307 			(void)e;
308 			DBG_ERROR(
309 				(OString("sd::CustomAnimationClonerImpl::transformValue(), "
310 						"exception caught: ") +
311 				rtl::OUStringToOString(
312 					comphelper::anyToString( cppu::getCaughtException() ),
313 					RTL_TEXTENCODING_UTF8 )).getStr() );
314 		}
315 
316 		return rValue;
317 	}
318 
319     Reference< XShape > CustomAnimationClonerImpl::getClonedShape( const Reference< XShape >& xSource ) const
320     {
321         if( xSource.is() )
322 		{
323 			if( maShapeMap.find(xSource) != maShapeMap.end() )
324 			{
325 				return maShapeMap[xSource];
326 			}
327 
328 			DBG_ASSERT( maShapeMap.empty(), "sd::CustomAnimationClonerImpl::getClonedShape() failed!" );
329 		}
330         return xSource;
331     }
332 
333 	Reference< XAnimationNode > CustomAnimationClonerImpl::getClonedNode( const Reference< XAnimationNode >& xSource ) const
334 	{
335 		sal_Int32 nNode, nNodeCount = maSourceNodeVector.size();
336 
337 		for( nNode = 0; nNode < nNodeCount; nNode++ )
338 		{
339 			if( maSourceNodeVector[nNode] == xSource )
340 				return maCloneNodeVector[nNode];
341 		}
342 
343 		DBG_ERROR( "sd::CustomAnimationClonerImpl::getClonedNode() failed!" );
344 		return xSource;
345 	}
346 }
347