/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * 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 . */ #include #include #include "scitems.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "prevwsh.hxx" #include "preview.hxx" #include "printfun.hxx" #include "attrib.hxx" #include "scmod.hxx" #include "inputhdl.hxx" #include "docsh.hxx" #include "tabvwsh.hxx" #include "stlpool.hxx" #include "editutil.hxx" #include "scresid.hxx" #include "globstr.hrc" #include "sc.hrc" #include "ViewSettingsSequenceDefines.hxx" #include "tpprint.hxx" #include "printopt.hxx" #include "viewuno.hxx" #include #include #include #include #include #include #include #include "scabstdlg.hxx" // for mouse wheel #define MINZOOM_SLIDER 10 #define MAXZOOM_SLIDER 400 #define SC_USERDATA_SEP ';' using namespace com::sun::star; #define ScPreviewShell #include "scslots.hxx" #include SFX_IMPL_INTERFACE(ScPreviewShell, SfxViewShell) void ScPreviewShell::InitInterface_Impl() { GetStaticInterface()->RegisterObjectBar(SFX_OBJECTBAR_OBJECT, SfxVisibilityFlags::Standard|SfxVisibilityFlags::Server|SfxVisibilityFlags::ReadonlyDoc, ToolbarId::Objectbar_Preview); GetStaticInterface()->RegisterPopupMenu("preview"); } SFX_IMPL_NAMED_VIEWFACTORY( ScPreviewShell, "PrintPreview" ) { SFX_VIEW_REGISTRATION(ScDocShell); } void ScPreviewShell::Construct( vcl::Window* pParent ) { // Find the top-most window, and set the close window handler to intercept // the window close event. vcl::Window* pWin = pParent; while (!pWin->IsSystemWindow()) { if (pWin->GetParent()) pWin = pWin->GetParent(); else break; } mpFrameWindow = dynamic_cast(pWin); if (mpFrameWindow) mpFrameWindow->SetCloseHdl(LINK(this, ScPreviewShell, CloseHdl)); eZoom = SvxZoomType::WHOLEPAGE; pCorner = VclPtr::Create( pParent, WB_SIZEABLE ); pHorScroll = VclPtr::Create(pParent, WB_HSCROLL ); pVerScroll = VclPtr::Create(pParent, WB_VSCROLL); // RTL: no mirroring for horizontal scrollbars pHorScroll->EnableRTL( false ); pHorScroll->SetEndScrollHdl( LINK( this, ScPreviewShell, ScrollHandler ) ); pVerScroll->SetEndScrollHdl( LINK( this, ScPreviewShell, ScrollHandler ) ); pPreview = VclPtr::Create( pParent, pDocShell, this ); SetPool( &SC_MOD()->GetPool() ); SetWindow( pPreview ); StartListening(*pDocShell,true); StartListening(*SfxGetpApp(),true); // #i62045# #i62046# application is needed for Calc's own hints SfxBroadcaster* pDrawBC = pDocShell->GetDocument().GetDrawBroadcaster(); if (pDrawBC) StartListening(*pDrawBC); pHorScroll->Show( false ); pVerScroll->Show( false ); pCorner->Show(); SetName("Preview"); } ScPreviewShell::ScPreviewShell( SfxViewFrame* pViewFrame, SfxViewShell* pOldSh ) : SfxViewShell( pViewFrame, SfxViewShellFlags::HAS_PRINTOPTIONS ), pDocShell( static_cast(pViewFrame->GetObjectShell()) ), mpFrameWindow(nullptr), nSourceDesignMode( TRISTATE_INDET ), nMaxVertPos(0), pAccessibilityBroadcaster( nullptr ) { Construct( &pViewFrame->GetWindow() ); if ( pOldSh && dynamic_cast( pOldSh) != nullptr ) { // store view settings, show table from TabView //! store live ScViewData instead, and update on ScTablesHint? //! or completely forget aSourceData on ScTablesHint? ScTabViewShell* pTabViewShell = static_cast(pOldSh); const ScViewData& rData = pTabViewShell->GetViewData(); rData.WriteUserDataSequence( aSourceData ); pPreview->SetSelectedTabs(rData.GetMarkData()); InitStartTable( rData.GetTabNo() ); // also have to store the TabView's DesignMode state // (only if draw view exists) SdrView* pDrawView = pTabViewShell->GetSdrView(); if ( pDrawView ) nSourceDesignMode = pDrawView->IsDesignMode() ? TRISTATE_TRUE : TRISTATE_FALSE; } new ScPreviewObj(this); } ScPreviewShell::~ScPreviewShell() { if (mpFrameWindow) mpFrameWindow->SetCloseHdl(Link()); // Remove close handler. // #108333#; notify Accessibility that Shell is dying and before destroy all BroadcastAccessibility( SfxHint( SfxHintId::Dying ) ); DELETEZ(pAccessibilityBroadcaster); SfxBroadcaster* pDrawBC = pDocShell->GetDocument().GetDrawBroadcaster(); if (pDrawBC) EndListening(*pDrawBC); EndListening(*SfxGetpApp()); EndListening(*pDocShell); SetWindow(nullptr); pPreview.disposeAndClear(); pHorScroll.disposeAndClear(); pVerScroll.disposeAndClear(); pCorner.disposeAndClear(); // normal mode of operation is switching back to default view in the same frame, // so there's no need to activate any other window here anymore } void ScPreviewShell::InitStartTable(SCTAB nTab) { pPreview->SetPageNo( pPreview->GetFirstPage(nTab) ); } void ScPreviewShell::AdjustPosSizePixel( const Point &rPos, const Size &rSize ) { Size aOutSize( rSize ); pPreview->SetPosSizePixel( rPos, aOutSize ); if ( SvxZoomType::WHOLEPAGE == eZoom ) pPreview->SetZoom( pPreview->GetOptimalZoom(false) ); else if ( SvxZoomType::PAGEWIDTH == eZoom ) pPreview->SetZoom( pPreview->GetOptimalZoom(true) ); UpdateNeededScrollBars(false); } void ScPreviewShell::InnerResizePixel( const Point &rOfs, const Size &rSize, bool ) { AdjustPosSizePixel( rOfs,rSize ); } void ScPreviewShell::OuterResizePixel( const Point &rOfs, const Size &rSize ) { AdjustPosSizePixel( rOfs,rSize ); } bool ScPreviewShell::GetPageSize( Size& aPageSize ) { ScDocument& rDoc = pDocShell->GetDocument(); SCTAB nTab = pPreview->GetTab(); ScStyleSheetPool* pStylePool = rDoc.GetStyleSheetPool(); SfxStyleSheetBase* pStyleSheet = pStylePool->Find( rDoc.GetPageStyle( nTab ), SfxStyleFamily::Page ); OSL_ENSURE(pStyleSheet,"No style sheet"); if (!pStyleSheet) return false; const SfxItemSet* pParamSet = &pStyleSheet->GetItemSet(); aPageSize = static_cast( pParamSet->Get(ATTR_PAGE_SIZE)).GetSize(); aPageSize.Width() = (long) (aPageSize.Width() * HMM_PER_TWIPS ); aPageSize.Height() = (long) (aPageSize.Height() * HMM_PER_TWIPS ); return true; } void ScPreviewShell::UpdateNeededScrollBars( bool bFromZoom ) { Size aPageSize; OutputDevice* pDevice = Application::GetDefaultDevice(); long nBarW = GetViewFrame()->GetWindow().GetSettings().GetStyleSettings().GetScrollBarSize(); long nBarH = nBarW; long aHeightOffSet = pDevice ? pDevice->PixelToLogic( Size( nBarW, nBarH ), pPreview->GetMapMode() ).Height() : 0; long aWidthOffSet = aHeightOffSet; if (!GetPageSize( aPageSize )) return; // for centering, page size without the shadow is used bool bVert = pVerScroll->IsVisible(); bool bHori = pHorScroll->IsVisible(); Size aWindowSize = pPreview->GetOutputSize(); Point aPos = pPreview->GetPosPixel(); Size aWindowPixelSize = pPreview->GetOutputSizePixel(); // if we are called from Zoom then we need to compensate for whatever // scrollbars were displayed before the zoom was called if ( bFromZoom ) { if ( bVert ) { aWindowPixelSize.Width() += nBarH; aWindowSize.Width() += aHeightOffSet; } if ( bHori ) { aWindowPixelSize.Height() += nBarW; aWindowSize.Height() += aWidthOffSet; } } // recalculate any needed scrollbars long nMaxWidthPos = aPageSize.Width() - aWindowSize.Width(); bHori = nMaxWidthPos >= 0; long nMaxHeightPos = aPageSize.Height() - aWindowSize.Height(); bVert = nMaxHeightPos >= 0; // see if having a scroll bar requires the other if ( bVert != bHori && ( bVert || bHori ) ) { if ( bVert && ( (nMaxWidthPos + aWidthOffSet ) > 0 ) ) bHori = true; else if ( (nMaxHeightPos + aHeightOffSet ) > 0 ) bVert = true; } pHorScroll->Show( bHori ); pVerScroll->Show( bVert ); // make room for needed scrollbars ( and reduce the size // of the preview appropriately ) if ( bHori ) aWindowPixelSize.Height() -= nBarW; if ( bVert ) aWindowPixelSize.Width() -= nBarH; pPreview->SetSizePixel( aWindowPixelSize ); pHorScroll->SetPosSizePixel( Point( aPos.X(), aPos.Y() + aWindowPixelSize.Height() ), Size( aWindowPixelSize.Width(), nBarH ) ); pVerScroll->SetPosSizePixel( Point( aPos.X() + aWindowPixelSize.Width(), aPos.Y() ), Size( nBarW, aWindowPixelSize.Height() ) ); pCorner->SetPosSizePixel( Point( aPos.X() + aWindowPixelSize.Width(), aPos.Y() + aWindowPixelSize.Height() ), Size( nBarW, nBarH ) ); UpdateScrollBars(); } void ScPreviewShell::UpdateScrollBars() { Size aPageSize; if ( !GetPageSize( aPageSize ) ) return; // for centering, page size without the shadow is used Size aWindowSize = pPreview->GetOutputSize(); Point aOfs = pPreview->GetOffset(); if( pHorScroll ) { pHorScroll->SetRange( Range( 0, aPageSize.Width() ) ); pHorScroll->SetLineSize( aWindowSize.Width() / 16 ); pHorScroll->SetPageSize( aWindowSize.Width() ); pHorScroll->SetVisibleSize( aWindowSize.Width() ); long nMaxPos = aPageSize.Width() - aWindowSize.Width(); if ( nMaxPos<0 ) { // page smaller than window -> center (but put scrollbar to 0) aOfs.X() = 0; pPreview->SetXOffset( nMaxPos / 2 ); } else if (aOfs.X() < 0) { // page larger than window -> never use negative offset aOfs.X() = 0; pPreview->SetXOffset( 0 ); } else if (aOfs.X() > nMaxPos) { // limit offset to align with right edge of window aOfs.X() = nMaxPos; pPreview->SetXOffset(nMaxPos); } pHorScroll->SetThumbPos( aOfs.X() ); } if( pVerScroll ) { long nPageNo = pPreview->GetPageNo(); long nTotalPages = pPreview->GetTotalPages(); nMaxVertPos = aPageSize.Height() - aWindowSize.Height(); pVerScroll->SetLineSize( aWindowSize.Height() / 16 ); pVerScroll->SetPageSize( aWindowSize.Height() ); pVerScroll->SetVisibleSize( aWindowSize.Height() ); if ( nMaxVertPos < 0 ) { // page smaller than window -> center (but put scrollbar to 0) aOfs.Y() = 0; pPreview->SetYOffset( nMaxVertPos / 2 ); pVerScroll->SetThumbPos( nPageNo * aWindowSize.Height() ); pVerScroll->SetRange( Range( 0, aWindowSize.Height() * nTotalPages )); } else if (aOfs.Y() < 0) { // page larger than window -> never use negative offset pVerScroll->SetRange( Range( 0, aPageSize.Height() ) ); aOfs.Y() = 0; pPreview->SetYOffset( 0 ); pVerScroll->SetThumbPos( aOfs.Y() ); } else if (aOfs.Y() > nMaxVertPos ) { // limit offset to align with window bottom pVerScroll->SetRange( Range( 0, aPageSize.Height() ) ); aOfs.Y() = nMaxVertPos; pPreview->SetYOffset( nMaxVertPos ); pVerScroll->SetThumbPos( aOfs.Y() ); } } } IMPL_LINK( ScPreviewShell, ScrollHandler, ScrollBar*, pScroll, void ) { long nPos = pScroll->GetThumbPos(); long nDelta = pScroll->GetDelta(); long nMaxRange = pScroll->GetRangeMax(); long nTotalPages = pPreview->GetTotalPages(); long nPageNo = 0; long nPerPageLength = 0; bool bIsDivide = true; if( nTotalPages ) nPerPageLength = nMaxRange / nTotalPages; if( nPerPageLength ) { nPageNo = nPos / nPerPageLength; if( nPos % nPerPageLength ) { bIsDivide = false; nPageNo ++; } } bool bHoriz = ( pScroll == pHorScroll ); if( bHoriz ) pPreview->SetXOffset( nPos ); else { if( nMaxVertPos > 0 ) pPreview->SetYOffset( nPos ); else { Point aMousePos = pScroll->OutputToNormalizedScreenPixel( pScroll->GetPointerPosPixel() ); Point aPos = pScroll->GetParent()->OutputToNormalizedScreenPixel( pScroll->GetPosPixel() ); OUString aHelpStr; tools::Rectangle aRect; QuickHelpFlags nAlign; if( nDelta < 0 ) { if ( nTotalPages && nPageNo > 0 && !bIsDivide ) pPreview->SetPageNo( nPageNo-1 ); if( bIsDivide ) pPreview->SetPageNo( nPageNo ); aHelpStr = ScGlobal::GetRscString( STR_PAGE ) + " " + OUString::number( nPageNo ) + " / " + OUString::number( nTotalPages ); } else if( nDelta > 0 ) { bool bAllTested = pPreview->AllTested(); if ( nTotalPages && ( nPageNo < nTotalPages || !bAllTested ) ) pPreview->SetPageNo( nPageNo ); aHelpStr = ScGlobal::GetRscString( STR_PAGE ) + " " + OUString::number( nPageNo+1 ) + " / " + OUString::number( nTotalPages ); } aRect.Left() = aPos.X() - 8; aRect.Top() = aMousePos.Y(); aRect.Right() = aRect.Left(); aRect.Bottom() = aRect.Top(); nAlign = QuickHelpFlags::Bottom|QuickHelpFlags::Center; Help::ShowQuickHelp( pScroll->GetParent(), aRect, aHelpStr, nAlign ); } } } IMPL_LINK_NOARG(ScPreviewShell, CloseHdl, SystemWindow&, void) { ExitPreview(); } bool ScPreviewShell::ScrollCommand( const CommandEvent& rCEvt ) { bool bDone = false; const CommandWheelData* pData = rCEvt.GetWheelData(); if ( pData && pData->GetMode() == CommandWheelMode::ZOOM ) { long nOld = pPreview->GetZoom(); long nNew; if ( pData->GetDelta() < 0 ) nNew = std::max( (long) MINZOOM, basegfx::zoomtools::zoomOut( nOld )); else nNew = std::min( (long) MAXZOOM, basegfx::zoomtools::zoomIn( nOld )); if ( nNew != nOld ) { eZoom = SvxZoomType::PERCENT; pPreview->SetZoom( (sal_uInt16)nNew ); } bDone = true; } else { bDone = pPreview->HandleScrollCommand( rCEvt, pHorScroll, pVerScroll ); } return bDone; } SfxPrinter* ScPreviewShell::GetPrinter( bool bCreate ) { return pDocShell->GetPrinter(bCreate); } sal_uInt16 ScPreviewShell::SetPrinter( SfxPrinter *pNewPrinter, SfxPrinterChangeFlags nDiffFlags ) { return pDocShell->SetPrinter( pNewPrinter, nDiffFlags ); } bool ScPreviewShell::HasPrintOptionsPage() const { return true; } VclPtr ScPreviewShell::CreatePrintOptionsPage( vcl::Window *pParent, const SfxItemSet &rOptions ) { ScAbstractDialogFactory* pFact = ScAbstractDialogFactory::Create(); OSL_ENSURE(pFact, "ScAbstractFactory create fail!"); ::CreateTabPage ScTpPrintOptionsCreate = pFact->GetTabPageCreatorFunc(RID_SC_TP_PRINT); if ( ScTpPrintOptionsCreate ) return ScTpPrintOptionsCreate( pParent, &rOptions ); return VclPtr(); } void ScPreviewShell::Activate(bool bMDI) { SfxViewShell::Activate(bMDI); //! Basic etc. -> outsource to its own file (see tabvwsh4) if (bMDI) { // InputHdl is now mostly Null, no moreasssertion! ScInputHandler* pInputHdl = SC_MOD()->GetInputHdl(); if ( pInputHdl ) pInputHdl->NotifyChange( nullptr ); } } void ScPreviewShell::Deactivate(bool bMDI) { SfxViewShell::Deactivate(bMDI); if (bMDI) { } } void ScPreviewShell::Execute( SfxRequest& rReq ) { sal_uInt16 nSlot = rReq.GetSlot(); const SfxItemSet* pReqArgs = rReq.GetArgs(); switch ( nSlot ) { case SID_FORMATPAGE: case SID_STATUS_PAGESTYLE: case SID_HFEDIT: pDocShell->ExecutePageStyle( *this, rReq, pPreview->GetTab() ); break; case SID_REPAINT: pPreview->Invalidate(); rReq.Done(); break; case SID_PREV_TABLE: // Accelerator case SID_PREVIEW_PREVIOUS: { long nPage = pPreview->GetPageNo(); long nTotal = pPreview->GetTotalPages(); if (nTotal && nPage > 0) pPreview->SetPageNo( nPage-1 ); } break; case SID_NEXT_TABLE: // Accelerator case SID_PREVIEW_NEXT: { bool bAllTested = pPreview->AllTested(); long nPage = pPreview->GetPageNo(); long nTotal = pPreview->GetTotalPages(); if (nTotal && (nPage+1 < nTotal || !bAllTested)) pPreview->SetPageNo( nPage+1 ); } break; case SID_CURSORTOPOFFILE: // Accelerator case SID_PREVIEW_FIRST: { long nPage = pPreview->GetPageNo(); long nTotal = pPreview->GetTotalPages(); if (nTotal && nPage != 0) pPreview->SetPageNo( 0 ); } break; case SID_CURSORENDOFFILE: // Accelerator case SID_PREVIEW_LAST: { if (!pPreview->AllTested()) pPreview->CalcAll(); long nPage = pPreview->GetPageNo(); long nTotal = pPreview->GetTotalPages(); if (nTotal && nPage+1 != nTotal) pPreview->SetPageNo( nTotal-1 ); } break; case SID_ATTR_ZOOM: case FID_SCALE: { sal_uInt16 nZoom = 100; bool bCancel = false; eZoom = SvxZoomType::PERCENT; if ( pReqArgs ) { const SvxZoomItem& rZoomItem = static_cast( pReqArgs->Get(SID_ATTR_ZOOM)); eZoom = rZoomItem.GetType(); nZoom = rZoomItem.GetValue(); } else { SfxItemSet aSet ( GetPool(), svl::Items{} ); SvxZoomItem aZoomItem( SvxZoomType::PERCENT, pPreview->GetZoom(), SID_ATTR_ZOOM ); aSet.Put( aZoomItem ); SvxAbstractDialogFactory* pFact = SvxAbstractDialogFactory::Create(); if(pFact) { ScopedVclPtr pDlg(pFact->CreateSvxZoomDialog(nullptr, aSet)); OSL_ENSURE(pDlg, "Dialog creation failed!"); pDlg->SetLimits( 20, 400 ); pDlg->HideButton( ZoomButtonId::OPTIMAL ); bCancel = ( RET_CANCEL == pDlg->Execute() ); if ( !bCancel ) { const SvxZoomItem& rZoomItem = static_cast( pDlg->GetOutputItemSet()-> Get( SID_ATTR_ZOOM )); eZoom = rZoomItem.GetType(); nZoom = rZoomItem.GetValue(); } } } if ( !bCancel ) { switch ( eZoom ) { case SvxZoomType::OPTIMAL: case SvxZoomType::WHOLEPAGE: nZoom = pPreview->GetOptimalZoom(false); break; case SvxZoomType::PAGEWIDTH: nZoom = pPreview->GetOptimalZoom(true); break; default: { // added to avoid warnings } } pPreview->SetZoom( nZoom ); rReq.Done(); } } break; case SID_PREVIEW_ZOOMIN: { sal_uInt16 nNew = pPreview->GetZoom() + 20 ; nNew -= nNew % 20; pPreview->SetZoom( nNew ); eZoom = SvxZoomType::PERCENT; rReq.Done(); } break; case SID_PREVIEW_ZOOMOUT: { sal_uInt16 nNew = pPreview->GetZoom() - 1; nNew -= nNew % 20; pPreview->SetZoom( nNew ); eZoom = SvxZoomType::PERCENT; rReq.Done(); } break; case SID_PREVIEW_MARGIN: { bool bMargin = pPreview->GetPageMargins(); pPreview->SetPageMargins( !bMargin ); pPreview->Invalidate(); rReq.Done(); } break; case SID_ATTR_ZOOMSLIDER: { const SfxPoolItem* pItem; eZoom = SvxZoomType::PERCENT; if( pReqArgs && SfxItemState::SET == pReqArgs->GetItemState( SID_ATTR_ZOOMSLIDER, true, &pItem ) ) { const sal_uInt16 nCurrentZoom = static_cast(pItem)->GetValue(); if( nCurrentZoom ) { pPreview->SetZoom( nCurrentZoom ); rReq.Done(); } } } break; case SID_PREVIEW_SCALINGFACTOR: { const SfxPoolItem* pItem; SCTAB nTab = pPreview->GetTab(); OUString aOldName = pDocShell->GetDocument().GetPageStyle( pPreview->GetTab() ); ScStyleSheetPool* pStylePool = pDocShell->GetDocument().GetStyleSheetPool(); SfxStyleSheetBase* pStyleSheet = pStylePool->Find( aOldName, SfxStyleFamily::Page ); OSL_ENSURE( pStyleSheet, "PageStyle not found! :-/" ); if ( pReqArgs && pStyleSheet && SfxItemState::SET == pReqArgs->GetItemState( SID_PREVIEW_SCALINGFACTOR, true, &pItem ) ) { const sal_uInt16 nCurrentZoom = static_cast(pItem)->GetValue(); SfxItemSet& rSet = pStyleSheet->GetItemSet(); rSet.Put( SfxUInt16Item( ATTR_PAGE_SCALE, nCurrentZoom ) ); ScPrintFunc aPrintFunc( pDocShell, pDocShell->GetPrinter(), nTab ); aPrintFunc.UpdatePages(); rReq.Done(); } GetViewFrame()->GetBindings().Invalidate( nSlot ); } break; case SID_PRINTPREVIEW: case SID_PREVIEW_CLOSE: // print preview is now always in the same frame as the tab view // -> always switch this frame back to normal view // (ScTabViewShell ctor reads stored view data) ExitPreview(); break; case SID_CURSORPAGEUP: case SID_CURSORPAGEDOWN: case SID_CURSORHOME: case SID_CURSOREND: case SID_CURSORUP: case SID_CURSORDOWN: case SID_CURSORLEFT: case SID_CURSORRIGHT: DoScroll( nSlot ); break; case SID_CANCEL: if( ScViewUtil::IsFullScreen( *this ) ) ScViewUtil::SetFullScreen( *this, false ); break; default: break; } } void ScPreviewShell::GetState( SfxItemSet& rSet ) { pPreview->SetInGetState(true); SCTAB nTab = pPreview->GetTab(); long nPage = pPreview->GetPageNo(); long nTotal = pPreview->GetTotalPages(); sal_uInt16 nZoom = pPreview->GetZoom(); bool bAllTested = pPreview->AllTested(); SfxWhichIter aIter(rSet); sal_uInt16 nWhich = aIter.FirstWhich(); while ( nWhich ) { switch (nWhich) { case SID_STATUS_PAGESTYLE: case SID_HFEDIT: pDocShell->GetStatePageStyle( rSet, nTab ); break; case SID_UNDO: case SID_REDO: case SID_REPEAT: case SID_SAVEDOC: case SID_SAVEASDOC: case SID_MAIL_SENDDOC: case SID_VIEW_DATA_SOURCE_BROWSER: case SID_QUITAPP: rSet.DisableItem(nWhich); break; case SID_PREVIEW_PREVIOUS: case SID_PREVIEW_FIRST: if (!nTotal || nPage==0) rSet.DisableItem(nWhich); break; case SID_PREVIEW_NEXT: case SID_PREVIEW_LAST: if (bAllTested) if (!nTotal || nPage==nTotal-1) rSet.DisableItem(nWhich); break; case SID_PREVIEW_ZOOMIN: if (nZoom >= 400) rSet.DisableItem(nWhich); break; case SID_PREVIEW_ZOOMOUT: if (nZoom <= 20) rSet.DisableItem(nWhich); break; case SID_ATTR_ZOOM: { SvxZoomItem aZoom( eZoom, nZoom, nWhich ); aZoom.SetValueSet( SvxZoomEnableFlags::ALL & ~SvxZoomEnableFlags::OPTIMAL ); rSet.Put( aZoom ); } break; case SID_ATTR_ZOOMSLIDER: { SvxZoomSliderItem aZoomSliderItem( nZoom, MINZOOM, MAXZOOM, SID_ATTR_ZOOMSLIDER ); aZoomSliderItem.AddSnappingPoint( 100 ); rSet.Put( aZoomSliderItem ); } break; case SID_PREVIEW_SCALINGFACTOR: { if( pDocShell->IsReadOnly() ) rSet.DisableItem( nWhich ); else { OUString aOldName = pDocShell->GetDocument().GetPageStyle( pPreview->GetTab() ); ScStyleSheetPool* pStylePool = pDocShell->GetDocument().GetStyleSheetPool(); SfxStyleSheetBase* pStyleSheet = pStylePool->Find( aOldName, SfxStyleFamily::Page ); OSL_ENSURE( pStyleSheet, "PageStyle not found! :-/" ); if ( pStyleSheet ) { SfxItemSet& rStyleSet = pStyleSheet->GetItemSet(); sal_uInt16 nCurrentZoom = static_cast(rStyleSet.Get(ATTR_PAGE_SCALE)).GetValue(); if( nCurrentZoom ) { SvxZoomSliderItem aZoomSliderItem( nCurrentZoom, MINZOOM_SLIDER, MAXZOOM_SLIDER, SID_PREVIEW_SCALINGFACTOR ); aZoomSliderItem.AddSnappingPoint( 100 ); rSet.Put( aZoomSliderItem ); } else rSet.DisableItem( nWhich ); } } } break; case SID_STATUS_DOCPOS: rSet.Put( SfxStringItem( nWhich, pPreview->GetPosString() ) ); break; case SID_PRINTPREVIEW: rSet.Put( SfxBoolItem( nWhich, true ) ); break; case SID_FORMATPAGE: case SID_PREVIEW_MARGIN: if( pDocShell->IsReadOnly() ) rSet.DisableItem( nWhich ); break; } nWhich = aIter.NextWhich(); } pPreview->SetInGetState(false); } void ScPreviewShell::FillFieldData( ScHeaderFieldData& rData ) { ScDocument& rDoc = pDocShell->GetDocument(); SCTAB nTab = pPreview->GetTab(); OUString aTmp; rDoc.GetName(nTab, aTmp); rData.aTabName = aTmp; if( pDocShell->getDocProperties()->getTitle().getLength() != 0 ) rData.aTitle = pDocShell->getDocProperties()->getTitle(); else rData.aTitle = pDocShell->GetTitle(); const INetURLObject& rURLObj = pDocShell->GetMedium()->GetURLObject(); rData.aLongDocName = rURLObj.GetMainURL( INetURLObject::DecodeMechanism::Unambiguous ); if ( !rData.aLongDocName.isEmpty() ) rData.aShortDocName = rURLObj.GetName( INetURLObject::DecodeMechanism::Unambiguous ); else rData.aShortDocName = rData.aLongDocName = rData.aTitle; rData.nPageNo = pPreview->GetPageNo() + 1; bool bAllTested = pPreview->AllTested(); if (bAllTested) rData.nTotalPages = pPreview->GetTotalPages(); else rData.nTotalPages = 99; // the dialog knows eNumType } void ScPreviewShell::WriteUserData(OUString& rData, bool /* bBrowse */) { // nZoom // nPageNo rData = OUString::number(pPreview->GetZoom()) + OUStringLiteral1(SC_USERDATA_SEP) + OUString::number(pPreview->GetPageNo()); } void ScPreviewShell::ReadUserData(const OUString& rData, bool /* bBrowse */) { if (!rData.isEmpty()) { sal_Int32 nIndex = 0; pPreview->SetZoom((sal_uInt16)rData.getToken(0, SC_USERDATA_SEP, nIndex).toInt32()); pPreview->SetPageNo(rData.getToken(0, SC_USERDATA_SEP, nIndex).toInt32()); eZoom = SvxZoomType::PERCENT; } } void ScPreviewShell::WriteUserDataSequence(uno::Sequence < beans::PropertyValue >& rSeq) { rSeq.realloc(3); beans::PropertyValue* pSeq = rSeq.getArray(); if(pSeq) { sal_uInt16 nViewID(GetViewFrame()->GetCurViewId()); pSeq[0].Name = SC_VIEWID; pSeq[0].Value <<= SC_VIEW + OUString::number(nViewID); pSeq[1].Name = SC_ZOOMVALUE; pSeq[1].Value <<= sal_Int32 (pPreview->GetZoom()); pSeq[2].Name = "PageNumber"; pSeq[2].Value <<= pPreview->GetPageNo(); } // Common SdrModel processing GetDocument().GetDrawLayer()->WriteUserDataSequence(rSeq); } void ScPreviewShell::ReadUserDataSequence(const uno::Sequence < beans::PropertyValue >& rSeq) { sal_Int32 nCount(rSeq.getLength()); if (nCount) { const beans::PropertyValue* pSeq = rSeq.getConstArray(); if(pSeq) { for(sal_Int32 i = 0; i < nCount; i++, pSeq++) { OUString sName(pSeq->Name); if(sName == SC_ZOOMVALUE) { sal_Int32 nTemp = 0; if (pSeq->Value >>= nTemp) pPreview->SetZoom(sal_uInt16(nTemp)); } else if (sName == "PageNumber") { sal_Int32 nTemp = 0; if (pSeq->Value >>= nTemp) pPreview->SetPageNo(nTemp); } // Fallback to common SdrModel processing else GetDocument().GetDrawLayer()->ReadUserDataSequenceValue(pSeq); } } } } void ScPreviewShell::DoScroll( sal_uInt16 nMode ) { Point aCurPos, aPrevPos; long nHRange = pHorScroll->GetRange().Max(); long nHLine = pHorScroll->GetLineSize(); long nHPage = pHorScroll->GetPageSize(); long nVRange = pVerScroll->GetRange().Max(); long nVLine = pVerScroll->GetLineSize(); long nVPage = pVerScroll->GetPageSize(); aCurPos.X() = pHorScroll->GetThumbPos(); aCurPos.Y() = pVerScroll->GetThumbPos(); aPrevPos = aCurPos; long nThumbPos = pVerScroll->GetThumbPos(); long nRangeMax = pVerScroll->GetRangeMax(); switch( nMode ) { case SID_CURSORUP: if( nMaxVertPos<0 ) { long nPage = pPreview->GetPageNo(); if( nPage>0 ) { SfxViewFrame* pSfxViewFrame = GetViewFrame(); SfxRequest aSfxRequest( pSfxViewFrame, SID_PREVIEW_PREVIOUS ); Execute( aSfxRequest ); } } else aCurPos.Y() -= nVLine; break; case SID_CURSORDOWN: if( nMaxVertPos<0 ) { long nPage = pPreview->GetPageNo(); long nTotal = pPreview->GetTotalPages(); // before testing for last page, make sure all page counts are calculated if ( nPage+1 == nTotal && !pPreview->AllTested() ) { pPreview->CalcAll(); nTotal = pPreview->GetTotalPages(); } if( nPageGetPageNo(); if( nPage>0 ) { SfxViewFrame* pSfxViewFrame = GetViewFrame(); SfxRequest aSfxRequest( pSfxViewFrame, SID_PREVIEW_PREVIOUS ); Execute( aSfxRequest ); aCurPos.Y() = nVRange; } } else aCurPos.Y() -= nVPage; break; case SID_CURSORPAGEDOWN: if( (std::abs(nVPage+nThumbPos-nRangeMax)<10) || nMaxVertPos<0 ) { long nPage = pPreview->GetPageNo(); long nTotal = pPreview->GetTotalPages(); // before testing for last page, make sure all page counts are calculated if ( nPage+1 == nTotal && !pPreview->AllTested() ) { pPreview->CalcAll(); nTotal = pPreview->GetTotalPages(); } if( nPageGetPageNo(); long nTotal = pPreview->GetTotalPages(); if( nTotal && nPage != 0 ) { SfxViewFrame* pSfxViewFrame = GetViewFrame(); SfxRequest aSfxRequest( pSfxViewFrame, SID_PREVIEW_FIRST ); Execute( aSfxRequest ); } } else { aCurPos.Y() = 0; aCurPos.X() = 0; } break; case SID_CURSOREND: if( nMaxVertPos<0 ) { if( !pPreview->AllTested() ) pPreview->CalcAll(); long nPage = pPreview->GetPageNo(); long nTotal = pPreview->GetTotalPages(); if( nTotal && nPage+1 != nTotal ) { SfxViewFrame* pSfxViewFrame = GetViewFrame(); SfxRequest aSfxRequest( pSfxViewFrame, SID_PREVIEW_LAST ); Execute( aSfxRequest ); } } else { aCurPos.Y() = nVRange; aCurPos.X() = nHRange; } break; } // nHRange-nHPage might be negative, that's why we check for < 0 afterwards if( aCurPos.Y() > (nVRange-nVPage) ) aCurPos.Y() = (nVRange-nVPage); if( aCurPos.Y() < 0 ) aCurPos.Y() = 0; if( aCurPos.X() > (nHRange-nHPage) ) aCurPos.X() = (nHRange-nHPage); if( aCurPos.X() < 0 ) aCurPos.X() = 0; if( nMaxVertPos>=0 ) { if( aCurPos.Y() != aPrevPos.Y() ) { pVerScroll->SetThumbPos( aCurPos.Y() ); pPreview->SetYOffset( aCurPos.Y() ); } } if( aCurPos.X() != aPrevPos.X() ) { pHorScroll->SetThumbPos( aCurPos.X() ); pPreview->SetXOffset( aCurPos.X() ); } } void ScPreviewShell::ExitPreview() { GetViewFrame()->GetDispatcher()->Execute(SID_VIEWSHELL0, SfxCallMode::ASYNCHRON); } void ScPreviewShell::AddAccessibilityObject( SfxListener& rObject ) { if (!pAccessibilityBroadcaster) pAccessibilityBroadcaster = new SfxBroadcaster; rObject.StartListening( *pAccessibilityBroadcaster ); } void ScPreviewShell::RemoveAccessibilityObject( SfxListener& rObject ) { if (pAccessibilityBroadcaster) rObject.EndListening( *pAccessibilityBroadcaster ); else { OSL_FAIL("no accessibility broadcaster?"); } } void ScPreviewShell::BroadcastAccessibility( const SfxHint &rHint ) { if (pAccessibilityBroadcaster) pAccessibilityBroadcaster->Broadcast( rHint ); } bool ScPreviewShell::HasAccessibilityObjects() { return pAccessibilityBroadcaster && pAccessibilityBroadcaster->HasListeners(); } const ScPreviewLocationData& ScPreviewShell::GetLocationData() { return pPreview->GetLocationData(); } ScDocument& ScPreviewShell::GetDocument() { return pDocShell->GetDocument(); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */