1 /*************************************************************************
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5 * Copyright 2008 by Sun Microsystems, Inc.
7 * OpenOffice.org - a multi-platform office productivity suite
9 * $RCSfile: unattr.cxx,v $
13 * This file is part of OpenOffice.org.
15 * OpenOffice.org is free software: you can redistribute it and/or modify
16 * it under the terms of the GNU Lesser General Public License version 3
17 * only, as published by the Free Software Foundation.
19 * OpenOffice.org is distributed in the hope that it will be useful,
20 * but WITHOUT ANY WARRANTY; without even the implied warranty of
21 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22 * GNU Lesser General Public License version 3 for more details
23 * (a copy is included in the LICENSE file that accompanied this code).
25 * You should have received a copy of the GNU Lesser General Public License
26 * version 3 along with OpenOffice.org. If not, see
27 * <http://www.openoffice.org/license.html>
28 * for a copy of the LGPLv3 License.
30 ************************************************************************/
32 // MARKER(update_precomp.py): autogen include statement, do not remove
33 #include "precompiled_sw.hxx"
36 #define _SVSTDARR_USHORTS
37 #define _SVSTDARR_USHORTSSORT
38 #include <hintids.hxx>
39 #include <svx/svdmodel.hxx>
40 #include <svx/tstpitem.hxx>
41 #include <svx/svdpage.hxx>
42 #include <svtools/itemiter.hxx>
45 #include <fmtflcnt.hxx>
47 #include <fmtornt.hxx>
48 #include <fmtanchr.hxx>
49 #include <fmtfsize.hxx>
51 #include <fmtcntnt.hxx>
55 #include <swundo.hxx> // fuer die UndoIds
58 #include <swtable.hxx>
59 #include <swtblfmt.hxx>
62 #include <ndnotxt.hxx>
63 #include <dcontact.hxx>
64 #include <ftninfo.hxx>
65 #include <redline.hxx>
66 #include <section.hxx>
67 #include <charfmt.hxx>
71 inline SwDoc
& SwUndoIter::GetDoc() const
72 { return *pAktPam
->GetDoc(); }
74 // -----------------------------------------------------
76 SwUndoFmtAttrHelper::SwUndoFmtAttrHelper( SwFmt
& rFmt
, bool bSvDrwPt
)
79 , m_bSaveDrawPt( bSvDrwPt
)
83 void SwUndoFmtAttrHelper::Modify( SfxPoolItem
* pOld
, SfxPoolItem
* pNew
)
87 if( POOLATTR_END
>= pOld
->Which() )
91 m_pUndo
->PutAttr( *pOld
);
95 m_pUndo
.reset( new SwUndoFmtAttr( *pOld
,
96 *static_cast<SwFmt
*>(pRegisteredIn
), m_bSaveDrawPt
) );
99 else if ( RES_ATTRSET_CHG
== pOld
->Which() )
104 *(static_cast<SwAttrSetChg
*>(pOld
))->GetChgSet() );
105 const SfxPoolItem
* pItem
= aIter
.GetCurItem();
108 m_pUndo
->PutAttr( *pItem
);
109 if( aIter
.IsAtEnd() )
111 pItem
= aIter
.NextItem();
116 m_pUndo
.reset( new SwUndoFmtAttr(
117 *static_cast<SwAttrSetChg
*>(pOld
)->GetChgSet(),
118 *static_cast<SwFmt
*>(pRegisteredIn
), m_bSaveDrawPt
) );
122 SwClient::Modify( pOld
, pNew
);
125 SwClient::Modify( pOld
, pNew
);
128 // -----------------------------------------------------
130 SwUndoFmtAttr::SwUndoFmtAttr( const SfxItemSet
& rOldSet
,
133 : SwUndo( UNDO_INSFMTATTR
)
135 // --> OD 2007-07-11 #i56253#
136 , m_pOldSet( new SfxItemSet( rOldSet
) )
139 , m_nFmtWhich( rChgFmt
.Which() )
140 , m_bSaveDrawPt( bSaveDrawPt
)
145 SwUndoFmtAttr::SwUndoFmtAttr( const SfxPoolItem
& rItem
, SwFmt
& rChgFmt
,
147 : SwUndo( UNDO_INSFMTATTR
)
149 , m_pOldSet( m_pFmt
->GetAttrSet().Clone( FALSE
) )
151 , m_nFmtWhich( rChgFmt
.Which() )
152 , m_bSaveDrawPt( bSaveDrawPt
)
154 m_pOldSet
->Put( rItem
);
158 void SwUndoFmtAttr::Init()
160 // treat change of anchor specially
161 if ( SFX_ITEM_SET
== m_pOldSet
->GetItemState( RES_ANCHOR
, FALSE
))
163 SaveFlyAnchor( m_bSaveDrawPt
);
165 else if ( RES_FRMFMT
== m_nFmtWhich
)
167 SwDoc
* pDoc
= m_pFmt
->GetDoc();
168 if (USHRT_MAX
!= pDoc
->GetTblFrmFmts()->GetPos(
169 static_cast<const SwFrmFmtPtr
>(m_pFmt
)))
171 // Table Format: save table position, table formats are volatile!
172 SwTable
* pTbl
= static_cast<SwTable
*>(
173 SwClientIter( *m_pFmt
).First( TYPE( SwTable
)) );
176 m_nNodeIndex
= pTbl
->GetTabSortBoxes()[ 0 ]->GetSttNd()
177 ->FindTableNode()->GetIndex();
180 else if (USHRT_MAX
!= pDoc
->GetSections().GetPos(
181 static_cast<const SwSectionFmtPtr
>(m_pFmt
)))
183 m_nNodeIndex
= m_pFmt
->GetCntnt().GetCntntIdx()->GetIndex();
185 else if ( 0 != dynamic_cast< SwTableBoxFmt
* >( m_pFmt
) )
187 SwTableBox
* pTblBox
= static_cast< SwTableBox
* >(
188 SwClientIter( *m_pFmt
).First( TYPE( SwTableBox
)));
191 m_nNodeIndex
= pTblBox
->GetSttIdx();
197 SwUndoFmtAttr::~SwUndoFmtAttr()
201 void SwUndoFmtAttr::Undo( SwUndoIter
& rUndoIter
)
203 // OD 2004-10-26 #i35443#
204 // Important note: <Undo(..)> also called by <ReDo(..)>
206 if ( !m_pOldSet
.get() || !m_pFmt
|| !IsFmtInDoc( &rUndoIter
.GetDoc() ))
209 // --> OD 2004-10-26 #i35443# - If anchor attribute has been successfull
210 // restored, all other attributes are also restored.
211 // Thus, keep track of its restoration
212 bool bAnchorAttrRestored( false );
213 if ( SFX_ITEM_SET
== m_pOldSet
->GetItemState( RES_ANCHOR
, FALSE
))
215 bAnchorAttrRestored
= RestoreFlyAnchor( rUndoIter
);
216 if ( bAnchorAttrRestored
)
218 // Anchor attribute successfull restored.
219 // Thus, keep anchor position for redo
224 // Anchor attribute not restored due to invalid anchor position.
225 // Thus, delete anchor attribute.
226 m_pOldSet
->ClearItem( RES_ANCHOR
);
230 if ( !bAnchorAttrRestored
)
233 SwUndoFmtAttrHelper
aTmp( *m_pFmt
, m_bSaveDrawPt
);
234 m_pFmt
->SetFmtAttr( *m_pOldSet
);
235 if ( aTmp
.GetUndo() )
237 // transfer ownership of helper object's old set
238 m_pOldSet
= aTmp
.GetUndo()->m_pOldSet
;
242 m_pOldSet
->ClearItem();
245 if ( RES_FLYFRMFMT
== m_nFmtWhich
|| RES_DRAWFRMFMT
== m_nFmtWhich
)
247 rUndoIter
.pSelFmt
= static_cast<SwFrmFmt
*>(m_pFmt
);
252 bool SwUndoFmtAttr::IsFmtInDoc( SwDoc
* pDoc
)
254 // search for the Format in the Document; if it does not exist any more,
255 // the attribute is not restored!
256 USHORT nPos
= USHRT_MAX
;
257 switch ( m_nFmtWhich
)
260 nPos
= pDoc
->GetTxtFmtColls()->GetPos(
261 static_cast<const SwTxtFmtCollPtr
>(m_pFmt
) );
265 nPos
= pDoc
->GetGrfFmtColls()->GetPos(
266 static_cast<const SwGrfFmtCollPtr
>(m_pFmt
) );
270 nPos
= pDoc
->GetCharFmts()->GetPos(
271 static_cast<SwCharFmtPtr
>(m_pFmt
) );
275 if ( m_nNodeIndex
&& (m_nNodeIndex
< pDoc
->GetNodes().Count()) )
277 SwNode
* pNd
= pDoc
->GetNodes()[ m_nNodeIndex
];
278 if ( pNd
->IsTableNode() )
281 static_cast<SwTableNode
*>(pNd
)->GetTable().GetFrmFmt();
285 else if ( pNd
->IsSectionNode() )
288 static_cast<SwSectionNode
*>(pNd
)->GetSection().GetFmt();
292 else if ( pNd
->IsStartNode() && (SwTableBoxStartNode
==
293 static_cast< SwStartNode
* >(pNd
)->GetStartNodeType()) )
295 SwTableNode
* pTblNode
= pNd
->FindTableNode();
299 pTblNode
->GetTable().GetTblBox( m_nNodeIndex
);
302 m_pFmt
= pBox
->GetFrmFmt();
312 nPos
= pDoc
->GetSpzFrmFmts()->GetPos(
313 static_cast<const SwFrmFmtPtr
>(m_pFmt
) );
314 if ( USHRT_MAX
== nPos
)
316 nPos
= pDoc
->GetFrmFmts()->GetPos(
317 static_cast<const SwFrmFmtPtr
>(m_pFmt
) );
322 if ( USHRT_MAX
== nPos
)
324 // Format does not exist; reset
331 // prueft, ob es noch im Doc ist!
332 SwFmt
* SwUndoFmtAttr::GetFmt( SwDoc
& rDoc
)
334 return m_pFmt
&& IsFmtInDoc( &rDoc
) ? m_pFmt
: 0;
337 void SwUndoFmtAttr::Redo( SwUndoIter
& rUndoIter
)
339 // --> OD 2004-10-26 #i35443# - Because the undo stores the attributes for
340 // redo, the same code as for <Undo(..)> can be applied for <Redo(..)>
345 void SwUndoFmtAttr::Repeat( SwUndoIter
& rUndoIter
)
347 if ( !m_pOldSet
.get() )
350 if ( UNDO_INSFMTATTR
== rUndoIter
.GetLastUndoId())
353 = static_cast<SwUndoFmtAttr
*>(rUndoIter
.pLastUndoObj
);
354 if (pLast
->m_pOldSet
.get() && pLast
->m_pFmt
)
360 switch ( m_nFmtWhich
)
364 SwNoTxtNode
* pNd
= rUndoIter
.pAktPam
->GetNode()->GetNoTxtNode();
367 rUndoIter
.GetDoc().SetAttr( m_pFmt
->GetAttrSet(),
368 *pNd
->GetFmtColl() );
375 SwTxtNode
* pNd
= rUndoIter
.pAktPam
->GetNode()->GetTxtNode();
378 rUndoIter
.GetDoc().SetAttr( m_pFmt
->GetAttrSet(),
379 *pNd
->GetFmtColl() );
389 // erstal pruefen, ob der Cursor ueberhaupt in einem fliegenden
390 // Rahmen steht. Der Weg ist: suche in allen FlyFrmFormaten
391 // nach dem FlyCntnt-Attribut und teste ob der Cursor in der
392 // entsprechenden Section liegt.
393 SwFrmFmt
* pFly
= rUndoIter
.pAktPam
->GetNode()->GetFlyFmt();
396 // Bug 43672: es duerfen nicht alle Attribute gesetzt werden!
398 m_pFmt
->GetAttrSet().GetItemState( RES_CNTNT
))
400 SfxItemSet
aTmpSet( m_pFmt
->GetAttrSet() );
401 aTmpSet
.ClearItem( RES_CNTNT
);
402 if( aTmpSet
.Count() )
403 rUndoIter
.GetDoc().SetAttr( aTmpSet
, *pFly
);
407 rUndoIter
.GetDoc().SetAttr( m_pFmt
->GetAttrSet(), *pFly
);
414 rUndoIter
.pLastUndoObj
= this;
417 SwRewriter
SwUndoFmtAttr::GetRewriter() const
419 SwRewriter aRewriter
;
423 aRewriter
.AddRule(UNDO_ARG1
, m_pFmt
->GetName());
429 void SwUndoFmtAttr::PutAttr( const SfxPoolItem
& rItem
)
431 m_pOldSet
->Put( rItem
);
432 if ( RES_ANCHOR
== rItem
.Which() )
434 SaveFlyAnchor( m_bSaveDrawPt
);
438 void SwUndoFmtAttr::SaveFlyAnchor( bool bSvDrwPt
)
440 // das Format ist gueltig, sonst wuerde man gar bis hier kommen
443 if ( RES_DRAWFRMFMT
== m_pFmt
->Which() )
445 Point
aPt( static_cast<SwFrmFmt
*>(m_pFmt
)->FindSdrObject()
446 ->GetRelativePos() );
447 // store old value as attribute, to keep SwUndoFmtAttr small
448 m_pOldSet
->Put( SwFmtFrmSize( ATT_VAR_SIZE
, aPt
.X(), aPt
.Y() ) );
452 pOldSet->Put( pFmt->GetVertOrient() );
453 pOldSet->Put( pFmt->GetHoriOrient() );
457 const SwFmtAnchor
& rAnchor
=
458 static_cast<const SwFmtAnchor
&>( m_pOldSet
->Get( RES_ANCHOR
, FALSE
) );
459 if( !rAnchor
.GetCntntAnchor() )
462 xub_StrLen nCntnt
= 0;
463 switch( rAnchor
.GetAnchorId() )
467 nCntnt
= rAnchor
.GetCntntAnchor()->nContent
.GetIndex();
470 m_nNodeIndex
= rAnchor
.GetCntntAnchor()->nNode
.GetIndex();
476 SwFmtAnchor
aAnchor( rAnchor
.GetAnchorId(), nCntnt
);
477 m_pOldSet
->Put( aAnchor
);
480 // --> OD 2004-10-26 #i35443# - Add return value, type <bool>.
481 // Return value indicates, if anchor attribute is restored.
482 // Note: If anchor attribute is restored, all other existing attributes
483 // are also restored.
484 bool SwUndoFmtAttr::RestoreFlyAnchor( SwUndoIter
& rIter
)
486 SwDoc
* pDoc
= &rIter
.GetDoc();
487 SwFlyFrmFmt
* pFrmFmt
= static_cast<SwFlyFrmFmt
*>(m_pFmt
);
488 const SwFmtAnchor
& rAnchor
=
489 static_cast<const SwFmtAnchor
&>( m_pOldSet
->Get( RES_ANCHOR
, FALSE
) );
491 SwFmtAnchor
aNewAnchor( rAnchor
.GetAnchorId() );
492 if( FLY_PAGE
!= rAnchor
.GetAnchorId() )
494 SwNode
* pNd
= pDoc
->GetNodes()[ m_nNodeIndex
];
496 if ( (FLY_AT_FLY
== rAnchor
.GetAnchorId())
497 ? ( !pNd
->IsStartNode() || (SwFlyStartNode
!=
498 static_cast<SwStartNode
*>(pNd
)->GetStartNodeType()) )
499 : !pNd
->IsTxtNode() )
501 // --> OD 2004-10-26 #i35443# - invalid position.
502 // Thus, anchor attribute not restored
507 SwPosition
aPos( *pNd
);
508 if( FLY_IN_CNTNT
== rAnchor
.GetAnchorId() ||
509 FLY_AUTO_CNTNT
== rAnchor
.GetAnchorId() )
511 aPos
.nContent
.Assign( (SwTxtNode
*)pNd
, rAnchor
.GetPageNum() );
512 if ( aPos
.nContent
.GetIndex() >
513 static_cast<SwTxtNode
*>(pNd
)->GetTxt().Len() )
515 // --> OD 2004-10-26 #i35443# - invalid position.
516 // Thus, anchor attribute not restored
521 aNewAnchor
.SetAnchor( &aPos
);
524 aNewAnchor
.SetPageNum( rAnchor
.GetPageNum() );
526 Point aDrawSavePt
, aDrawOldPt
;
527 if( pDoc
->GetRootFrm() )
529 if( RES_DRAWFRMFMT
== pFrmFmt
->Which() )
531 // den alten zwischengespeicherten Wert herausholen.
532 const SwFmtFrmSize
& rOldSize
= static_cast<const SwFmtFrmSize
&>(
533 m_pOldSet
->Get( RES_FRM_SIZE
) );
534 aDrawSavePt
.X() = rOldSize
.GetWidth();
535 aDrawSavePt
.Y() = rOldSize
.GetHeight();
536 m_pOldSet
->ClearItem( RES_FRM_SIZE
);
538 // den akt. wieder zwischenspeichern
539 aDrawOldPt
= pFrmFmt
->FindSdrObject()->GetRelativePos();
540 //JP 08.10.97: ist laut AMA/MA nicht mehr noetig
541 // pCont->DisconnectFromLayout();
545 pFrmFmt
->DelFrms(); // delete Frms
549 const SwFmtAnchor
&rOldAnch
= pFrmFmt
->GetAnchor();
550 // --> OD 2006-03-13 #i54336#
551 // Consider case, that as-character anchored object has moved its anchor position.
552 if ( FLY_IN_CNTNT
== rOldAnch
.GetAnchorId() )
555 //Bei InCntnt's wird es spannend: Das TxtAttribut muss vernichtet
556 //werden. Leider reisst dies neben den Frms auch noch das Format mit
557 //in sein Grab. Um dass zu unterbinden loesen wir vorher die
558 //Verbindung zwischen Attribut und Format.
559 const SwPosition
*pPos
= rOldAnch
.GetCntntAnchor();
560 SwTxtNode
*pTxtNode
= (SwTxtNode
*)&pPos
->nNode
.GetNode();
561 ASSERT( pTxtNode
->HasHints(), "Missing FlyInCnt-Hint." );
562 const xub_StrLen nIdx
= pPos
->nContent
.GetIndex();
563 SwTxtAttr
* pHnt
= pTxtNode
->GetTxtAttr( nIdx
, RES_TXTATR_FLYCNT
);
565 ASSERT( pHnt
&& pHnt
->Which() == RES_TXTATR_FLYCNT
,
566 "Missing FlyInCnt-Hint." );
567 ASSERT( pHnt
&& pHnt
->GetFlyCnt().GetFrmFmt() == pFrmFmt
,
568 "Wrong TxtFlyCnt-Hint." );
570 const_cast<SwFmtFlyCnt
&>(pHnt
->GetFlyCnt()).SetFlyFmt();
572 //Die Verbindung ist geloest, jetzt muss noch das Attribut vernichtet
574 pTxtNode
->Delete( RES_TXTATR_FLYCNT
, nIdx
, nIdx
);
578 m_pOldSet
->Put( aNewAnchor
);
579 SwUndoFmtAttrHelper
aTmp( *m_pFmt
, m_bSaveDrawPt
);
580 m_pFmt
->SetFmtAttr( *m_pOldSet
);
581 if ( aTmp
.GetUndo() )
583 m_nNodeIndex
= aTmp
.GetUndo()->m_nNodeIndex
;
584 // transfer ownership of helper object's old set
585 m_pOldSet
= aTmp
.GetUndo()->m_pOldSet
;
589 m_pOldSet
->ClearItem();
593 if ( RES_DRAWFRMFMT
== pFrmFmt
->Which() )
595 SwDrawContact
*pCont
=
596 static_cast<SwDrawContact
*>(pFrmFmt
->FindContactObj());
597 // das Draw-Model hat auch noch ein Undo-Object fuer die
598 // richtige Position vorbereitet; dieses ist aber relativ.
599 // Darum verhinder hier, das durch setzen des Ankers das
600 // Contact-Object seine Position aendert.
601 //JP 08.10.97: ist laut AMA/MA nicht mehr noetig
602 // pCont->ConnectToLayout();
603 SdrObject
* pObj
= pCont
->GetMaster();
605 if( pCont
->GetAnchorFrm() && !pObj
->IsInserted() )
607 ASSERT( pDoc
->GetDrawModel(), "RestoreFlyAnchor without DrawModel" );
608 pDoc
->GetDrawModel()->GetPage( 0 )->InsertObject( pObj
);
610 pObj
->SetRelativePos( aDrawSavePt
);
612 // den alten Wert wieder zwischenspeichern.
614 SwFmtFrmSize( ATT_VAR_SIZE
, aDrawOldPt
.X(), aDrawOldPt
.Y() ) );
617 if( FLY_IN_CNTNT
== aNewAnchor
.GetAnchorId() )
619 const SwPosition
* pPos
= aNewAnchor
.GetCntntAnchor();
620 SwTxtNode
* pTxtNd
= pPos
->nNode
.GetNode().GetTxtNode();
621 ASSERT( pTxtNd
, "Kein Textnode an dieser Position" );
622 pTxtNd
->InsertItem( SwFmtFlyCnt( pFrmFmt
), pPos
->nContent
.GetIndex(), 0 );
626 if( RES_DRAWFRMFMT
!= pFrmFmt
->Which() )
629 rIter
.pSelFmt
= pFrmFmt
;
631 // --> OD 2004-10-26 #i35443# - anchor attribute restored.
636 // -----------------------------------------------------
638 // --> OD 2008-02-12 #newlistlevelattrs#
639 SwUndoFmtResetAttr::SwUndoFmtResetAttr( SwFmt
& rChangedFormat
,
640 const USHORT nWhichId
)
641 : SwUndo( UNDO_RESETATTR
)
642 , m_pChangedFormat( &rChangedFormat
)
643 , m_nWhichId( nWhichId
)
646 const SfxPoolItem
* pItem
= 0;
647 if (rChangedFormat
.GetItemState( nWhichId
, FALSE
, &pItem
) == SFX_ITEM_SET
)
649 m_pOldItem
.reset( pItem
->Clone() );
653 SwUndoFmtResetAttr::~SwUndoFmtResetAttr()
657 void SwUndoFmtResetAttr::Undo( SwUndoIter
& )
659 if ( m_pOldItem
.get() )
661 m_pChangedFormat
->SetFmtAttr( *m_pOldItem
);
665 void SwUndoFmtResetAttr::Redo( SwUndoIter
& )
667 if ( m_pOldItem
.get() )
669 m_pChangedFormat
->ResetFmtAttr( m_nWhichId
);
674 // -----------------------------------------------------
676 SwUndoResetAttr::SwUndoResetAttr( const SwPaM
& rRange
, USHORT nFmtId
)
677 : SwUndo( UNDO_RESETATTR
), SwUndRng( rRange
)
678 , m_pHistory( new SwHistory
)
679 , m_nFormatId( nFmtId
)
683 SwUndoResetAttr::SwUndoResetAttr( const SwPosition
& rPos
, USHORT nFmtId
)
684 : SwUndo( UNDO_RESETATTR
)
685 , m_pHistory( new SwHistory
)
686 , m_nFormatId( nFmtId
)
688 nSttNode
= nEndNode
= rPos
.nNode
.GetIndex();
689 nSttCntnt
= nEndCntnt
= rPos
.nContent
.GetIndex();
692 SwUndoResetAttr::~SwUndoResetAttr()
696 void SwUndoResetAttr::Undo( SwUndoIter
& rUndoIter
)
699 SwDoc
& rDoc
= rUndoIter
.GetDoc();
700 m_pHistory
->TmpRollback( &rDoc
, 0 );
701 m_pHistory
->SetTmpEnd( m_pHistory
->Count() );
703 if ((RES_CONDTXTFMTCOLL
== m_nFormatId
) &&
704 (nSttNode
== nEndNode
) && (nSttCntnt
== nEndCntnt
))
706 SwTxtNode
* pTNd
= rDoc
.GetNodes()[ nSttNode
]->GetTxtNode();
709 SwIndex
aIdx( pTNd
, nSttCntnt
);
710 pTNd
->DontExpandFmt( aIdx
, FALSE
);
714 // setze noch den Cursor auf den Undo-Bereich
718 void SwUndoResetAttr::Redo( SwUndoIter
& rUndoIter
)
720 // setze Attribut in dem Bereich:
722 SwDoc
& rDoc
= rUndoIter
.GetDoc();
723 rUndoIter
.pLastUndoObj
= 0;
724 SvUShortsSort
* pIdArr
= m_Ids
.Count() ? &m_Ids
: 0;
726 switch ( m_nFormatId
)
729 rUndoIter
.GetDoc().RstTxtAttrs( *rUndoIter
.pAktPam
);
732 rUndoIter
.GetDoc().ResetAttrs( *rUndoIter
.pAktPam
, FALSE
, pIdArr
);
734 case RES_CONDTXTFMTCOLL
:
735 rUndoIter
.GetDoc().ResetAttrs( *rUndoIter
.pAktPam
, TRUE
, pIdArr
);
738 case RES_TXTATR_TOXMARK
:
739 // special treatment for TOXMarks
742 SwNodeIndex
aIdx( rDoc
.GetNodes(), nSttNode
);
743 SwPosition
aPos( aIdx
, SwIndex( aIdx
.GetNode().GetCntntNode(),
746 USHORT nCnt
= rDoc
.GetCurTOXMark( aPos
, aArr
);
751 // search for the right one
752 SwHistoryHint
* pHHint
= (GetHistory())[ 0 ];
753 if( pHHint
&& HSTRY_SETTOXMARKHNT
== pHHint
->Which() )
757 if ( static_cast<SwHistorySetTOXMark
*>(pHHint
)
758 ->IsEqual( *aArr
[ --nCnt
] ) )
768 // gefunden, also loeschen
770 rDoc
.Delete( aArr
[ nCnt
] );
775 rUndoIter
.pLastUndoObj
= 0;
778 void SwUndoResetAttr::Repeat( SwUndoIter
& rUndoIter
)
780 if ( (RES_FMT_BEGIN
> m_nFormatId
) ||
781 ( (UNDO_RESETATTR
== rUndoIter
.GetLastUndoId()) &&
782 (m_nFormatId
== static_cast<SwUndoResetAttr
*>(rUndoIter
.pLastUndoObj
)
788 SvUShortsSort
* pIdArr
= m_Ids
.Count() ? &m_Ids
: 0;
789 switch ( m_nFormatId
)
792 rUndoIter
.GetDoc().RstTxtAttrs( *rUndoIter
.pAktPam
);
795 rUndoIter
.GetDoc().ResetAttrs( *rUndoIter
.pAktPam
, FALSE
, pIdArr
);
797 case RES_CONDTXTFMTCOLL
:
798 rUndoIter
.GetDoc().ResetAttrs( *rUndoIter
.pAktPam
, TRUE
, pIdArr
);
801 rUndoIter
.pLastUndoObj
= this;
805 void SwUndoResetAttr::SetAttrs( const SvUShortsSort
& rArr
)
809 m_Ids
.Remove( 0, m_Ids
.Count() );
811 m_Ids
.Insert( &rArr
);
814 // -----------------------------------------------------
817 SwUndoAttr::SwUndoAttr( const SwPaM
& rRange
, const SfxPoolItem
& rAttr
,
819 : SwUndo( UNDO_INSATTR
), SwUndRng( rRange
)
820 , m_AttrSet( rRange
.GetDoc()->GetAttrPool(), rAttr
.Which(), rAttr
.Which() )
821 , m_pHistory( new SwHistory
)
822 , m_pRedlineData( 0 )
823 , m_pRedlineSaveData( 0 )
824 , m_nNodeIndex( ULONG_MAX
)
825 , m_nInsertFlags( nFlags
)
827 m_AttrSet
.Put( rAttr
);
830 SwUndoAttr::SwUndoAttr( const SwPaM
& rRange
, const SfxItemSet
& rSet
,
832 : SwUndo( UNDO_INSATTR
), SwUndRng( rRange
)
834 , m_pHistory( new SwHistory
)
835 , m_pRedlineData( 0 )
836 , m_pRedlineSaveData( 0 )
837 , m_nNodeIndex( ULONG_MAX
)
838 , m_nInsertFlags( nFlags
)
842 SwUndoAttr::~SwUndoAttr()
846 void SwUndoAttr::SaveRedlineData( const SwPaM
& rPam
, BOOL bIsCntnt
)
848 SwDoc
* pDoc
= rPam
.GetDoc();
849 if ( pDoc
->IsRedlineOn() )
851 m_pRedlineData
.reset( new SwRedlineData( bIsCntnt
852 ? nsRedlineType_t::REDLINE_INSERT
853 : nsRedlineType_t::REDLINE_FORMAT
,
854 pDoc
->GetRedlineAuthor() ) );
857 m_pRedlineSaveData
.reset( new SwRedlineSaveDatas
);
858 if ( !FillSaveDataForFmt( rPam
, *m_pRedlineSaveData
))
860 m_pRedlineSaveData
.reset(0);
863 SetRedlineMode( pDoc
->GetRedlineMode() );
866 m_nNodeIndex
= rPam
.GetPoint()->nNode
.GetIndex();
870 void SwUndoAttr::Undo( SwUndoIter
& rUndoIter
)
872 SwDoc
* pDoc
= &rUndoIter
.GetDoc();
876 if( IDocumentRedlineAccess::IsRedlineOn( GetRedlineMode() ) )
878 SwPaM
& rPam
= *rUndoIter
.pAktPam
;
879 if ( ULONG_MAX
!= m_nNodeIndex
)
882 rPam
.GetPoint()->nNode
= m_nNodeIndex
;
883 rPam
.GetPoint()->nContent
.Assign( rPam
.GetCntntNode(), nSttCntnt
);
885 rPam
.GetPoint()->nContent
++;
886 pDoc
->DeleteRedline( rPam
, false, USHRT_MAX
);
890 // alle Format-Redlines entfernen, werden ggfs. neu gesetzt
892 pDoc
->DeleteRedline( rPam
, false, nsRedlineType_t::REDLINE_FORMAT
);
893 if ( m_pRedlineSaveData
.get() )
895 SetSaveData( *pDoc
, *m_pRedlineSaveData
);
900 const bool bToLast
= (1 == m_AttrSet
.Count())
901 && (RES_TXTATR_FIELD
<= *m_AttrSet
.GetRanges())
902 && (*m_AttrSet
.GetRanges() <= RES_TXTATR_HARDBLANK
);
904 // restore old values
905 m_pHistory
->TmpRollback( pDoc
, 0, !bToLast
);
906 m_pHistory
->SetTmpEnd( m_pHistory
->Count() );
908 // set cursor onto Undo area
912 int lcl_HasEqualItems( const SfxItemSet
& rSet1
, const SfxItemSet
& rSet2
)
915 SfxItemIter
aIter1( rSet1
), aIter2( rSet2
);
916 const SfxPoolItem
*pI1
= aIter1
.FirstItem(), *pI2
= aIter2
.FirstItem();
920 if( pI1
->Which() != pI2
->Which() ||
921 aIter1
.IsAtEnd() != aIter2
.IsAtEnd() )
926 if( aIter1
.IsAtEnd() )
928 pI1
= aIter1
.NextItem();
929 pI2
= aIter2
.NextItem();
934 void SwUndoAttr::Repeat( SwUndoIter
& rUndoIter
)
936 if ( UNDO_INSATTR
== rUndoIter
.GetLastUndoId() )
938 SwUndoAttr
* pLast
= static_cast<SwUndoAttr
*>(rUndoIter
.pLastUndoObj
);
939 if ((pLast
->m_AttrSet
.Count() == m_AttrSet
.Count()) &&
940 (pLast
->m_nInsertFlags
== m_nInsertFlags
) &&
941 lcl_HasEqualItems( m_AttrSet
, pLast
->m_AttrSet
))
948 // RefMarks are not repeat capable
949 if ( SFX_ITEM_SET
!= m_AttrSet
.GetItemState( RES_TXTATR_REFMARK
, FALSE
) )
951 rUndoIter
.GetDoc().Insert( *rUndoIter
.pAktPam
,
952 m_AttrSet
, m_nInsertFlags
);
954 else if ( 1 < m_AttrSet
.Count() )
956 SfxItemSet
aTmpSet( m_AttrSet
);
957 aTmpSet
.ClearItem( RES_TXTATR_REFMARK
);
958 rUndoIter
.GetDoc().Insert( *rUndoIter
.pAktPam
,
959 aTmpSet
, m_nInsertFlags
);
961 rUndoIter
.pLastUndoObj
= this;
964 void SwUndoAttr::Redo( SwUndoIter
& rUndoIter
)
966 // setze Attribut in dem Bereich:
968 SwPaM
& rPam
= *rUndoIter
.pAktPam
;
969 SwDoc
& rDoc
= rUndoIter
.GetDoc();
971 if ( m_pRedlineData
.get() &&
972 IDocumentRedlineAccess::IsRedlineOn( GetRedlineMode() ) )
974 RedlineMode_t eOld
= rDoc
.GetRedlineMode();
975 rDoc
.SetRedlineMode_intern(static_cast<RedlineMode_t
>(
976 eOld
& ~nsRedlineMode_t::REDLINE_IGNORE
));
977 rDoc
.Insert( rPam
, m_AttrSet
, m_nInsertFlags
);
979 if ( ULONG_MAX
!= m_nNodeIndex
)
982 if ( rPam
.Move( fnMoveBackward
) )
984 rDoc
.AppendRedline( new SwRedline( *m_pRedlineData
, rPam
),
991 rDoc
.AppendRedline( new SwRedline( *m_pRedlineData
, rPam
), true);
994 rDoc
.SetRedlineMode_intern( eOld
);
998 rDoc
.Insert( rPam
, m_AttrSet
, m_nInsertFlags
);
1001 rUndoIter
.pLastUndoObj
= 0;
1005 void SwUndoAttr::RemoveIdx( SwDoc
& rDoc
)
1007 if ( SFX_ITEM_SET
!= m_AttrSet
.GetItemState( RES_TXTATR_FTN
, FALSE
))
1010 SwHistoryHint
* pHstHnt
;
1011 SwNodes
& rNds
= rDoc
.GetNodes();
1012 for ( USHORT n
= 0; n
< m_pHistory
->Count(); ++n
)
1014 xub_StrLen nCntnt
= 0;
1016 pHstHnt
= (*m_pHistory
)[ n
];
1017 switch ( pHstHnt
->Which() )
1019 case HSTRY_RESETTXTHNT
:
1021 SwHistoryResetTxt
* pHistoryHint
1022 = static_cast<SwHistoryResetTxt
*>(pHstHnt
);
1023 if ( RES_TXTATR_FTN
== pHistoryHint
->GetWhich() )
1025 nNode
= pHistoryHint
->GetNode();
1026 nCntnt
= pHistoryHint
->GetCntnt();
1031 case HSTRY_RESETATTRSET
:
1033 SwHistoryResetAttrSet
* pHistoryHint
1034 = static_cast<SwHistoryResetAttrSet
*>(pHstHnt
);
1035 nCntnt
= pHistoryHint
->GetCntnt();
1036 if ( STRING_MAXLEN
!= nCntnt
)
1038 const SvUShorts
& rArr
= pHistoryHint
->GetArr();
1039 for ( USHORT i
= rArr
.Count(); i
; )
1041 if ( RES_TXTATR_FTN
== rArr
[ --i
] )
1043 nNode
= pHistoryHint
->GetNode();
1057 SwTxtNode
* pTxtNd
= rNds
[ nNode
]->GetTxtNode();
1060 SwIndex
aIdx( pTxtNd
, nCntnt
);
1061 SwTxtAttr
* pTxtHt
= pTxtNd
->GetTxtAttr( aIdx
, RES_TXTATR_FTN
);
1064 // ok, dann hole mal die Werte
1065 SwTxtFtn
* pFtn
= static_cast<SwTxtFtn
*>(pTxtHt
);
1066 RemoveIdxFromSection( rDoc
, pFtn
->GetStartNode()->GetIndex() );
1074 // -----------------------------------------------------
1076 SwUndoDefaultAttr::SwUndoDefaultAttr( const SfxItemSet
& rSet
)
1077 : SwUndo( UNDO_SETDEFTATTR
)
1081 const SfxPoolItem
* pItem
;
1082 if( SFX_ITEM_SET
== rSet
.GetItemState( RES_PARATR_TABSTOP
, FALSE
, &pItem
) )
1084 // store separately, because it may change!
1085 m_pTabStop
.reset( static_cast<SvxTabStopItem
*>(pItem
->Clone()) );
1086 if ( 1 != rSet
.Count() ) // are there more attributes?
1088 m_pOldSet
.reset( new SfxItemSet( rSet
) );
1093 m_pOldSet
.reset( new SfxItemSet( rSet
) );
1097 SwUndoDefaultAttr::~SwUndoDefaultAttr()
1101 void SwUndoDefaultAttr::Undo( SwUndoIter
& rUndoIter
)
1103 SwDoc
& rDoc
= rUndoIter
.GetDoc();
1104 if ( m_pOldSet
.get() )
1106 SwUndoFmtAttrHelper
aTmp(
1107 *const_cast<SwTxtFmtColl
*>(rDoc
.GetDfltTxtFmtColl()) );
1108 rDoc
.SetDefault( *m_pOldSet
);
1109 m_pOldSet
.reset( 0 );
1110 if ( aTmp
.GetUndo() )
1112 // transfer ownership of helper object's old set
1113 m_pOldSet
= aTmp
.GetUndo()->m_pOldSet
;
1116 if ( m_pTabStop
.get() )
1118 SvxTabStopItem
* pOld
= static_cast<SvxTabStopItem
*>(
1119 rDoc
.GetDefault( RES_PARATR_TABSTOP
).Clone() );
1120 rDoc
.SetDefault( *m_pTabStop
);
1121 m_pTabStop
.reset( pOld
);
1125 void SwUndoDefaultAttr::Redo( SwUndoIter
& rUndoIter
)
1130 // -----------------------------------------------------
1132 SwUndoMoveLeftMargin::SwUndoMoveLeftMargin(
1133 const SwPaM
& rPam
, BOOL bFlag
, BOOL bMod
)
1134 : SwUndo( bFlag
? UNDO_INC_LEFTMARGIN
: UNDO_DEC_LEFTMARGIN
)
1136 , m_pHistory( new SwHistory
)
1137 , m_bModulus( bMod
)
1141 SwUndoMoveLeftMargin::~SwUndoMoveLeftMargin()
1145 void SwUndoMoveLeftMargin::Undo( SwUndoIter
& rIter
)
1147 SwDoc
* pDoc
= &rIter
.GetDoc();
1148 BOOL bUndo
= pDoc
->DoesUndo();
1149 pDoc
->DoUndo( FALSE
);
1151 // restore old values
1152 m_pHistory
->TmpRollback( pDoc
, 0 );
1153 m_pHistory
->SetTmpEnd( m_pHistory
->Count() );
1155 pDoc
->DoUndo( bUndo
);
1159 void SwUndoMoveLeftMargin::Redo( SwUndoIter
& rIter
)
1161 SwDoc
* pDoc
= &rIter
.GetDoc();
1163 pDoc
->MoveLeftMargin( *rIter
.pAktPam
, GetId() == UNDO_INC_LEFTMARGIN
,
1167 void SwUndoMoveLeftMargin::Repeat( SwUndoIter
& rIter
)
1169 SwDoc
* pDoc
= &rIter
.GetDoc();
1170 pDoc
->MoveLeftMargin( *rIter
.pAktPam
, GetId() == UNDO_INC_LEFTMARGIN
,
1172 rIter
.pLastUndoObj
= this;
1175 // -----------------------------------------------------
1177 SwUndoChangeFootNote::SwUndoChangeFootNote(
1178 const SwPaM
& rRange
, const String
& rTxt
,
1179 USHORT nNum
, bool bIsEndNote
)
1180 : SwUndo( UNDO_CHGFTN
), SwUndRng( rRange
)
1181 , m_pHistory( new SwHistory() )
1184 , m_bEndNote( bIsEndNote
)
1188 SwUndoChangeFootNote::~SwUndoChangeFootNote()
1192 void SwUndoChangeFootNote::Undo( SwUndoIter
& rIter
)
1194 SwDoc
& rDoc
= rIter
.GetDoc();
1197 BOOL bUndo
= rDoc
.DoesUndo();
1198 rDoc
.DoUndo( FALSE
);
1200 m_pHistory
->TmpRollback( &rDoc
, 0 );
1201 m_pHistory
->SetTmpEnd( m_pHistory
->Count() );
1203 rDoc
.GetFtnIdxs().UpdateAllFtn();
1206 rDoc
.DoUndo( bUndo
);
1209 void SwUndoChangeFootNote::Redo( SwUndoIter
& rIter
)
1212 rIter
.GetDoc().SetCurFtn( *rIter
.pAktPam
, m_Text
, m_nNumber
, m_bEndNote
);
1216 void SwUndoChangeFootNote::Repeat( SwUndoIter
& rIter
)
1218 SwDoc
& rDoc
= rIter
.GetDoc();
1219 rDoc
.SetCurFtn( *rIter
.pAktPam
, m_Text
, m_nNumber
, m_bEndNote
);
1220 rIter
.pLastUndoObj
= this;
1224 // -----------------------------------------------------
1227 SwUndoFootNoteInfo::SwUndoFootNoteInfo( const SwFtnInfo
&rInfo
)
1228 : SwUndo( UNDO_FTNINFO
)
1229 , m_pFootNoteInfo( new SwFtnInfo( rInfo
) )
1233 SwUndoFootNoteInfo::~SwUndoFootNoteInfo()
1237 void SwUndoFootNoteInfo::Undo( SwUndoIter
&rIter
)
1239 SwDoc
&rDoc
= rIter
.GetDoc();
1240 SwFtnInfo
*pInf
= new SwFtnInfo( rDoc
.GetFtnInfo() );
1241 rDoc
.SetFtnInfo( *m_pFootNoteInfo
);
1242 m_pFootNoteInfo
.reset( pInf
);
1245 void SwUndoFootNoteInfo::Redo( SwUndoIter
&rIter
)
1247 SwDoc
&rDoc
= rIter
.GetDoc();
1248 SwFtnInfo
*pInf
= new SwFtnInfo( rDoc
.GetFtnInfo() );
1249 rDoc
.SetFtnInfo( *m_pFootNoteInfo
);
1250 m_pFootNoteInfo
.reset( pInf
);
1254 // -----------------------------------------------------
1256 SwUndoEndNoteInfo::SwUndoEndNoteInfo( const SwEndNoteInfo
&rInfo
)
1257 : SwUndo( UNDO_FTNINFO
)
1258 , m_pEndNoteInfo( new SwEndNoteInfo( rInfo
) )
1262 SwUndoEndNoteInfo::~SwUndoEndNoteInfo()
1266 void SwUndoEndNoteInfo::Undo( SwUndoIter
&rIter
)
1268 SwDoc
&rDoc
= rIter
.GetDoc();
1269 SwEndNoteInfo
*pInf
= new SwEndNoteInfo( rDoc
.GetEndNoteInfo() );
1270 rDoc
.SetEndNoteInfo( *m_pEndNoteInfo
);
1271 m_pEndNoteInfo
.reset( pInf
);
1274 void SwUndoEndNoteInfo::Redo( SwUndoIter
&rIter
)
1276 SwDoc
&rDoc
= rIter
.GetDoc();
1277 SwEndNoteInfo
*pInf
= new SwEndNoteInfo( rDoc
.GetEndNoteInfo() );
1278 rDoc
.SetEndNoteInfo( *m_pEndNoteInfo
);
1279 m_pEndNoteInfo
.reset( pInf
);
1282 // -----------------------------------------------------
1284 SwUndoDontExpandFmt::SwUndoDontExpandFmt( const SwPosition
& rPos
)
1285 : SwUndo( UNDO_DONTEXPAND
)
1286 , m_nNodeIndex( rPos
.nNode
.GetIndex() )
1287 , m_nContentIndex( rPos
.nContent
.GetIndex() )
1291 void SwUndoDontExpandFmt::Undo( SwUndoIter
& rIter
)
1293 SwPaM
* pPam
= rIter
.pAktPam
;
1294 SwDoc
* pDoc
= pPam
->GetDoc();
1296 SwPosition
& rPos
= *pPam
->GetPoint();
1297 rPos
.nNode
= m_nNodeIndex
;
1298 rPos
.nContent
.Assign( rPos
.nNode
.GetNode().GetCntntNode(), m_nContentIndex
);
1299 pDoc
->DontExpandFmt( rPos
, FALSE
);
1303 void SwUndoDontExpandFmt::Redo( SwUndoIter
& rIter
)
1305 SwPaM
* pPam
= rIter
.pAktPam
;
1306 SwDoc
* pDoc
= pPam
->GetDoc();
1308 SwPosition
& rPos
= *pPam
->GetPoint();
1309 rPos
.nNode
= m_nNodeIndex
;
1310 rPos
.nContent
.Assign( rPos
.nNode
.GetNode().GetCntntNode(), m_nContentIndex
);
1311 pDoc
->DontExpandFmt( rPos
);
1314 void SwUndoDontExpandFmt::Repeat( SwUndoIter
& rIter
)
1316 SwPaM
* pPam
= rIter
.pAktPam
;
1317 SwDoc
* pDoc
= pPam
->GetDoc();
1318 pDoc
->DontExpandFmt( *pPam
->GetPoint() );