xref: /trunk/main/oox/source/ppt/timenode.cxx (revision ca5ec200)
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 #include "oox/ppt/timenode.hxx"
25 
26 #include <boost/bind.hpp>
27 
28 #include <com/sun/star/beans/XPropertySet.hpp>
29 #include <com/sun/star/beans/NamedValue.hpp>
30 #include <com/sun/star/container/XEnumerationAccess.hpp>
31 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
32 #include <com/sun/star/frame/XModel.hpp>
33 #include <com/sun/star/animations/XAnimateColor.hpp>
34 #include <com/sun/star/animations/XAnimateMotion.hpp>
35 #include <com/sun/star/animations/XAnimateTransform.hpp>
36 #include <com/sun/star/animations/XCommand.hpp>
37 #include <com/sun/star/animations/XIterateContainer.hpp>
38 #include <com/sun/star/animations/XAnimationNodeSupplier.hpp>
39 #include <com/sun/star/animations/XTimeContainer.hpp>
40 #include <com/sun/star/animations/AnimationNodeType.hpp>
41 #include <com/sun/star/animations/Event.hpp>
42 #include <com/sun/star/animations/EventTrigger.hpp>
43 #include <com/sun/star/presentation/EffectNodeType.hpp>
44 
45 #include "oox/helper/helper.hxx"
46 #include "oox/core/xmlfilterbase.hxx"
47 
48 using ::rtl::OUString;
49 using namespace ::oox::core;
50 using namespace ::com::sun::star::beans;
51 using namespace ::com::sun::star::container;
52 using namespace ::com::sun::star::uno;
53 using namespace ::com::sun::star::lang;
54 using namespace ::com::sun::star::animations;
55 using namespace ::com::sun::star::frame;
56 using namespace ::com::sun::star::presentation;
57 
58 namespace oox { namespace ppt {
59 
getServiceName(sal_Int16 nNodeType)60 		OUString TimeNode::getServiceName( sal_Int16 nNodeType )
61 		{
62 			OUString sServiceName;
63 			switch( nNodeType )
64 			{
65 			case AnimationNodeType::PAR:
66 //				sServiceName = CREATE_OUSTRING("com.sun.star.animations.IterateContainer");
67 				sServiceName = CREATE_OUSTRING("com.sun.star.animations.ParallelTimeContainer");
68 				break;
69 			case AnimationNodeType::SEQ:
70 				sServiceName = CREATE_OUSTRING("com.sun.star.animations.SequenceTimeContainer");
71 				break;
72 			case AnimationNodeType::ANIMATE:
73 				sServiceName = CREATE_OUSTRING("com.sun.star.animations.Animate");
74 				break;
75 			case AnimationNodeType::ANIMATECOLOR:
76 				sServiceName = CREATE_OUSTRING("com.sun.star.animations.AnimateColor");
77 				break;
78 			case AnimationNodeType::TRANSITIONFILTER:
79 				sServiceName = CREATE_OUSTRING("com.sun.star.animations.TransitionFilter");
80 				break;
81 			case AnimationNodeType::ANIMATEMOTION:
82 				sServiceName = CREATE_OUSTRING("com.sun.star.animations.AnimateMotion");
83 				break;
84 			case AnimationNodeType::ANIMATETRANSFORM:
85 				sServiceName = CREATE_OUSTRING("com.sun.star.animations.AnimateTransform");
86 				break;
87 			case AnimationNodeType::COMMAND:
88 				sServiceName = CREATE_OUSTRING("com.sun.star.animations.Command");
89 				break;
90 			case AnimationNodeType::SET:
91 				sServiceName = CREATE_OUSTRING("com.sun.star.animations.AnimateSet");
92 				break;
93 			case AnimationNodeType::AUDIO:
94 				sServiceName = CREATE_OUSTRING("com.sun.star.animations.Audio");
95 				break;
96 			default:
97 				OSL_TRACE( "OOX: uhandled type %x", nNodeType );
98 				break;
99 			}
100 			return sServiceName;
101 		}
102 
103 
104 
TimeNode(sal_Int16 nNodeType)105 	TimeNode::TimeNode( sal_Int16 nNodeType )
106 		: mnNodeType( nNodeType )
107 		, mbHasEndSyncValue( false )
108 	{
109 	}
110 
111 
~TimeNode()112 	TimeNode::~TimeNode()
113 	{
114 	}
115 
116 // BEGIN CUT&PASTE from sd/source/filter/ppt/pptinanimations.hxx
117 // --------------------------------------------------------------------
fixMainSequenceTiming(const::com::sun::star::uno::Reference<::com::sun::star::animations::XAnimationNode> & xNode)118 	static void fixMainSequenceTiming( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xNode )
119 	{
120 		try
121 		{
122 			bool bFirst = true;
123 			Reference< XEnumerationAccess > xEA( xNode, UNO_QUERY_THROW );
124 			Reference< XEnumeration > xE( xEA->createEnumeration(), UNO_QUERY_THROW );
125 			while( xE->hasMoreElements() )
126 			{
127 				// click node
128 				Reference< XAnimationNode > xClickNode( xE->nextElement(), UNO_QUERY );
129 
130 				Event aEvent;
131 				aEvent.Trigger = EventTrigger::ON_NEXT;
132 				aEvent.Repeat = 0;
133 				xClickNode->setBegin( makeAny( aEvent ) );
134 
135 				if( bFirst )
136 				{
137 					bFirst = false;
138 					Reference< XEnumerationAccess > xEA2( xClickNode, UNO_QUERY_THROW );
139 					Reference< XEnumeration > xE2( xEA2->createEnumeration(), UNO_QUERY_THROW );
140 					if( xE2->hasMoreElements() )
141 					{
142 						// with node
143 						xE2->nextElement() >>= xEA2;
144 						if( xEA2.is() )
145 							xE2.query( xEA2->createEnumeration() );
146 						else
147 							xE2.clear();
148 
149 						if( xE2.is() && xE2->hasMoreElements() )
150 						{
151 							Reference< XAnimationNode > xEffectNode( xE2->nextElement(), UNO_QUERY_THROW );
152 							const Sequence< NamedValue > aUserData( xEffectNode->getUserData() );
153 							const NamedValue* p = aUserData.getConstArray();
154 							sal_Int32 nLength = aUserData.getLength();
155 							while( nLength-- )
156 							{
157 								if( p->Name.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "node-type" ) ) )
158 								{
159 									sal_Int16 nNodeType = 0;
160 									p->Value >>= nNodeType;
161 									if( nNodeType != ::com::sun::star::presentation::EffectNodeType::ON_CLICK )
162 									{
163 										// first effect does not start on click, so correct
164 										// first click nodes begin to 0s
165 										xClickNode->setBegin( makeAny( (double)0.0 ) );
166 										break;
167 									}
168 								}
169 								p++;
170 							}
171 						}
172 					}
173 				}
174 			}
175 		}
176 		catch( Exception& e )
177 		{
178 			(void)e;
179 			OSL_TRACE("fixMainSequenceTiming(), exception caught!" );
180 		}
181 	}
182 
183 // --------------------------------------------------------------------
184 
fixInteractiveSequenceTiming(const::com::sun::star::uno::Reference<::com::sun::star::animations::XAnimationNode> & xNode)185 	static void fixInteractiveSequenceTiming( const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xNode )
186 	{
187 		try
188 		{
189 			Any aBegin( xNode->getBegin() );
190 			Any aEmpty;
191 			xNode->setBegin( aEmpty );
192 
193 			Reference< XEnumerationAccess > xEA( xNode, UNO_QUERY_THROW );
194 			Reference< XEnumeration > xE( xEA->createEnumeration(), UNO_QUERY_THROW );
195 			while( xE->hasMoreElements() )
196 			{
197 				// click node
198 				Reference< XAnimationNode > xClickNode( xE->nextElement(), UNO_QUERY );
199 				xClickNode->setBegin( aBegin );
200 			}
201 		}
202 		catch( Exception& e )
203 		{
204 			(void)e;
205 			OSL_TRACE("fixInteractiveSequenceTiming(), exception caught!" );
206 		}
207 	}
208 
209 // END CUT&PASTE
210 
addNode(const XmlFilterBase & rFilter,const Reference<XAnimationNode> & rxNode,const SlidePersistPtr & pSlide)211     void TimeNode::addNode( const XmlFilterBase& rFilter, const Reference< XAnimationNode >& rxNode, const SlidePersistPtr & pSlide )
212 	{
213 		try {
214 			OUString sServiceName = getServiceName( mnNodeType );
215             Reference< XAnimationNode > xNode = createAndInsert( rFilter, sServiceName, rxNode );
216             setNode( rFilter, xNode, pSlide );
217 		}
218 		catch( const Exception& e )
219 		{
220 			OSL_TRACE( "OOX: exception raised in TimeNode::addNode() - %s",
221 								 OUStringToOString( e.Message, RTL_TEXTENCODING_ASCII_US ).getStr() );
222 		}
223 	}
224 
setNode(const XmlFilterBase & rFilter,const Reference<XAnimationNode> & xNode,const SlidePersistPtr & pSlide)225     void TimeNode::setNode( const XmlFilterBase& rFilter, const Reference< XAnimationNode >& xNode, const SlidePersistPtr & pSlide )
226 	{
227 		OSL_ENSURE( xNode.is(), "null node passed" );
228 
229 		try {
230 			if( msId.getLength() )
231 			{
232 				pSlide->getAnimNodesMap()[ msId ] = xNode;
233 			}
234 
235 			if( mpTarget )
236 			{
237 				sal_Int16 nSubType;
238 				maNodeProperties[ NP_TARGET ] = mpTarget->convert( pSlide, nSubType );
239 				if( mpTarget->mnType == XML_spTgt )
240 				{
241 					maNodeProperties[ NP_SUBITEM ] <<= nSubType;
242 				}
243 			}
244 
245 			if( !maStCondList.empty() )
246 			{
247 				Any aAny = AnimationCondition::convertList( pSlide, maStCondList );
248  				if( aAny.hasValue() )
249 				{
250 					xNode->setBegin( aAny );
251 				}
252 
253 			}
254 			if( !maEndCondList.empty() )
255 			{
256 				Any aAny = AnimationCondition::convertList( pSlide, maEndCondList );
257 				if( aAny.hasValue() )
258 				{
259 					xNode->setEnd( aAny );
260 				}
261 			}
262 #if 0  // FIXME even the binary filter has this disabled.
263 			if( !maNextCondList.empty() )
264 			{
265 				Any aAny = AnimationCondition::convertList( pSlide, maNextCondList );
266 				if( aAny.hasValue() )
267 				{
268 					xNode->setNext( aAny );
269 				}
270 			}
271 			if( !maPrevCondList.empty() )
272 			{
273 				Any aAny = AnimationCondition::convertList( pSlide, maPrevCondList );
274 				if( aAny.hasValue() )
275 				{
276 					xNode->setPrev( aAny );
277 				}
278 			}
279 #endif
280 			if( mbHasEndSyncValue )
281 			{
282 				Any aValue = maEndSyncValue.convert( pSlide );
283 				xNode->setEndSync(aValue);
284 			}
285 
286             if( !maUserData.empty() )
287             {
288                 Sequence< NamedValue > aUserDataSeq( static_cast< sal_Int32 >( maUserData.size() ) );
289                 NamedValue* pValues = aUserDataSeq.getArray();
290                 for( UserDataMap::const_iterator aIt = maUserData.begin(), aEnd = maUserData.end(); aIt != aEnd; ++aIt, ++pValues )
291                 {
292                     pValues->Name = aIt->first;
293                     pValues->Value = aIt->second;
294                 }
295                 maNodeProperties[ NP_USERDATA ] <<= aUserDataSeq;
296 			}
297 
298 			Reference< XAnimate > xAnimate( xNode, UNO_QUERY );
299 			Reference< XAnimateColor > xAnimateColor( xNode, UNO_QUERY );
300 			Reference< XAnimateMotion > xAnimateMotion( xNode, UNO_QUERY );
301 			Reference< XAnimateTransform > xAnimateTransform( xNode, UNO_QUERY );
302 			Reference< XCommand > xCommand( xNode, UNO_QUERY );
303 			Reference< XIterateContainer > xIterateContainer( xNode, UNO_QUERY );
304 			sal_Int16 nInt16 = 0;
305 			sal_Bool bBool = sal_False;
306 			double fDouble = 0;
307 			OUString sString;
308 			Sequence< NamedValue > aSeq;
309 
310 			for( int i = 0; i < _NP_SIZE; i++)
311 			{
312 				Any & aValue( maNodeProperties[ i ] );
313 				if( aValue.hasValue() )
314 				{
315 					switch( i )
316 					{
317 					case NP_TO:
318 						if( xAnimate.is() )
319 							xAnimate->setTo( aValue );
320 						break;
321 					case NP_FROM:
322 						if( xAnimate.is() )
323 							xAnimate->setFrom( aValue );
324 						break;
325 					case NP_BY:
326 						if( xAnimate.is() )
327 							xAnimate->setBy( aValue );
328 						break;
329 					case NP_TARGET:
330 						if( xAnimate.is() )
331 							xAnimate->setTarget( aValue );
332 						break;
333 					case NP_SUBITEM:
334 						if( xAnimate.is() )
335 						{
336 							if( aValue >>= nInt16 )
337 								xAnimate->setSubItem( nInt16 );
338 							else
339 							{
340 								OSL_TRACE( "any >>= failed %d", __LINE__ );
341 							}
342 						}
343 						break;
344 					case NP_ATTRIBUTENAME:
345 						if( xAnimate.is() )
346 						{
347 							if(	aValue >>= sString )
348 								xAnimate->setAttributeName( sString );
349 							else
350 							{
351 								OSL_TRACE( "any >>= failed %d", __LINE__ );
352 							}
353 						}
354 						break;
355 					case NP_CALCMODE:
356 						if( xAnimate.is() )
357 						{
358 							if( aValue >>= nInt16 )
359 								xAnimate->setCalcMode( nInt16 );
360 							else
361 							{
362 								OSL_TRACE( "any >>= failed %d", __LINE__ );
363 							}
364 						}
365 						break;
366 					case NP_KEYTIMES:
367 						if( xAnimate.is() )
368 						{
369 							Sequence<double> aKeyTimes;
370 							if( aValue >>= aKeyTimes )
371 								xAnimate->setKeyTimes(aKeyTimes);
372 							else
373 							{
374 								OSL_TRACE( "any >>= failed %d", __LINE__ );
375 							}
376 						}
377 						break;
378 					case NP_VALUES:
379 						if( xAnimate.is() )
380 						{
381 							Sequence<Any> aValues;
382 							if( aValue >>= aValues )
383 								xAnimate->setValues(aValues);
384 							else
385 							{
386 								OSL_TRACE( "any >>= failed %d", __LINE__ );
387 							}
388 						}
389 						break;
390 					case NP_FORMULA:
391 						if( xAnimate.is() )
392 						{
393 							if( aValue >>= sString )
394 								xAnimate->setFormula(sString);
395 							else
396 							{
397 								OSL_TRACE( "any >>= failed %d", __LINE__ );
398 							}
399 						}
400 						break;
401 					case NP_COLORINTERPOLATION:
402 						if( xAnimateColor.is() )
403 						{
404 							if( aValue >>= nInt16 )
405 								xAnimateColor->setColorInterpolation( nInt16 );
406 							else
407 							{
408 								OSL_TRACE( "any >>= failed %d", __LINE__ );
409 							}
410 						}
411 						break;
412 					case NP_DIRECTION:
413 						if( xAnimateColor.is() )
414 						{
415 							if( aValue >>= bBool )
416 								xAnimateColor->setDirection( bBool );
417 							else
418 							{
419 								OSL_TRACE( "any >>= failed %d", __LINE__ );
420 							}
421 						}
422 						break;
423 					case NP_PATH:
424 						if( xAnimateMotion.is() )
425 							xAnimateMotion->setPath( aValue );
426 						break;
427 					case NP_TRANSFORMTYPE:
428 						if( xAnimateTransform.is() )
429 						{
430 							if( aValue >>= nInt16 )
431 								xAnimateTransform->setTransformType( nInt16 );
432 							else
433 							{
434 								OSL_TRACE( "any >>= failed %d", __LINE__ );
435 							}
436 						}
437 						break;
438 					case NP_USERDATA:
439 						if( aValue >>= aSeq )
440 							xNode->setUserData( aSeq );
441 						else
442 						{
443 							OSL_TRACE( "any >>= failed %d", __LINE__ );
444 						}
445 						break;
446 					case NP_ACCELERATION:
447 						if( aValue >>= fDouble )
448 							xNode->setAcceleration( fDouble );
449 						else
450 						{
451 							OSL_TRACE( "any >>= failed %d", __LINE__ );
452 						}
453 						break;
454 					case NP_DECELERATE:
455 						if( aValue >>= fDouble )
456 							xNode->setDecelerate( fDouble );
457 						else
458 						{
459 							OSL_TRACE( "any >>= failed %d", __LINE__ );
460 						}
461 						break;
462 					case NP_AUTOREVERSE:
463 						if( aValue >>= bBool )
464 							xNode->setAutoReverse( bBool );
465 						else
466 						{
467 							OSL_TRACE( "any >>= failed %d", __LINE__ );
468 						}
469 						break;
470 					case NP_DURATION:
471 						xNode->setDuration( aValue );
472 						break;
473 					case NP_FILL:
474 						if( aValue >>= nInt16 )
475 							xNode->setFill( nInt16 );
476 						else
477 						{
478 							OSL_TRACE( "any >>= failed %d", __LINE__ );
479 						}
480 						break;
481 					case NP_REPEATCOUNT:
482 						xNode->setRepeatCount( aValue );
483 						break;
484 					case NP_REPEATDURATION:
485 						xNode->setRepeatDuration( aValue );
486 						break;
487 					case NP_RESTART:
488 						if( aValue >>= nInt16 )
489 							xNode->setRestart( nInt16 );
490 						else
491 						{
492 							OSL_TRACE( "any >>= failed %d", __LINE__ );
493 						}
494 						break;
495 					case NP_COMMAND:
496 						if( xCommand.is() )
497 						{
498 							if( aValue >>= nInt16 )
499 								xCommand->setCommand( nInt16 );
500 							else
501 							{
502 								OSL_TRACE( "any >>= failed %d", __LINE__ );
503 							}
504 						}
505 						break;
506 					case NP_PARAMETER:
507 						if( xCommand.is() )
508 							xCommand->setParameter( aValue );
509 						break;
510 					case NP_ITERATETYPE:
511 						if( xIterateContainer.is() )
512 						{
513 							if( aValue >>= nInt16 )
514 								xIterateContainer->setIterateType( nInt16 );
515 							else
516 							{
517 								OSL_TRACE( "any >>= failed %d", __LINE__ );
518 							}
519 						}
520 						break;
521 					case NP_ITERATEINTERVAL:
522 						if( xIterateContainer.is() )
523 						{
524 							if( aValue >>= fDouble )
525 								xIterateContainer->setIterateInterval( fDouble );
526 							else
527 							{
528 								OSL_TRACE( "any >>= failed %d", __LINE__ );
529 							}
530 						}
531 						break;
532 					default:
533 						OSL_TRACE( "ERR-OOX: unknown prop index %d", i );
534 						break;
535 					}
536 				}
537 			}
538 
539 			if( mnNodeType == AnimationNodeType::TRANSITIONFILTER )
540 			{
541 
542 				Reference< XTransitionFilter > xFilter( xNode, UNO_QUERY );
543 				maTransitionFilter.setTransitionFilterProperties( xFilter );
544 			}
545 
546             std::for_each( maChildren.begin(), maChildren.end(),
547                            boost::bind(&TimeNode::addNode, _1, boost::cref(rFilter), boost::ref(xNode),
548 									   boost::ref(pSlide) ) );
549 
550 			switch( mnNodeType )
551 			{
552 			case AnimationNodeType::SEQ:
553 			{
554 				sal_Int16 nEnum = 0;
555 				if( maUserData[ CREATE_OUSTRING( "node-type" ) ] >>= nEnum )
556 				{
557 					if( nEnum == EffectNodeType::MAIN_SEQUENCE )
558 					{
559 						fixMainSequenceTiming( xNode );
560 					}
561 					else if( nEnum ==  EffectNodeType::INTERACTIVE_SEQUENCE )
562 					{
563 						fixInteractiveSequenceTiming( xNode );
564 					}
565 				}
566 				break;
567 			}
568 			case AnimationNodeType::PAR:
569 				// some other cut&paste... from AnimationImporter::importAnimationContainer()
570 				break;
571 			}
572 		}
573 		catch( const Exception& e )
574 		{
575 			OSL_TRACE( "OOX: exception raised in TimeNode::setNode() - %s",
576 								 OUStringToOString( e.Message, RTL_TEXTENCODING_ASCII_US ).getStr() );
577 		}
578 	}
579 
580 
createAndInsert(const XmlFilterBase & rFilter,const OUString & rServiceName,const Reference<XAnimationNode> & rxNode)581     Reference< XAnimationNode > TimeNode::createAndInsert(
582             const XmlFilterBase& rFilter,
583             const OUString& rServiceName,
584             const Reference< XAnimationNode >& rxNode )
585 	{
586 		try {
587             Reference< XAnimationNode > xNode ( rFilter.getServiceFactory()->createInstance( rServiceName ), UNO_QUERY_THROW );
588 			Reference< XTimeContainer > xParentContainer( rxNode, UNO_QUERY_THROW );
589 
590 			xParentContainer->appendChild( xNode );
591 			return xNode;
592 		}
593 		catch( const Exception& e )
594 		{
595 			OSL_TRACE( "OOX: exception raised in TimeNode::createAndInsert() trying to create a service %s = %s",
596 								 OUStringToOString( rServiceName, RTL_TEXTENCODING_ASCII_US ).getStr(),
597 								 OUStringToOString( e.Message, RTL_TEXTENCODING_ASCII_US ).getStr() );
598 		}
599 
600 		return Reference< XAnimationNode >();
601 	}
602 
603 
setId(sal_Int32 nId)604 	void 	TimeNode::setId( sal_Int32 nId )
605 	{
606 		msId = OUString::valueOf(nId);
607 	}
608 
setTo(const Any & aTo)609 	void TimeNode::setTo( const Any & aTo )
610 	{
611 		maNodeProperties[ NP_TO ] = aTo;
612 	}
613 
614 
setFrom(const Any & aFrom)615 	void TimeNode::setFrom( const Any & aFrom )
616 	{
617 		maNodeProperties[ NP_FROM ] = aFrom;
618 	}
619 
setBy(const Any & aBy)620 	void TimeNode::setBy( const Any & aBy )
621 	{
622 		maNodeProperties[ NP_BY ] = aBy;
623 	}
624 
625 
626 } }
627