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 27 //#include <com/sun/star/frame/XController.hpp> 28 #include <com/sun/star/beans/PropertyAttribute.hpp> 29 #include <com/sun/star/drawing/framework/XControllerManager.hpp> 30 #include <com/sun/star/container/XIndexAccess.hpp> 31 #include <comphelper/serviceinfohelper.hxx> 32 33 #include <cppuhelper/bootstrap.hxx> 34 35 #include <comphelper/processfactory.hxx> 36 #include <vos/mutex.hxx> 37 38 #include <vcl/svapp.hxx> 39 #include <vcl/wrkwin.hxx> 40 #include <svx/svdpool.hxx> 41 #include <svl/itemprop.hxx> 42 43 #include <sfx2/viewfrm.hxx> 44 45 #include <toolkit/unohlp.hxx> 46 #include <svx/unoprov.hxx> 47 48 #include "framework/FrameworkHelper.hxx" 49 50 #include "FrameView.hxx" 51 #include "unomodel.hxx" 52 #include "slideshow.hxx" 53 #include "slideshowimpl.hxx" 54 #include "sdattr.hrc" 55 #include "FactoryIds.hxx" 56 #include "ViewShell.hxx" 57 #include "SlideShowRestarter.hxx" 58 #include "DrawController.hxx" 59 #include <boost/bind.hpp> 60 61 using ::com::sun::star::presentation::XSlideShowController; 62 using ::com::sun::star::container::XIndexAccess; 63 using ::sd::framework::FrameworkHelper; 64 using ::rtl::OUString; 65 using ::com::sun::star::awt::XWindow; 66 using namespace ::sd; 67 using namespace ::cppu; 68 using namespace ::vos; 69 using namespace ::com::sun::star::uno; 70 using namespace ::com::sun::star::presentation; 71 using namespace ::com::sun::star::drawing; 72 using namespace ::com::sun::star::beans; 73 using namespace ::com::sun::star::lang; 74 using namespace ::com::sun::star::animations; 75 using namespace ::com::sun::star::drawing::framework; 76 77 extern String getUiNameFromPageApiNameImpl( const ::rtl::OUString& rApiName ); 78 79 #define C2U(x) OUString( RTL_CONSTASCII_USTRINGPARAM(x) ) 80 81 82 namespace { 83 /** This local version of the work window overloads DataChanged() so that it 84 can restart the slide show when a display is added or removed. 85 */ 86 class FullScreenWorkWindow : public WorkWindow 87 { 88 public: 89 FullScreenWorkWindow ( 90 const ::rtl::Reference<SlideShow>& rpSlideShow, 91 ViewShellBase* pViewShellBase) 92 : WorkWindow(NULL, WB_HIDE | WB_CLIPCHILDREN), 93 mpRestarter(new SlideShowRestarter(rpSlideShow, pViewShellBase)) 94 {} 95 96 97 virtual void DataChanged (const DataChangedEvent& rEvent) 98 { 99 if (rEvent.GetType() == DATACHANGED_DISPLAY) 100 { 101 mpRestarter->Restart(); 102 } 103 } 104 105 private: 106 ::boost::shared_ptr<SlideShowRestarter> mpRestarter; 107 }; 108 109 /** Return the default display id (or -1 when that can not be 110 determined.) 111 */ 112 sal_Int32 GetDefaultDisplay (void) 113 { 114 try 115 { 116 Reference< XMultiServiceFactory > xFactory(::comphelper::getProcessServiceFactory(), UNO_QUERY_THROW ); 117 Reference< XPropertySet > xMonProps(xFactory->createInstance(OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.awt.DisplayAccess" ) ) ), UNO_QUERY_THROW ); 118 const OUString sPropertyName( RTL_CONSTASCII_USTRINGPARAM( "DefaultDisplay" ) ); 119 sal_Int32 nPrimaryIndex (-1); 120 if (xMonProps->getPropertyValue( sPropertyName ) >>= nPrimaryIndex) 121 return nPrimaryIndex; 122 } 123 catch( Exception& ) 124 { 125 } 126 return -1; 127 } 128 } 129 130 131 ////////////////////////////////////////////////////////////////////////////// 132 // -------------------------------------------------------------------- 133 134 const SfxItemPropertyMapEntry* ImplGetPresentationPropertyMap() 135 { 136 // NOTE: First member must be sorted 137 static const SfxItemPropertyMapEntry aPresentationPropertyMap_Impl[] = 138 { 139 { MAP_CHAR_LEN("AllowAnimations"), ATTR_PRESENT_ANIMATION_ALLOWED, &::getBooleanCppuType(), 0, 0 }, 140 { MAP_CHAR_LEN("CustomShow"), ATTR_PRESENT_CUSTOMSHOW, &::getCppuType((const OUString*)0), 0, 0 }, 141 { MAP_CHAR_LEN("Display"), ATTR_PRESENT_DISPLAY, &::getCppuType((const sal_Int32*)0), 0, 0 }, 142 { MAP_CHAR_LEN("FirstPage"), ATTR_PRESENT_DIANAME, &::getCppuType((const OUString*)0), 0, 0 }, 143 { MAP_CHAR_LEN("IsAlwaysOnTop"), ATTR_PRESENT_ALWAYS_ON_TOP, &::getBooleanCppuType(), 0, 0 }, 144 { MAP_CHAR_LEN("IsAutomatic"), ATTR_PRESENT_MANUEL, &::getBooleanCppuType(), 0, 0 }, 145 { MAP_CHAR_LEN("IsEndless"), ATTR_PRESENT_ENDLESS, &::getBooleanCppuType(), 0, 0 }, 146 { MAP_CHAR_LEN("IsFullScreen"), ATTR_PRESENT_FULLSCREEN, &::getBooleanCppuType(), 0, 0 }, 147 { MAP_CHAR_LEN("IsShowAll"), ATTR_PRESENT_ALL, &::getBooleanCppuType(), 0, 0 }, 148 { MAP_CHAR_LEN("IsMouseVisible"), ATTR_PRESENT_MOUSE, &::getBooleanCppuType(), 0, 0 }, 149 { MAP_CHAR_LEN("IsShowLogo"), ATTR_PRESENT_SHOW_PAUSELOGO, &::getBooleanCppuType(), 0, 0 }, 150 { MAP_CHAR_LEN("IsTransitionOnClick"), ATTR_PRESENT_CHANGE_PAGE, &::getBooleanCppuType(), 0, 0 }, 151 { MAP_CHAR_LEN("Pause"), ATTR_PRESENT_PAUSE_TIMEOUT, &::getCppuType((const sal_Int32*)0), 0, 0 }, 152 { MAP_CHAR_LEN("StartWithNavigator"), ATTR_PRESENT_NAVIGATOR, &::getBooleanCppuType(), 0, 0 }, 153 { MAP_CHAR_LEN("UsePen"), ATTR_PRESENT_PEN, &::getBooleanCppuType(), 0, 0 }, 154 { 0,0,0,0,0,0} 155 }; 156 157 return aPresentationPropertyMap_Impl; 158 } 159 160 //SfxItemPropertyMap map_impl[] = { { 0,0,0,0,0,0 } }; 161 162 // -------------------------------------------------------------------- 163 // class SlideShow 164 // -------------------------------------------------------------------- 165 166 SlideShow::SlideShow( SdDrawDocument* pDoc ) 167 : SlideshowBase( m_aMutex ) 168 , maPropSet(ImplGetPresentationPropertyMap(), SdrObject::GetGlobalDrawObjectItemPool()) 169 , mbIsInStartup(false) 170 , mpDoc( pDoc ) 171 , mpCurrentViewShellBase( 0 ) 172 , mpFullScreenViewShellBase( 0 ) 173 , mpFullScreenFrameView( 0 ) 174 , mnInPlaceConfigEvent( 0 ) 175 { 176 } 177 178 // -------------------------------------------------------------------- 179 180 void SlideShow::ThrowIfDisposed() throw (RuntimeException) 181 { 182 if( mpDoc == 0 ) 183 throw DisposedException(); 184 } 185 186 // -------------------------------------------------------------------- 187 188 /// used by the model to create a slideshow for it 189 rtl::Reference< SlideShow > SlideShow::Create( SdDrawDocument* pDoc ) 190 { 191 return new SlideShow( pDoc ); 192 } 193 194 // -------------------------------------------------------------------- 195 196 rtl::Reference< SlideShow > SlideShow::GetSlideShow( SdDrawDocument* pDocument ) 197 { 198 rtl::Reference< SlideShow > xRet; 199 200 if( pDocument ) 201 xRet = rtl::Reference< SlideShow >( dynamic_cast< SlideShow* >( pDocument->getPresentation().get() ) ); 202 203 return xRet; 204 } 205 206 // -------------------------------------------------------------------- 207 208 rtl::Reference< SlideShow > SlideShow::GetSlideShow( ViewShellBase& rBase ) 209 { 210 return GetSlideShow( rBase.GetDocument() ); 211 } 212 213 // -------------------------------------------------------------------- 214 215 ::com::sun::star::uno::Reference< ::com::sun::star::presentation::XSlideShowController > SlideShow::GetSlideShowController(ViewShellBase& rBase ) 216 { 217 rtl::Reference< SlideShow > xSlideShow( GetSlideShow( rBase ) ); 218 219 Reference< XSlideShowController > xRet; 220 if( xSlideShow.is() ) 221 xRet = xSlideShow->getController(); 222 223 return xRet; 224 } 225 226 // -------------------------------------------------------------------- 227 228 bool SlideShow::StartPreview( ViewShellBase& rBase, 229 const ::com::sun::star::uno::Reference< ::com::sun::star::drawing::XDrawPage >& xDrawPage, 230 const ::com::sun::star::uno::Reference< ::com::sun::star::animations::XAnimationNode >& xAnimationNode, 231 ::Window* pParent /* = 0 */ ) 232 { 233 rtl::Reference< SlideShow > xSlideShow( GetSlideShow( rBase ) ); 234 if( xSlideShow.is() ) 235 return xSlideShow->startPreview( xDrawPage, xAnimationNode, pParent ); 236 237 return false; 238 } 239 240 // -------------------------------------------------------------------- 241 242 void SlideShow::Stop( ViewShellBase& rBase ) 243 { 244 rtl::Reference< SlideShow > xSlideShow( GetSlideShow( rBase ) ); 245 if( xSlideShow.is() ) 246 xSlideShow->end(); 247 } 248 249 // -------------------------------------------------------------------- 250 251 bool SlideShow::IsRunning( ViewShellBase& rBase ) 252 { 253 rtl::Reference< SlideShow > xSlideShow( GetSlideShow( rBase ) ); 254 return xSlideShow.is() && xSlideShow->isRunning(); 255 } 256 257 // -------------------------------------------------------------------- 258 259 bool SlideShow::IsRunning( ViewShell& rViewShell ) 260 { 261 rtl::Reference< SlideShow > xSlideShow( GetSlideShow( rViewShell.GetViewShellBase() ) ); 262 return xSlideShow.is() && xSlideShow->isRunning() && (xSlideShow->mxController->getViewShell() == &rViewShell); 263 } 264 265 // -------------------------------------------------------------------- 266 267 void SlideShow::CreateController( ViewShell* pViewSh, ::sd::View* pView, ::Window* pParentWindow ) 268 { 269 DBG_ASSERT( !mxController.is(), "sd::SlideShow::CreateController(), clean up old controller first!" ); 270 271 Reference< XPresentation2 > xThis( this ); 272 273 rtl::Reference<SlideshowImpl> xController ( 274 new SlideshowImpl(xThis, pViewSh, pView, mpDoc, pParentWindow)); 275 276 // Reset mbIsInStartup. From here mxController.is() is used to prevent 277 // multiple slide show instances for one document. 278 mxController = xController; 279 mbIsInStartup = false; 280 } 281 282 // -------------------------------------------------------------------- 283 // XServiceInfo 284 // -------------------------------------------------------------------- 285 286 OUString SAL_CALL SlideShow::getImplementationName( ) throw(RuntimeException) 287 { 288 return OUString( RTL_CONSTASCII_USTRINGPARAM("com.sun.star.comp.sd.SlideShow") ); 289 } 290 291 // -------------------------------------------------------------------- 292 293 sal_Bool SAL_CALL SlideShow::supportsService( const OUString& ServiceName ) throw(RuntimeException) 294 { 295 return comphelper::ServiceInfoHelper::supportsService( ServiceName, getSupportedServiceNames( ) ); 296 } 297 298 // -------------------------------------------------------------------- 299 300 Sequence< OUString > SAL_CALL SlideShow::getSupportedServiceNames( ) throw(RuntimeException) 301 { 302 OUString aService( RTL_CONSTASCII_USTRINGPARAM("com.sun.star.presentation.Presentation") ); 303 Sequence< OUString > aSeq( &aService, 1 ); 304 return aSeq; 305 } 306 307 // -------------------------------------------------------------------- 308 // XPropertySet 309 // -------------------------------------------------------------------- 310 311 Reference< XPropertySetInfo > SAL_CALL SlideShow::getPropertySetInfo() throw(RuntimeException) 312 { 313 OGuard aGuard( Application::GetSolarMutex() ); 314 static Reference< XPropertySetInfo > xInfo = maPropSet.getPropertySetInfo(); 315 return xInfo; 316 } 317 318 // -------------------------------------------------------------------- 319 320 void SAL_CALL SlideShow::setPropertyValue( const OUString& aPropertyName, const Any& aValue ) throw(UnknownPropertyException, PropertyVetoException, IllegalArgumentException, WrappedTargetException, RuntimeException) 321 { 322 OGuard aGuard( Application::GetSolarMutex() ); 323 ThrowIfDisposed(); 324 325 sd::PresentationSettings& rPresSettings = mpDoc->getPresentationSettings(); 326 327 const SfxItemPropertySimpleEntry* pEntry = maPropSet.getPropertyMapEntry(aPropertyName); 328 329 if( pEntry && ((pEntry->nFlags & PropertyAttribute::READONLY) != 0) ) 330 throw PropertyVetoException(); 331 332 bool bValuesChanged = false; 333 bool bIllegalArgument = true; 334 335 switch( pEntry ? pEntry->nWID : -1 ) 336 { 337 case ATTR_PRESENT_ALL: 338 { 339 sal_Bool bVal = sal_False; 340 341 if( aValue >>= bVal ) 342 { 343 bIllegalArgument = false; 344 345 if( rPresSettings.mbAll != bVal ) 346 { 347 rPresSettings.mbAll = bVal; 348 bValuesChanged = true; 349 if( bVal ) 350 rPresSettings.mbCustomShow = sal_False; 351 } 352 } 353 break; 354 } 355 case ATTR_PRESENT_CHANGE_PAGE: 356 { 357 sal_Bool bVal = sal_False; 358 359 if( aValue >>= bVal ) 360 { 361 bIllegalArgument = false; 362 363 if( bVal == rPresSettings.mbLockedPages ) 364 { 365 bValuesChanged = true; 366 rPresSettings.mbLockedPages = !bVal; 367 } 368 } 369 break; 370 } 371 372 case ATTR_PRESENT_ANIMATION_ALLOWED: 373 { 374 sal_Bool bVal = sal_False; 375 376 if( aValue >>= bVal ) 377 { 378 bIllegalArgument = false; 379 380 if(rPresSettings.mbAnimationAllowed != bVal) 381 { 382 bValuesChanged = true; 383 rPresSettings.mbAnimationAllowed = bVal; 384 } 385 } 386 break; 387 } 388 case ATTR_PRESENT_CUSTOMSHOW: 389 { 390 OUString aShow; 391 if( aValue >>= aShow ) 392 { 393 bIllegalArgument = false; 394 395 const String aShowName( aShow ); 396 397 List* pCustomShowList = mpDoc->GetCustomShowList(sal_False); 398 if(pCustomShowList) 399 { 400 SdCustomShow* pCustomShow; 401 for( pCustomShow = (SdCustomShow*) pCustomShowList->First(); pCustomShow != NULL; pCustomShow = (SdCustomShow*) pCustomShowList->Next() ) 402 { 403 if( pCustomShow->GetName() == aShowName ) 404 break; 405 } 406 407 rPresSettings.mbCustomShow = sal_True; 408 bValuesChanged = true; 409 } 410 } 411 break; 412 } 413 case ATTR_PRESENT_ENDLESS: 414 { 415 sal_Bool bVal = sal_False; 416 417 if( aValue >>= bVal ) 418 { 419 bIllegalArgument = false; 420 421 if( rPresSettings.mbEndless != bVal) 422 { 423 bValuesChanged = true; 424 rPresSettings.mbEndless = bVal; 425 } 426 } 427 break; 428 } 429 case ATTR_PRESENT_FULLSCREEN: 430 { 431 sal_Bool bVal = sal_False; 432 433 if( aValue >>= bVal ) 434 { 435 bIllegalArgument = false; 436 if( rPresSettings.mbFullScreen != bVal) 437 { 438 bValuesChanged = true; 439 rPresSettings.mbFullScreen = bVal; 440 } 441 } 442 break; 443 } 444 case ATTR_PRESENT_DIANAME: 445 { 446 OUString aPresPage; 447 aValue >>= aPresPage; 448 bIllegalArgument = false; 449 if( (rPresSettings.maPresPage != aPresPage) || !rPresSettings.mbCustomShow || !rPresSettings.mbAll ) 450 { 451 bValuesChanged = true; 452 rPresSettings.maPresPage = getUiNameFromPageApiNameImpl(aPresPage); 453 rPresSettings.mbCustomShow = sal_False; 454 rPresSettings.mbAll = sal_False; 455 } 456 break; 457 } 458 case ATTR_PRESENT_MANUEL: 459 { 460 sal_Bool bVal = sal_False; 461 462 if( aValue >>= bVal ) 463 { 464 bIllegalArgument = false; 465 466 if( rPresSettings.mbManual != bVal) 467 { 468 bValuesChanged = true; 469 rPresSettings.mbManual = bVal; 470 } 471 } 472 break; 473 } 474 case ATTR_PRESENT_MOUSE: 475 { 476 sal_Bool bVal = sal_False; 477 478 if( aValue >>= bVal ) 479 { 480 bIllegalArgument = false; 481 if( rPresSettings.mbMouseVisible != bVal) 482 { 483 bValuesChanged = true; 484 rPresSettings.mbMouseVisible = bVal; 485 } 486 } 487 break; 488 } 489 case ATTR_PRESENT_ALWAYS_ON_TOP: 490 { 491 sal_Bool bVal = sal_False; 492 493 if( aValue >>= bVal ) 494 { 495 bIllegalArgument = false; 496 497 if( rPresSettings.mbAlwaysOnTop != bVal) 498 { 499 bValuesChanged = true; 500 rPresSettings.mbAlwaysOnTop = bVal; 501 } 502 } 503 break; 504 } 505 case ATTR_PRESENT_NAVIGATOR: 506 { 507 sal_Bool bVal = sal_False; 508 509 if( aValue >>= bVal ) 510 { 511 bIllegalArgument = false; 512 513 if( rPresSettings.mbStartWithNavigator != bVal) 514 { 515 bValuesChanged = true; 516 rPresSettings.mbStartWithNavigator = bVal; 517 } 518 } 519 break; 520 } 521 case ATTR_PRESENT_PEN: 522 { 523 sal_Bool bVal = sal_False; 524 525 if( aValue >>= bVal ) 526 { 527 bIllegalArgument = false; 528 529 if(rPresSettings.mbMouseAsPen != bVal) 530 { 531 bValuesChanged = true; 532 rPresSettings.mbMouseAsPen = bVal; 533 } 534 } 535 break; 536 } 537 case ATTR_PRESENT_PAUSE_TIMEOUT: 538 { 539 sal_Int32 nValue = 0; 540 if( (aValue >>= nValue) && (nValue >= 0) ) 541 { 542 bIllegalArgument = false; 543 if( rPresSettings.mnPauseTimeout != nValue ) 544 { 545 bValuesChanged = true; 546 rPresSettings.mnPauseTimeout = nValue; 547 } 548 } 549 break; 550 } 551 case ATTR_PRESENT_SHOW_PAUSELOGO: 552 { 553 sal_Bool bVal = sal_False; 554 555 if( aValue >>= bVal ) 556 { 557 bIllegalArgument = false; 558 559 if( rPresSettings.mbShowPauseLogo != bVal ) 560 { 561 bValuesChanged = true; 562 rPresSettings.mbShowPauseLogo = bVal; 563 } 564 } 565 break; 566 } 567 case ATTR_PRESENT_DISPLAY: 568 { 569 sal_Int32 nDisplay = 0; 570 if( aValue >>= nDisplay ) 571 { 572 // Convert value to true display id. 573 if (nDisplay == 0) 574 nDisplay = GetDefaultDisplay(); 575 else if (nDisplay < 0) 576 nDisplay = -1; 577 else 578 --nDisplay; 579 580 bIllegalArgument = false; 581 582 SdOptions* pOptions = SD_MOD()->GetSdOptions(DOCUMENT_TYPE_IMPRESS); 583 pOptions->SetDisplay( nDisplay ); 584 } 585 break; 586 } 587 588 default: 589 throw UnknownPropertyException(); 590 } 591 592 if( bIllegalArgument ) 593 throw IllegalArgumentException(); 594 595 if( bValuesChanged ) 596 mpDoc->SetChanged( true ); 597 } 598 599 // -------------------------------------------------------------------- 600 601 Any SAL_CALL SlideShow::getPropertyValue( const OUString& PropertyName ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) 602 { 603 OGuard aGuard( Application::GetSolarMutex() ); 604 ThrowIfDisposed(); 605 606 const sd::PresentationSettings& rPresSettings = mpDoc->getPresentationSettings(); 607 608 const SfxItemPropertySimpleEntry* pEntry = maPropSet.getPropertyMapEntry(PropertyName); 609 610 switch( pEntry ? pEntry->nWID : -1 ) 611 { 612 case ATTR_PRESENT_ALL: 613 return Any( (sal_Bool) ( !rPresSettings.mbCustomShow && rPresSettings.mbAll ) ); 614 case ATTR_PRESENT_CHANGE_PAGE: 615 return Any( (sal_Bool) !rPresSettings.mbLockedPages ); 616 case ATTR_PRESENT_ANIMATION_ALLOWED: 617 return Any( rPresSettings.mbAnimationAllowed ); 618 case ATTR_PRESENT_CUSTOMSHOW: 619 { 620 List* pList = mpDoc->GetCustomShowList(sal_False); 621 SdCustomShow* pShow = (pList && rPresSettings.mbCustomShow)?(SdCustomShow*)pList->GetCurObject():NULL; 622 OUString aShowName; 623 624 if(pShow) 625 aShowName = pShow->GetName(); 626 627 return Any( aShowName ); 628 } 629 case ATTR_PRESENT_ENDLESS: 630 return Any( rPresSettings.mbEndless ); 631 case ATTR_PRESENT_FULLSCREEN: 632 return Any( rPresSettings.mbFullScreen ); 633 case ATTR_PRESENT_DIANAME: 634 { 635 OUString aSlideName; 636 637 if( !rPresSettings.mbCustomShow && !rPresSettings.mbAll ) 638 aSlideName = getPageApiNameFromUiName( rPresSettings.maPresPage ); 639 640 return Any( aSlideName ); 641 } 642 case ATTR_PRESENT_MANUEL: 643 return Any( rPresSettings.mbManual ); 644 case ATTR_PRESENT_MOUSE: 645 return Any( rPresSettings.mbMouseVisible ); 646 case ATTR_PRESENT_ALWAYS_ON_TOP: 647 return Any( rPresSettings.mbAlwaysOnTop ); 648 case ATTR_PRESENT_NAVIGATOR: 649 return Any( rPresSettings.mbStartWithNavigator ); 650 case ATTR_PRESENT_PEN: 651 return Any( rPresSettings.mbMouseAsPen ); 652 case ATTR_PRESENT_PAUSE_TIMEOUT: 653 return Any( rPresSettings.mnPauseTimeout ); 654 case ATTR_PRESENT_SHOW_PAUSELOGO: 655 return Any( rPresSettings.mbShowPauseLogo ); 656 case ATTR_PRESENT_DISPLAY: 657 { 658 SdOptions* pOptions = SD_MOD()->GetSdOptions(DOCUMENT_TYPE_IMPRESS); 659 const sal_Int32 nDisplay (pOptions->GetDisplay()); 660 // Convert true display id to the previously used schema. 661 if (nDisplay == GetDefaultDisplay()) 662 return Any(sal_Int32(0)); 663 else if (nDisplay < 0) 664 return Any(sal_Int32(-1)); 665 else 666 return Any(nDisplay+1); 667 } 668 669 default: 670 throw UnknownPropertyException(); 671 } 672 } 673 674 // -------------------------------------------------------------------- 675 676 void SAL_CALL SlideShow::addPropertyChangeListener( const OUString& , const Reference< XPropertyChangeListener >& ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) 677 { 678 } 679 680 // -------------------------------------------------------------------- 681 682 void SAL_CALL SlideShow::removePropertyChangeListener( const OUString& , const Reference< XPropertyChangeListener >& ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) 683 { 684 } 685 686 // -------------------------------------------------------------------- 687 688 void SAL_CALL SlideShow::addVetoableChangeListener( const OUString& , const Reference< XVetoableChangeListener >& ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) 689 { 690 } 691 692 // -------------------------------------------------------------------- 693 694 void SAL_CALL SlideShow::removeVetoableChangeListener( const OUString& , const Reference< XVetoableChangeListener >& ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) 695 { 696 } 697 698 // -------------------------------------------------------------------- 699 // XPresentation 700 // -------------------------------------------------------------------- 701 702 void SAL_CALL SlideShow::start() throw(RuntimeException) 703 { 704 const Sequence< PropertyValue > aArguments; 705 startWithArguments( aArguments ); 706 } 707 708 // -------------------------------------------------------------------- 709 710 void SAL_CALL SlideShow::end() throw(RuntimeException) 711 { 712 OGuard aGuard( Application::GetSolarMutex() ); 713 714 // The mbIsInStartup flag should have been reset during the start of the 715 // slide show. Reset it here just in case that something has horribly 716 // gone wrong. 717 OSL_ASSERT(!mbIsInStartup); 718 mbIsInStartup = false; 719 720 rtl::Reference< SlideshowImpl > xController( mxController ); 721 if( xController.is() ) 722 { 723 mxController.clear(); 724 725 if( mpFullScreenFrameView ) 726 { 727 delete mpFullScreenFrameView; 728 mpFullScreenFrameView = 0; 729 } 730 731 ViewShellBase* pFullScreenViewShellBase = mpFullScreenViewShellBase; 732 mpFullScreenViewShellBase = 0; 733 734 if( pFullScreenViewShellBase ) 735 { 736 PresentationViewShell* pShell = dynamic_cast<PresentationViewShell*>(pFullScreenViewShellBase->GetMainViewShell().get()); 737 738 if( pShell && pShell->GetViewFrame() ) 739 { 740 WorkWindow* pWorkWindow = dynamic_cast<WorkWindow*>(pShell->GetViewFrame()->GetTopFrame().GetWindow().GetParent()); 741 if( pWorkWindow ) 742 { 743 pWorkWindow->StartPresentationMode( sal_False, isAlwaysOnTop() ); 744 } 745 } 746 } 747 748 xController->dispose(); 749 750 if( pFullScreenViewShellBase ) 751 { 752 PresentationViewShell* pShell = NULL; 753 { 754 // Get the shell pointer in its own scope to be sure that 755 // the shared_ptr to the shell is released before DoClose() 756 // is called. 757 ::boost::shared_ptr<ViewShell> pSharedView (pFullScreenViewShellBase->GetMainViewShell()); 758 pShell = dynamic_cast<PresentationViewShell*>(pSharedView.get()); 759 } 760 if( pShell && pShell->GetViewFrame() ) 761 pShell->GetViewFrame()->DoClose(); 762 } 763 else if( mpCurrentViewShellBase ) 764 { 765 ViewShell* pViewShell = mpCurrentViewShellBase->GetMainViewShell().get(); 766 767 if( pViewShell ) 768 { 769 FrameView* pFrameView = pViewShell->GetFrameView(); 770 771 if( pFrameView && (pFrameView->GetPresentationViewShellId() != SID_VIEWSHELL0) ) 772 { 773 ViewShell::ShellType ePreviousType (pFrameView->GetPreviousViewShellType()); 774 pFrameView->SetPreviousViewShellType(ViewShell::ST_NONE); 775 776 pFrameView->SetPresentationViewShellId(SID_VIEWSHELL0); 777 pFrameView->SetSlotId(SID_OBJECT_SELECT); 778 pFrameView->SetPreviousViewShellType(pViewShell->GetShellType()); 779 780 framework::FrameworkHelper::Instance(*mpCurrentViewShellBase)->RequestView( 781 framework::FrameworkHelper::GetViewURL(ePreviousType), 782 framework::FrameworkHelper::msCenterPaneURL); 783 784 pViewShell->GetViewFrame()->GetBindings().InvalidateAll( sal_True ); 785 } 786 } 787 } 788 789 if( mpCurrentViewShellBase ) 790 { 791 ViewShell* pViewShell = mpCurrentViewShellBase->GetMainViewShell().get(); 792 if( pViewShell ) 793 { 794 // invalidate the view shell so the presentation slot will be re-enabled 795 // and the rehersing will be updated 796 pViewShell->Invalidate(); 797 798 if( xController->meAnimationMode ==ANIMATIONMODE_SHOW ) 799 { 800 // switch to the previously visible Slide 801 DrawViewShell* pDrawViewShell = dynamic_cast<DrawViewShell*>( pViewShell ); 802 if( pDrawViewShell ) 803 pDrawViewShell->SwitchPage( (sal_uInt16)xController->getRestoreSlide() ); 804 else 805 { 806 Reference<XDrawView> xDrawView ( 807 Reference<XWeak>(&mpCurrentViewShellBase->GetDrawController()), UNO_QUERY); 808 if (xDrawView.is()) 809 xDrawView->setCurrentPage( 810 Reference<XDrawPage>( 811 mpDoc->GetSdPage(xController->getRestoreSlide(), PK_STANDARD)->getUnoPage(), 812 UNO_QUERY)); 813 } 814 } 815 } 816 } 817 mpCurrentViewShellBase = 0; 818 } 819 } 820 821 // -------------------------------------------------------------------- 822 823 void SAL_CALL SlideShow::rehearseTimings() throw(RuntimeException) 824 { 825 Sequence< PropertyValue > aArguments(1); 826 aArguments[0].Name = C2U("RehearseTimings"); 827 aArguments[0].Value <<= sal_True; 828 startWithArguments( aArguments ); 829 } 830 831 // -------------------------------------------------------------------- 832 // XPresentation2 833 // -------------------------------------------------------------------- 834 835 void SAL_CALL SlideShow::startWithArguments( const Sequence< PropertyValue >& rArguments ) throw (RuntimeException) 836 { 837 OGuard aGuard( Application::GetSolarMutex() ); 838 ThrowIfDisposed(); 839 840 // Stop a running show before starting a new one. 841 if( mxController.is() ) 842 { 843 OSL_ASSERT(!mbIsInStartup); 844 end(); 845 } 846 else if (mbIsInStartup) 847 { 848 // We are already somewhere in process of starting a slide show but 849 // have not yet got to the point where mxController is set. There 850 // is not yet a slide show to end so return silently. 851 return; 852 } 853 854 // Prevent multiple instance of the SlideShow class for one document. 855 mbIsInStartup = true; 856 857 mxCurrentSettings.reset( new PresentationSettingsEx( mpDoc->getPresentationSettings() ) ); 858 mxCurrentSettings->SetArguments( rArguments ); 859 860 // if there is no view shell base set, use the current one or the first using this document 861 if( mpCurrentViewShellBase == 0 ) 862 { 863 // first check current 864 ::sd::ViewShellBase* pBase = ::sd::ViewShellBase::GetViewShellBase( SfxViewFrame::Current() ); 865 if( pBase && pBase->GetDocument() == mpDoc ) 866 { 867 mpCurrentViewShellBase = pBase; 868 } 869 else 870 { 871 // current is not ours, so get first from ours 872 mpCurrentViewShellBase = ::sd::ViewShellBase::GetViewShellBase( SfxViewFrame::GetFirst( mpDoc->GetDocSh() ) ); 873 } 874 } 875 876 // #118456# make sure TextEdit changes get pushed to model. 877 // mpDrawView is tested against NULL above already. 878 if(mpCurrentViewShellBase) 879 { 880 ViewShell* pViewShell = mpCurrentViewShellBase->GetMainViewShell().get(); 881 882 if(pViewShell && pViewShell->GetView()) 883 { 884 pViewShell->GetView()->SdrEndTextEdit(); 885 } 886 } 887 888 // Start either a full-screen or an in-place show. 889 if(mxCurrentSettings->mbFullScreen && !mxCurrentSettings->mbPreview) 890 StartFullscreenPresentation(); 891 else 892 StartInPlacePresentation(); 893 } 894 895 // -------------------------------------------------------------------- 896 897 ::sal_Bool SAL_CALL SlideShow::isRunning( ) throw (RuntimeException) 898 { 899 OGuard aGuard( Application::GetSolarMutex() ); 900 return mxController.is() && mxController->isRunning(); 901 } 902 903 // -------------------------------------------------------------------- 904 905 Reference< XSlideShowController > SAL_CALL SlideShow::getController( ) throw (RuntimeException) 906 { 907 ThrowIfDisposed(); 908 909 Reference< XSlideShowController > xController( mxController.get() ); 910 return xController; 911 } 912 913 // -------------------------------------------------------------------- 914 // XComponent 915 // -------------------------------------------------------------------- 916 917 void SAL_CALL SlideShow::disposing (void) 918 { 919 OGuard aGuard( Application::GetSolarMutex() ); 920 921 if( mnInPlaceConfigEvent ) 922 { 923 Application::RemoveUserEvent( mnInPlaceConfigEvent ); 924 mnInPlaceConfigEvent = 0; 925 } 926 927 if( mxController.is() ) 928 { 929 mxController->dispose(); 930 mxController.clear(); 931 } 932 933 mpCurrentViewShellBase = 0; 934 mpFullScreenViewShellBase = 0; 935 mpDoc = 0; 936 } 937 938 // --------------------------------------------------------- 939 940 bool SlideShow::startPreview( const Reference< XDrawPage >& xDrawPage, const Reference< XAnimationNode >& xAnimationNode, ::Window* pParent ) 941 { 942 Sequence< PropertyValue > aArguments(4); 943 944 aArguments[0].Name = C2U("Preview"); 945 aArguments[0].Value <<= sal_True; 946 947 aArguments[1].Name = C2U("FirstPage"); 948 aArguments[1].Value <<= xDrawPage; 949 950 aArguments[2].Name = C2U("AnimationNode"); 951 aArguments[2].Value <<= xAnimationNode; 952 953 Reference< XWindow > xParentWindow; 954 if( pParent ) 955 xParentWindow = VCLUnoHelper::GetInterface( pParent ); 956 957 aArguments[3].Name = C2U("ParentWindow"); 958 aArguments[3].Value <<= xParentWindow; 959 960 startWithArguments( aArguments ); 961 962 return true; 963 } 964 965 // --------------------------------------------------------- 966 967 ShowWindow* SlideShow::getShowWindow() 968 { 969 return mxController.is() ? mxController->mpShowWindow : 0; 970 } 971 972 // --------------------------------------------------------- 973 974 int SlideShow::getAnimationMode() 975 { 976 return mxController.is() ? mxController->meAnimationMode : ANIMATIONMODE_SHOW; 977 } 978 979 // --------------------------------------------------------- 980 981 void SlideShow::jumpToPageIndex( sal_Int32 nPageIndex ) 982 { 983 if( mxController.is() ) 984 mxController->displaySlideIndex( nPageIndex ); 985 } 986 987 // --------------------------------------------------------- 988 989 void SlideShow::jumpToPageNumber( sal_Int32 nPageNumber ) 990 { 991 if( mxController.is() ) 992 mxController->displaySlideNumber( nPageNumber ); 993 } 994 995 // --------------------------------------------------------- 996 997 sal_Int32 SlideShow::getCurrentPageNumber() 998 { 999 return mxController.is() ? mxController->getCurrentSlideNumber() : 0; 1000 } 1001 1002 // --------------------------------------------------------- 1003 1004 void SlideShow::jumpToBookmark( const OUString& sBookmark ) 1005 { 1006 if( mxController.is() ) 1007 mxController->jumpToBookmark( sBookmark ); 1008 } 1009 1010 // --------------------------------------------------------- 1011 1012 bool SlideShow::isFullScreen() 1013 { 1014 return mxController.is() ? mxController->maPresSettings.mbFullScreen : false; 1015 } 1016 1017 // --------------------------------------------------------- 1018 1019 void SlideShow::resize( const Size &rSize ) 1020 { 1021 if( mxController.is() ) 1022 mxController->resize( rSize ); 1023 } 1024 1025 // --------------------------------------------------------- 1026 1027 void SlideShow::activate( ViewShellBase& rBase ) 1028 { 1029 if( (mpFullScreenViewShellBase == &rBase) && !mxController.is() ) 1030 { 1031 ::boost::shared_ptr<PresentationViewShell> pShell = ::boost::dynamic_pointer_cast<PresentationViewShell>(rBase.GetMainViewShell()); 1032 if(pShell.get() != NULL) 1033 { 1034 pShell->FinishInitialization( mpFullScreenFrameView ); 1035 mpFullScreenFrameView = 0; 1036 1037 CreateController( pShell.get(), pShell->GetView(), rBase.GetViewWindow() ); 1038 1039 if( mxController->startShow(mxCurrentSettings.get()) ) 1040 { 1041 pShell->Resize(); 1042 } 1043 else 1044 { 1045 end(); 1046 return; 1047 } 1048 } 1049 } 1050 1051 if( mxController.is() ) 1052 mxController->activate(); 1053 } 1054 1055 // --------------------------------------------------------- 1056 1057 void SlideShow::deactivate( ViewShellBase& /*rBase*/ ) 1058 { 1059 mxController->deactivate(); 1060 } 1061 1062 // --------------------------------------------------------- 1063 1064 bool SlideShow::keyInput(const KeyEvent& rKEvt) 1065 { 1066 return mxController.is() ? mxController->keyInput(rKEvt) : false; 1067 } 1068 1069 // --------------------------------------------------------- 1070 1071 void SlideShow::paint( const Rectangle& rRect ) 1072 { 1073 if( mxController.is() ) 1074 mxController->paint( rRect ); 1075 } 1076 1077 // --------------------------------------------------------- 1078 1079 bool SlideShow::isAlwaysOnTop() 1080 { 1081 return mxController.is() ? mxController->maPresSettings.mbAlwaysOnTop : false; 1082 } 1083 1084 // --------------------------------------------------------- 1085 1086 bool SlideShow::pause( bool bPause ) 1087 { 1088 if( mxController.is() ) 1089 { 1090 if( bPause ) 1091 mxController->pause(); 1092 else 1093 mxController->resume(); 1094 } 1095 return true; 1096 } 1097 1098 // --------------------------------------------------------- 1099 1100 void SlideShow::receiveRequest(SfxRequest& rReq) 1101 { 1102 if( mxController.is() ) 1103 mxController->receiveRequest( rReq ); 1104 } 1105 1106 // --------------------------------------------------------- 1107 1108 sal_Int32 SlideShow::getFirstPageNumber() 1109 { 1110 return mxController.is() ? mxController->getFirstSlideNumber() : 0; 1111 } 1112 1113 // --------------------------------------------------------- 1114 1115 sal_Int32 SlideShow::getLastPageNumber() 1116 { 1117 return mxController.is() ? mxController->getLastSlideNumber() : 0; 1118 } 1119 1120 // --------------------------------------------------------- 1121 1122 bool SlideShow::isEndless() 1123 { 1124 return mxController.is() ? mxController->isEndless() : false; 1125 } 1126 1127 // --------------------------------------------------------- 1128 1129 bool SlideShow::isDrawingPossible() 1130 { 1131 return mxController.is() ? mxController->getUsePen() : false; 1132 } 1133 1134 // --------------------------------------------------------- 1135 1136 void SlideShow::StartInPlacePresentationConfigurationCallback() 1137 { 1138 if( mnInPlaceConfigEvent != 0 ) 1139 Application::RemoveUserEvent( mnInPlaceConfigEvent ); 1140 1141 mnInPlaceConfigEvent = Application::PostUserEvent( LINK( this, SlideShow, StartInPlacePresentationConfigurationHdl ) ); 1142 } 1143 1144 // --------------------------------------------------------- 1145 1146 IMPL_LINK( SlideShow, StartInPlacePresentationConfigurationHdl, void *, EMPTYARG ) 1147 { 1148 mnInPlaceConfigEvent = 0; 1149 StartInPlacePresentation(); 1150 return 0; 1151 } 1152 1153 // --------------------------------------------------------- 1154 1155 void SlideShow::StartInPlacePresentation() 1156 { 1157 if( mpCurrentViewShellBase ) 1158 { 1159 // Save the current view shell type so that it can be restored after the 1160 // show has ended. If there already is a saved shell type then that is 1161 // not overwritten. 1162 1163 ViewShell::ShellType eShell = ViewShell::ST_NONE; 1164 1165 ::boost::shared_ptr<FrameworkHelper> pHelper(FrameworkHelper::Instance(*mpCurrentViewShellBase)); 1166 ::boost::shared_ptr<ViewShell> pMainViewShell(pHelper->GetViewShell(FrameworkHelper::msCenterPaneURL)); 1167 1168 if( pMainViewShell.get() ) 1169 eShell = pMainViewShell->GetShellType(); 1170 1171 if( eShell != ViewShell::ST_IMPRESS ) 1172 { 1173 // Switch temporary to a DrawViewShell which supports the in-place presentation. 1174 1175 if( pMainViewShell.get() ) 1176 { 1177 FrameView* pFrameView = pMainViewShell->GetFrameView(); 1178 pFrameView->SetPresentationViewShellId(SID_VIEWSHELL1); 1179 pFrameView->SetPreviousViewShellType (pMainViewShell->GetShellType()); 1180 pFrameView->SetPageKind (PK_STANDARD); 1181 } 1182 1183 pHelper->RequestView( FrameworkHelper::msImpressViewURL, FrameworkHelper::msCenterPaneURL ); 1184 pHelper->RunOnConfigurationEvent( FrameworkHelper::msConfigurationUpdateEndEvent, ::boost::bind(&SlideShow::StartInPlacePresentationConfigurationCallback, this) ); 1185 return; 1186 } 1187 else 1188 { 1189 ::Window* pParentWindow = mxCurrentSettings->mpParentWindow; 1190 if( pParentWindow == 0 ) 1191 pParentWindow = mpCurrentViewShellBase->GetViewWindow(); 1192 1193 CreateController( pMainViewShell.get(), pMainViewShell->GetView(), pParentWindow ); 1194 } 1195 } 1196 else if( mxCurrentSettings->mpParentWindow ) 1197 { 1198 // no current view shell, but parent window 1199 CreateController( 0, 0, mxCurrentSettings->mpParentWindow ); 1200 } 1201 1202 if( mxController.is() ) 1203 { 1204 sal_Bool bSuccess = sal_False; 1205 if( mxCurrentSettings.get() && mxCurrentSettings->mbPreview ) 1206 { 1207 bSuccess = mxController->startPreview(mxCurrentSettings->mxStartPage, mxCurrentSettings->mxAnimationNode, mxCurrentSettings->mpParentWindow ); 1208 } 1209 else 1210 { 1211 bSuccess = mxController->startShow(mxCurrentSettings.get()); 1212 } 1213 1214 if( !bSuccess ) 1215 end(); 1216 } 1217 } 1218 1219 // --------------------------------------------------------- 1220 1221 void SlideShow::StartFullscreenPresentation( ) 1222 { 1223 // Create the top level window in which the PresentationViewShell(Base) 1224 // will be created. This is done here explicitly so that we can make it 1225 // fullscreen. 1226 const sal_Int32 nDisplay (GetDisplay()); 1227 WorkWindow* pWorkWindow = new FullScreenWorkWindow(this, mpCurrentViewShellBase); 1228 pWorkWindow->SetBackground(Wallpaper(COL_BLACK)); 1229 pWorkWindow->StartPresentationMode( sal_True, mpDoc->getPresentationSettings().mbAlwaysOnTop ? PRESENTATION_HIDEALLAPPS : 0, nDisplay); 1230 // pWorkWindow->ShowFullScreenMode(sal_False, nDisplay); 1231 1232 if (pWorkWindow->IsVisible()) 1233 { 1234 // Initialize the new presentation view shell with a copy of the 1235 // frame view of the current view shell. This avoids that 1236 // changes made by the presentation have an effect on the other 1237 // view shells. 1238 FrameView* pOriginalFrameView = mpCurrentViewShellBase ? mpCurrentViewShellBase->GetMainViewShell()->GetFrameView() : 0; 1239 1240 if( mpFullScreenFrameView ) 1241 delete mpFullScreenFrameView; 1242 mpFullScreenFrameView = new FrameView(mpDoc, pOriginalFrameView); 1243 1244 // Reference<XController> xController; 1245 1246 // The new frame is created hidden. To make it visible and activate the 1247 // new view shell--a prerequisite to process slot calls and initialize 1248 // its panes--a GrabFocus() has to be called later on. 1249 SfxFrame* pNewFrame = SfxFrame::Create( *mpDoc->GetDocSh(), *pWorkWindow, PRESENTATION_FACTORY_ID, true ); 1250 pNewFrame->SetPresentationMode(sal_True); 1251 1252 mpFullScreenViewShellBase = static_cast<ViewShellBase*>(pNewFrame->GetCurrentViewFrame()->GetViewShell()); 1253 if(mpFullScreenViewShellBase != NULL) 1254 { 1255 // The following GrabFocus() is responsible for activating the 1256 // new view shell. Without it the screen remains blank (under 1257 // Windows and some Linux variants.) 1258 mpFullScreenViewShellBase->GetWindow()->GrabFocus(); 1259 } 1260 } 1261 } 1262 1263 // --------------------------------------------------------- 1264 1265 sal_Int32 SlideShow::GetDisplay() 1266 1267 { 1268 sal_Int32 nDisplay = 0; 1269 1270 SdOptions* pOptions = SD_MOD()->GetSdOptions(DOCUMENT_TYPE_IMPRESS); 1271 if( pOptions ) 1272 nDisplay = pOptions->GetDisplay(); 1273 1274 return nDisplay; 1275 } 1276 1277 // --------------------------------------------------------- 1278 1279 1280 bool SlideShow::dependsOn( ViewShellBase* pViewShellBase ) 1281 { 1282 return mxController.is() && (pViewShellBase == mpCurrentViewShellBase) && mpFullScreenViewShellBase; 1283 } 1284 1285 // --------------------------------------------------------- 1286 1287 Reference< XPresentation2 > CreatePresentation( const SdDrawDocument& rDocument ) 1288 { 1289 return Reference< XPresentation2 >( SlideShow::Create( const_cast< SdDrawDocument* >( &rDocument ) ).get() ); 1290 } 1291 1292 // --------------------------------------------------------- 1293 1294