/* -*- 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 "undobase.hxx" #include "refundo.hxx" #include "docsh.hxx" #include "tabvwsh.hxx" #include "undoolk.hxx" #include "undodraw.hxx" #include "dbdata.hxx" #include "attrib.hxx" #include "queryparam.hxx" #include "subtotalparam.hxx" #include "bcaslot.hxx" #include "globstr.hrc" #include #include #include ScSimpleUndo::ScSimpleUndo( ScDocShell* pDocSh ) : pDocShell( pDocSh ), pDetectiveUndo( nullptr ) { } ScSimpleUndo::~ScSimpleUndo() { delete pDetectiveUndo; } bool ScSimpleUndo::SetViewMarkData( const ScMarkData& rMarkData ) { if ( IsPaintLocked() ) return false; ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if ( !pViewShell ) return false; pViewShell->SetMarkData( rMarkData ); return true; } bool ScSimpleUndo::Merge( SfxUndoAction *pNextAction ) { // A SdrUndoGroup for updating detective arrows can belong // to each Undo-Action. // DetectiveRefresh is always called next, // the SdrUndoGroup is encapsulated in a ScUndoDraw action. // AddUndoAction is only called with bTryMerg=sal_True // for automatic update. if ( !pDetectiveUndo && dynamic_cast( pNextAction) != nullptr ) { // Take SdrUndoAction from ScUndoDraw Action, // ScUndoDraw is later deleted by the UndoManager ScUndoDraw* pCalcUndo = static_cast(pNextAction); pDetectiveUndo = pCalcUndo->GetDrawUndo(); pCalcUndo->ForgetDrawUndo(); return true; } return false; } void ScSimpleUndo::BeginUndo() { pDocShell->SetInUndo( true ); ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if (pViewShell) pViewShell->HideAllCursors(); // for example due to merged cells // detective updates happened last, must be undone first if (pDetectiveUndo) pDetectiveUndo->Undo(); } void ScSimpleUndo::EndUndo() { pDocShell->SetDocumentModified(); ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if (pViewShell) { pViewShell->UpdateAutoFillMark(); pViewShell->UpdateInputHandler(); pViewShell->ShowAllCursors(); } pDocShell->SetInUndo( false ); } void ScSimpleUndo::BeginRedo() { pDocShell->SetInUndo( true ); //! own Flag for Redo? ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if (pViewShell) pViewShell->HideAllCursors(); // for example due to merged cells } void ScSimpleUndo::EndRedo() { if (pDetectiveUndo) pDetectiveUndo->Redo(); pDocShell->SetDocumentModified(); ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if (pViewShell) { pViewShell->UpdateAutoFillMark(); pViewShell->UpdateInputHandler(); pViewShell->ShowAllCursors(); } pDocShell->SetInUndo( false ); } void ScSimpleUndo::BroadcastChanges( const ScRange& rRange ) { ScDocument& rDoc = pDocShell->GetDocument(); rDoc.BroadcastCells(rRange, SC_HINT_DATACHANGED); } namespace { class SpanBroadcaster : public sc::ColumnSpanSet::ColumnAction { ScDocument& mrDoc; SCTAB mnCurTab; SCCOL mnCurCol; public: explicit SpanBroadcaster( ScDocument& rDoc ) : mrDoc(rDoc), mnCurTab(-1), mnCurCol(-1) {} virtual void startColumn( ScColumn* pCol ) override { mnCurTab = pCol->GetTab(); mnCurCol = pCol->GetCol(); } virtual void execute( SCROW nRow1, SCROW nRow2, bool bVal ) override { if (!bVal) return; ScRange aRange(mnCurCol, nRow1, mnCurTab, mnCurCol, nRow2, mnCurTab); mrDoc.BroadcastCells(aRange, SC_HINT_DATACHANGED); }; }; } void ScSimpleUndo::BroadcastChanges( const DataSpansType& rSpans ) { ScDocument& rDoc = pDocShell->GetDocument(); SpanBroadcaster aBroadcaster(rDoc); DataSpansType::const_iterator it = rSpans.begin(), itEnd = rSpans.end(); for (; it != itEnd; ++it) { const sc::ColumnSpanSet& rSet = *it->second; rSet.executeColumnAction(rDoc, aBroadcaster); } } void ScSimpleUndo::ShowTable( SCTAB nTab ) { ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if (pViewShell) pViewShell->SetTabNo( nTab ); } void ScSimpleUndo::ShowTable( const ScRange& rRange ) { ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if (pViewShell) { SCTAB nStart = rRange.aStart.Tab(); SCTAB nEnd = rRange.aEnd.Tab(); SCTAB nTab = pViewShell->GetViewData().GetTabNo(); if ( nTab < nStart || nTab > nEnd ) // if not in range: pViewShell->SetTabNo( nStart ); // at beginning of the range } } ScBlockUndo::ScBlockUndo( ScDocShell* pDocSh, const ScRange& rRange, ScBlockUndoMode eBlockMode ) : ScSimpleUndo( pDocSh ), aBlockRange( rRange ), eMode( eBlockMode ) { pDrawUndo = GetSdrUndoAction( &pDocShell->GetDocument() ); } ScBlockUndo::~ScBlockUndo() { DeleteSdrUndoAction( pDrawUndo ); } void ScBlockUndo::BeginUndo() { ScSimpleUndo::BeginUndo(); EnableDrawAdjust( &pDocShell->GetDocument(), false ); } void ScBlockUndo::EndUndo() { if (eMode == SC_UNDO_AUTOHEIGHT) AdjustHeight(); EnableDrawAdjust( &pDocShell->GetDocument(), true ); DoSdrUndoAction( pDrawUndo, &pDocShell->GetDocument() ); ShowBlock(); ScSimpleUndo::EndUndo(); } void ScBlockUndo::EndRedo() { if (eMode == SC_UNDO_AUTOHEIGHT) AdjustHeight(); ShowBlock(); ScSimpleUndo::EndRedo(); } bool ScBlockUndo::AdjustHeight() { ScDocument& rDoc = pDocShell->GetDocument(); ScopedVclPtrInstance< VirtualDevice > pVirtDev; Fraction aZoomX( 1, 1 ); Fraction aZoomY = aZoomX; double nPPTX, nPPTY; ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if (pViewShell) { ScViewData& rData = pViewShell->GetViewData(); nPPTX = rData.GetPPTX(); nPPTY = rData.GetPPTY(); aZoomX = rData.GetZoomX(); aZoomY = rData.GetZoomY(); } else { // Leave zoom at 100 nPPTX = ScGlobal::nScreenPPTX; nPPTY = ScGlobal::nScreenPPTY; } sc::RowHeightContext aCxt(nPPTX, nPPTY, aZoomX, aZoomY, pVirtDev); bool bRet = rDoc.SetOptimalHeight( aCxt, aBlockRange.aStart.Row(), aBlockRange.aEnd.Row(), aBlockRange.aStart.Tab()); if (bRet) pDocShell->PostPaint( 0, aBlockRange.aStart.Row(), aBlockRange.aStart.Tab(), MAXCOL, MAXROW, aBlockRange.aEnd.Tab(), PAINT_GRID | PAINT_LEFT ); return bRet; } void ScBlockUndo::ShowBlock() { if ( IsPaintLocked() ) return; ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if (pViewShell) { ShowTable( aBlockRange ); // with multiple sheets in range each of them is good pViewShell->MoveCursorAbs( aBlockRange.aStart.Col(), aBlockRange.aStart.Row(), SC_FOLLOW_JUMP, false, false ); SCTAB nTab = pViewShell->GetViewData().GetTabNo(); ScRange aRange = aBlockRange; aRange.aStart.SetTab( nTab ); aRange.aEnd.SetTab( nTab ); pViewShell->MarkRange( aRange ); // not through SetMarkArea to MarkData, due to possibly lacking paint } } ScMultiBlockUndo::ScMultiBlockUndo( ScDocShell* pDocSh, const ScRangeList& rRanges) : ScSimpleUndo(pDocSh), maBlockRanges(rRanges), meMode(SC_UNDO_SIMPLE) { mpDrawUndo = GetSdrUndoAction( &pDocShell->GetDocument() ); } ScMultiBlockUndo::~ScMultiBlockUndo() { DeleteSdrUndoAction( mpDrawUndo ); } void ScMultiBlockUndo::BeginUndo() { ScSimpleUndo::BeginUndo(); EnableDrawAdjust(&pDocShell->GetDocument(), false); } void ScMultiBlockUndo::EndUndo() { if (meMode == SC_UNDO_AUTOHEIGHT) AdjustHeight(); EnableDrawAdjust(&pDocShell->GetDocument(), true); DoSdrUndoAction(mpDrawUndo, &pDocShell->GetDocument()); ShowBlock(); ScSimpleUndo::EndUndo(); } void ScMultiBlockUndo::EndRedo() { if (meMode == SC_UNDO_AUTOHEIGHT) AdjustHeight(); ShowBlock(); ScSimpleUndo::EndRedo(); } void ScMultiBlockUndo::AdjustHeight() { ScDocument& rDoc = pDocShell->GetDocument(); ScopedVclPtrInstance< VirtualDevice > pVirtDev; Fraction aZoomX( 1, 1 ); Fraction aZoomY = aZoomX; double nPPTX, nPPTY; ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if (pViewShell) { ScViewData& rData = pViewShell->GetViewData(); nPPTX = rData.GetPPTX(); nPPTY = rData.GetPPTY(); aZoomX = rData.GetZoomX(); aZoomY = rData.GetZoomY(); } else { // Leave zoom at 100 nPPTX = ScGlobal::nScreenPPTX; nPPTY = ScGlobal::nScreenPPTY; } sc::RowHeightContext aCxt(nPPTX, nPPTY, aZoomX, aZoomY, pVirtDev); for (size_t i = 0, n = maBlockRanges.size(); i < n; ++i) { const ScRange& r = *maBlockRanges[i]; bool bRet = rDoc.SetOptimalHeight(aCxt, r.aStart.Row(), r.aEnd.Row(), r.aStart.Tab()); if (bRet) pDocShell->PostPaint( 0, r.aStart.Row(), r.aStart.Tab(), MAXCOL, MAXROW, r.aEnd.Tab(), PAINT_GRID | PAINT_LEFT); } } void ScMultiBlockUndo::ShowBlock() { if ( IsPaintLocked() ) return; ScTabViewShell* pViewShell = ScTabViewShell::GetActiveViewShell(); if (!pViewShell) return; if (maBlockRanges.empty()) return; // Move to the sheet of the first range. ScRange aRange = *maBlockRanges.front(); ShowTable(aRange); pViewShell->MoveCursorAbs( aRange.aStart.Col(), aRange.aStart.Row(), SC_FOLLOW_JUMP, false, false); SCTAB nTab = pViewShell->GetViewData().GetTabNo(); aRange.aStart.SetTab(nTab); aRange.aEnd.SetTab(nTab); pViewShell->MarkRange(aRange, false); for (size_t i = 1, n = maBlockRanges.size(); i < n; ++i) { aRange = *maBlockRanges[i]; aRange.aStart.SetTab(nTab); aRange.aEnd.SetTab(nTab); pViewShell->MarkRange(aRange, false, true); } } ScMoveUndo::ScMoveUndo( ScDocShell* pDocSh, ScDocument* pRefDoc, ScRefUndoData* pRefData, ScMoveUndoMode eRefMode ) : ScSimpleUndo( pDocSh ), pRefUndoDoc( pRefDoc ), pRefUndoData( pRefData ), eMode( eRefMode ) { ScDocument& rDoc = pDocShell->GetDocument(); if (pRefUndoData) pRefUndoData->DeleteUnchanged(&rDoc); pDrawUndo = GetSdrUndoAction( &rDoc ); } ScMoveUndo::~ScMoveUndo() { delete pRefUndoData; delete pRefUndoDoc; DeleteSdrUndoAction( pDrawUndo ); } void ScMoveUndo::UndoRef() { ScDocument& rDoc = pDocShell->GetDocument(); ScRange aRange(0,0,0, MAXCOL,MAXROW,pRefUndoDoc->GetTableCount()-1); pRefUndoDoc->CopyToDocument( aRange, InsertDeleteFlags::FORMULA, false, &rDoc, nullptr, false ); if (pRefUndoData) pRefUndoData->DoUndo( &rDoc, (eMode == SC_UNDO_REFFIRST) ); // HACK: ScDragDropUndo is the only one with REFFIRST. // If not, results possibly in a too frequent adjustment // of ChartRefs. Not that pretty, but not too bad either.. } void ScMoveUndo::BeginUndo() { ScSimpleUndo::BeginUndo(); EnableDrawAdjust( &pDocShell->GetDocument(), false ); if (pRefUndoDoc && eMode == SC_UNDO_REFFIRST) UndoRef(); } void ScMoveUndo::EndUndo() { DoSdrUndoAction( pDrawUndo, &pDocShell->GetDocument() ); // must also be called when pointer is null if (pRefUndoDoc && eMode == SC_UNDO_REFLAST) UndoRef(); EnableDrawAdjust( &pDocShell->GetDocument(), true ); ScSimpleUndo::EndUndo(); } ScDBFuncUndo::ScDBFuncUndo( ScDocShell* pDocSh, const ScRange& rOriginal, SdrUndoAction* pDrawUndo ) : ScSimpleUndo( pDocSh ), aOriginalRange( rOriginal ), mpDrawUndo( pDrawUndo ) { pAutoDBRange = pDocSh->GetOldAutoDBRange(); } ScDBFuncUndo::~ScDBFuncUndo() { DeleteSdrUndoAction( mpDrawUndo ); delete pAutoDBRange; } void ScDBFuncUndo::BeginUndo() { ScSimpleUndo::BeginUndo(); DoSdrUndoAction( mpDrawUndo, &pDocShell->GetDocument() ); } void ScDBFuncUndo::EndUndo() { ScSimpleUndo::EndUndo(); if ( pAutoDBRange ) { ScDocument& rDoc = pDocShell->GetDocument(); SCTAB nTab = rDoc.GetVisibleTab(); ScDBData* pNoNameData = rDoc.GetAnonymousDBData(nTab); if (pNoNameData ) { SCCOL nRangeX1; SCROW nRangeY1; SCCOL nRangeX2; SCROW nRangeY2; SCTAB nRangeTab; pNoNameData->GetArea( nRangeTab, nRangeX1, nRangeY1, nRangeX2, nRangeY2 ); pDocShell->DBAreaDeleted( nRangeTab, nRangeX1, nRangeY1, nRangeX2, nRangeY2 ); *pNoNameData = *pAutoDBRange; if ( pAutoDBRange->HasAutoFilter() ) { // restore AutoFilter buttons pAutoDBRange->GetArea( nRangeTab, nRangeX1, nRangeY1, nRangeX2, nRangeY2 ); rDoc.ApplyFlagsTab( nRangeX1, nRangeY1, nRangeX2, nRangeY1, nRangeTab, SC_MF_AUTO ); pDocShell->PostPaint( nRangeX1, nRangeY1, nRangeTab, nRangeX2, nRangeY1, nRangeTab, PAINT_GRID ); } } } } void ScDBFuncUndo::BeginRedo() { RedoSdrUndoAction( mpDrawUndo ); if ( pAutoDBRange ) { // move the database range to this function's position again (see ScDocShell::GetDBData) ScDocument& rDoc = pDocShell->GetDocument(); ScDBData* pNoNameData = rDoc.GetAnonymousDBData(aOriginalRange.aStart.Tab()); if ( pNoNameData ) { SCCOL nRangeX1; SCROW nRangeY1; SCCOL nRangeX2; SCROW nRangeY2; SCTAB nRangeTab; pNoNameData->GetArea( nRangeTab, nRangeX1, nRangeY1, nRangeX2, nRangeY2 ); pDocShell->DBAreaDeleted( nRangeTab, nRangeX1, nRangeY1, nRangeX2, nRangeY2 ); pNoNameData->SetSortParam( ScSortParam() ); pNoNameData->SetQueryParam( ScQueryParam() ); pNoNameData->SetSubTotalParam( ScSubTotalParam() ); pNoNameData->SetArea( aOriginalRange.aStart.Tab(), aOriginalRange.aStart.Col(), aOriginalRange.aStart.Row(), aOriginalRange.aEnd.Col(), aOriginalRange.aEnd.Row() ); pNoNameData->SetByRow( true ); pNoNameData->SetAutoFilter( false ); // header is always set with the operation in redo } } ScSimpleUndo::BeginRedo(); } void ScDBFuncUndo::EndRedo() { ScSimpleUndo::EndRedo(); } ScUndoWrapper::ScUndoWrapper( SfxUndoAction* pUndo ) : pWrappedUndo( pUndo ) { } ScUndoWrapper::~ScUndoWrapper() { delete pWrappedUndo; } void ScUndoWrapper::ForgetWrappedUndo() { pWrappedUndo = nullptr; // don't delete in dtor - pointer must be stored outside } OUString ScUndoWrapper::GetComment() const { if (pWrappedUndo) return pWrappedUndo->GetComment(); return OUString(); } OUString ScUndoWrapper::GetRepeatComment(SfxRepeatTarget& rTarget) const { if (pWrappedUndo) return pWrappedUndo->GetRepeatComment(rTarget); return OUString(); } sal_uInt16 ScUndoWrapper::GetId() const { if (pWrappedUndo) return pWrappedUndo->GetId(); else return 0; } void ScUndoWrapper::SetLinkToSfxLinkUndoAction(SfxLinkUndoAction* pSfxLinkUndoAction) { if (pWrappedUndo) pWrappedUndo->SetLinkToSfxLinkUndoAction(pSfxLinkUndoAction); else SetLinkToSfxLinkUndoAction(pSfxLinkUndoAction); } bool ScUndoWrapper::Merge( SfxUndoAction* pNextAction ) { if (pWrappedUndo) return pWrappedUndo->Merge(pNextAction); else return false; } void ScUndoWrapper::Undo() { if (pWrappedUndo) pWrappedUndo->Undo(); } void ScUndoWrapper::Redo() { if (pWrappedUndo) pWrappedUndo->Redo(); } void ScUndoWrapper::Repeat(SfxRepeatTarget& rTarget) { if (pWrappedUndo) pWrappedUndo->Repeat(rTarget); } bool ScUndoWrapper::CanRepeat(SfxRepeatTarget& rTarget) const { if (pWrappedUndo) return pWrappedUndo->CanRepeat(rTarget); else return false; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */