/************************************************************** * * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. * *************************************************************/ #include "precompiled_sd.hxx" #include "SlideSorter.hxx" #include "model/SlideSorterModel.hxx" #include "model/SlsPageDescriptor.hxx" #include "controller/SlsPageSelector.hxx" #include "controller/SlideSorterController.hxx" #include "controller/SlsCurrentSlideManager.hxx" #include "controller/SlsFocusManager.hxx" #include "view/SlideSorterView.hxx" #include "ViewShellBase.hxx" #include "ViewShell.hxx" #include "DrawViewShell.hxx" #include "sdpage.hxx" #include "FrameView.hxx" #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::sd::slidesorter::model; namespace sd { namespace slidesorter { namespace controller { CurrentSlideManager::CurrentSlideManager (SlideSorter& rSlideSorter) : mrSlideSorter(rSlideSorter), mnCurrentSlideIndex(-1), mpCurrentSlide(), maSwitchPageDelayTimer() { maSwitchPageDelayTimer.SetTimeout(100); maSwitchPageDelayTimer.SetTimeoutHdl(LINK(this,CurrentSlideManager,SwitchPageCallback)); } CurrentSlideManager::~CurrentSlideManager (void) { } void CurrentSlideManager::NotifyCurrentSlideChange (const SdPage* pPage) { if (pPage != NULL) NotifyCurrentSlideChange( mrSlideSorter.GetModel().GetIndex( Reference( const_cast(pPage)->getUnoPage(), UNO_QUERY))); else NotifyCurrentSlideChange(-1); } void CurrentSlideManager::NotifyCurrentSlideChange (const sal_Int32 nSlideIndex) { if (mnCurrentSlideIndex != nSlideIndex) { PageSelector::BroadcastLock aBroadcastLock (mrSlideSorter.GetController().GetPageSelector()); mrSlideSorter.GetController().GetPageSelector().DeselectAllPages(); ReleaseCurrentSlide(); AcquireCurrentSlide(nSlideIndex); // Update the selection. if (mpCurrentSlide) { mrSlideSorter.GetController().GetPageSelector().SelectPage(mpCurrentSlide); mrSlideSorter.GetController().GetFocusManager().SetFocusedPage(mpCurrentSlide); } } } void CurrentSlideManager::ReleaseCurrentSlide (void) { if (mpCurrentSlide.get() != NULL) mrSlideSorter.GetView().SetState(mpCurrentSlide, PageDescriptor::ST_Current, false); mpCurrentSlide.reset(); mnCurrentSlideIndex = -1; } bool CurrentSlideManager::IsCurrentSlideIsValid (void) { return mnCurrentSlideIndex >= 0 && mnCurrentSlideIndexGetPage()->GetPageNum()-1)/2); ViewShell* pViewShell = mrSlideSorter.GetViewShell(); if (pViewShell != NULL && pViewShell->IsMainViewShell()) { // The slide sorter is the main view. FrameView* pFrameView = pViewShell->GetFrameView(); if (pFrameView != NULL) pFrameView->SetSelectedPage(sal::static_int_cast(mnCurrentSlideIndex)); mrSlideSorter.GetController().GetPageSelector().SetCoreSelection(); } // We do not tell the XController/ViewShellBase about the new // slide right away. This is done asynchronously after a short // delay to allow for more slide switches in the slide sorter. // This goes under the assumption that slide switching inside // the slide sorter is fast (no expensive redraw of the new page // (unless the preview of the new slide is not yet preset)) and // that slide switching in the edit view is slow (all shapes of // the new slide have to be repainted.) maSwitchPageDelayTimer.Start(); // We have to store the (index of the) new current slide at // the tab control because there are other asynchronous // notifications of the slide switching that otherwise // overwrite the correct value. SetCurrentSlideAtTabControl(mpCurrentSlide); if (bUpdateSelection) { mrSlideSorter.GetController().GetPageSelector().DeselectAllPages(); mrSlideSorter.GetController().GetPageSelector().SelectPage(rpDescriptor); } mrSlideSorter.GetController().GetFocusManager().SetFocusedPage(rpDescriptor); } } void CurrentSlideManager::SetCurrentSlideAtViewShellBase (const SharedPageDescriptor& rpDescriptor) { OSL_ASSERT(rpDescriptor.get() != NULL); ViewShellBase* pBase = mrSlideSorter.GetViewShellBase(); if (pBase != NULL) { DrawViewShell* pDrawViewShell = dynamic_cast( pBase->GetMainViewShell().get()); if (pDrawViewShell != NULL) { sal_uInt16 nPageNumber = (rpDescriptor->GetPage()->GetPageNum()-1)/2; pDrawViewShell->SwitchPage(nPageNumber); pDrawViewShell->GetPageTabControl()->SetCurPageId(nPageNumber+1); } } } void CurrentSlideManager::SetCurrentSlideAtTabControl (const SharedPageDescriptor& rpDescriptor) { OSL_ASSERT(rpDescriptor.get() != NULL); ViewShellBase* pBase = mrSlideSorter.GetViewShellBase(); if (pBase != NULL) { ::boost::shared_ptr pDrawViewShell ( ::boost::dynamic_pointer_cast(pBase->GetMainViewShell())); if (pDrawViewShell) { sal_uInt16 nPageNumber = (rpDescriptor->GetPage()->GetPageNum()-1)/2; pDrawViewShell->GetPageTabControl()->SetCurPageId(nPageNumber+1); } } } void CurrentSlideManager::SetCurrentSlideAtXController (const SharedPageDescriptor& rpDescriptor) { OSL_ASSERT(rpDescriptor.get() != NULL); try { Reference xSet (mrSlideSorter.GetXController(), UNO_QUERY); if (xSet.is()) { Any aPage; aPage <<= rpDescriptor->GetPage()->getUnoPage(); xSet->setPropertyValue ( String::CreateFromAscii("CurrentPage"), aPage); } } catch (Exception aException) { // We have not been able to set the current page at the main view. // This is sad but still leaves us in a valid state. Therefore, // this exception is silently ignored. } } SharedPageDescriptor CurrentSlideManager::GetCurrentSlide (void) { return mpCurrentSlide; } void CurrentSlideManager::PrepareModelChange (void) { mpCurrentSlide.reset(); } void CurrentSlideManager::HandleModelChange (void) { if (mnCurrentSlideIndex >= 0) { mpCurrentSlide = mrSlideSorter.GetModel().GetPageDescriptor(mnCurrentSlideIndex); if (mpCurrentSlide.get() != NULL) mrSlideSorter.GetView().SetState(mpCurrentSlide, PageDescriptor::ST_Current, true); } } IMPL_LINK(CurrentSlideManager, SwitchPageCallback, void*, EMPTYARG) { if (mpCurrentSlide) { // Set current page. At the moment we have to do this in two // different ways. The UNO way is the preferable one but, alas, // it does not work always correctly (after some kinds of model // changes). Therefore, we call DrawViewShell::SwitchPage(), // too. ViewShell* pViewShell = mrSlideSorter.GetViewShell(); if (pViewShell==NULL || ! pViewShell->IsMainViewShell()) SetCurrentSlideAtViewShellBase(mpCurrentSlide); SetCurrentSlideAtXController(mpCurrentSlide); } return 1; } } } } // end of namespace ::sd::slidesorter::controller