update dev300-m58
[ooovba.git] / sw / source / core / undo / unattr.cxx
blobfd5b1ee6761eb11cbe7912f99cb8657bc894f77f
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 $
11 * $Revision: 1.21 $
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>
46 #include <txtftn.hxx>
47 #include <fmtornt.hxx>
48 #include <fmtanchr.hxx>
49 #include <fmtfsize.hxx>
50 #include <frmfmt.hxx>
51 #include <fmtcntnt.hxx>
52 #include <ftnidx.hxx>
53 #include <doc.hxx>
54 #include <docary.hxx>
55 #include <swundo.hxx> // fuer die UndoIds
56 #include <pam.hxx>
57 #include <ndtxt.hxx>
58 #include <swtable.hxx>
59 #include <swtblfmt.hxx>
60 #include <undobj.hxx>
61 #include <rolbck.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 )
77 : SwClient( &rFmt )
78 , m_pUndo( 0 )
79 , m_bSaveDrawPt( bSvDrwPt )
83 void SwUndoFmtAttrHelper::Modify( SfxPoolItem* pOld, SfxPoolItem* pNew )
85 if( pOld && pNew )
87 if( POOLATTR_END >= pOld->Which() )
89 if ( GetUndo() )
91 m_pUndo->PutAttr( *pOld );
93 else
95 m_pUndo.reset( new SwUndoFmtAttr( *pOld,
96 *static_cast<SwFmt*>(pRegisteredIn), m_bSaveDrawPt ) );
99 else if ( RES_ATTRSET_CHG == pOld->Which() )
101 if ( GetUndo() )
103 SfxItemIter aIter(
104 *(static_cast<SwAttrSetChg*>(pOld))->GetChgSet() );
105 const SfxPoolItem* pItem = aIter.GetCurItem();
106 while ( pItem )
108 m_pUndo->PutAttr( *pItem );
109 if( aIter.IsAtEnd() )
110 break;
111 pItem = aIter.NextItem();
114 else
116 m_pUndo.reset( new SwUndoFmtAttr(
117 *static_cast<SwAttrSetChg*>(pOld)->GetChgSet(),
118 *static_cast<SwFmt*>(pRegisteredIn), m_bSaveDrawPt ) );
121 else
122 SwClient::Modify( pOld, pNew );
124 else
125 SwClient::Modify( pOld, pNew );
128 // -----------------------------------------------------
130 SwUndoFmtAttr::SwUndoFmtAttr( const SfxItemSet& rOldSet,
131 SwFmt& rChgFmt,
132 bool bSaveDrawPt )
133 : SwUndo( UNDO_INSFMTATTR )
134 , m_pFmt( &rChgFmt )
135 // --> OD 2007-07-11 #i56253#
136 , m_pOldSet( new SfxItemSet( rOldSet ) )
137 // <--
138 , m_nNodeIndex( 0 )
139 , m_nFmtWhich( rChgFmt.Which() )
140 , m_bSaveDrawPt( bSaveDrawPt )
142 Init();
145 SwUndoFmtAttr::SwUndoFmtAttr( const SfxPoolItem& rItem, SwFmt& rChgFmt,
146 bool bSaveDrawPt )
147 : SwUndo( UNDO_INSFMTATTR )
148 , m_pFmt( &rChgFmt )
149 , m_pOldSet( m_pFmt->GetAttrSet().Clone( FALSE ) )
150 , m_nNodeIndex( 0 )
151 , m_nFmtWhich( rChgFmt.Which() )
152 , m_bSaveDrawPt( bSaveDrawPt )
154 m_pOldSet->Put( rItem );
155 Init();
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 )) );
174 if ( pTbl )
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 )));
189 if ( pTblBox )
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() ))
207 return;
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
220 SaveFlyAnchor();
222 else
224 // Anchor attribute not restored due to invalid anchor position.
225 // Thus, delete anchor attribute.
226 m_pOldSet->ClearItem( RES_ANCHOR );
230 if ( !bAnchorAttrRestored )
231 // <--
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;
240 else
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 )
259 case RES_TXTFMTCOLL:
260 nPos = pDoc->GetTxtFmtColls()->GetPos(
261 static_cast<const SwTxtFmtCollPtr>(m_pFmt) );
262 break;
264 case RES_GRFFMTCOLL:
265 nPos = pDoc->GetGrfFmtColls()->GetPos(
266 static_cast<const SwGrfFmtCollPtr>(m_pFmt) );
267 break;
269 case RES_CHRFMT:
270 nPos = pDoc->GetCharFmts()->GetPos(
271 static_cast<SwCharFmtPtr>(m_pFmt) );
272 break;
274 case RES_FRMFMT:
275 if ( m_nNodeIndex && (m_nNodeIndex < pDoc->GetNodes().Count()) )
277 SwNode* pNd = pDoc->GetNodes()[ m_nNodeIndex ];
278 if ( pNd->IsTableNode() )
280 m_pFmt =
281 static_cast<SwTableNode*>(pNd)->GetTable().GetFrmFmt();
282 nPos = 0;
283 break;
285 else if ( pNd->IsSectionNode() )
287 m_pFmt =
288 static_cast<SwSectionNode*>(pNd)->GetSection().GetFmt();
289 nPos = 0;
290 break;
292 else if ( pNd->IsStartNode() && (SwTableBoxStartNode ==
293 static_cast< SwStartNode* >(pNd)->GetStartNodeType()) )
295 SwTableNode* pTblNode = pNd->FindTableNode();
296 if ( pTblNode )
298 SwTableBox* pBox =
299 pTblNode->GetTable().GetTblBox( m_nNodeIndex );
300 if ( pBox )
302 m_pFmt = pBox->GetFrmFmt();
303 nPos = 0;
304 break;
309 // no break!
310 case RES_DRAWFRMFMT:
311 case RES_FLYFRMFMT:
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) );
319 break;
322 if ( USHRT_MAX == nPos )
324 // Format does not exist; reset
325 m_pFmt = 0;
328 return 0 != m_pFmt;
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(..)>
341 Undo( rUndoIter );
342 // <--
345 void SwUndoFmtAttr::Repeat( SwUndoIter& rUndoIter)
347 if ( !m_pOldSet.get() )
348 return;
350 if ( UNDO_INSFMTATTR == rUndoIter.GetLastUndoId())
352 SwUndoFmtAttr* pLast
353 = static_cast<SwUndoFmtAttr*>(rUndoIter.pLastUndoObj);
354 if (pLast->m_pOldSet.get() && pLast->m_pFmt)
356 return;
360 switch ( m_nFmtWhich )
362 case RES_GRFFMTCOLL:
364 SwNoTxtNode * pNd = rUndoIter.pAktPam->GetNode()->GetNoTxtNode();
365 if( pNd )
367 rUndoIter.GetDoc().SetAttr( m_pFmt->GetAttrSet(),
368 *pNd->GetFmtColl() );
371 break;
373 case RES_TXTFMTCOLL:
375 SwTxtNode * pNd = rUndoIter.pAktPam->GetNode()->GetTxtNode();
376 if( pNd )
378 rUndoIter.GetDoc().SetAttr( m_pFmt->GetAttrSet(),
379 *pNd->GetFmtColl() );
382 break;
384 // case RES_CHRFMT:
385 // case RES_FRMFMT:
387 case RES_FLYFRMFMT:
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();
394 if( pFly )
396 // Bug 43672: es duerfen nicht alle Attribute gesetzt werden!
397 if (SFX_ITEM_SET ==
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 );
405 else
407 rUndoIter.GetDoc().SetAttr( m_pFmt->GetAttrSet(), *pFly );
410 break;
414 rUndoIter.pLastUndoObj = this;
417 SwRewriter SwUndoFmtAttr::GetRewriter() const
419 SwRewriter aRewriter;
421 if (m_pFmt)
423 aRewriter.AddRule(UNDO_ARG1, m_pFmt->GetName());
426 return aRewriter;
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
441 if( bSvDrwPt )
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() ) );
450 /* else
452 pOldSet->Put( pFmt->GetVertOrient() );
453 pOldSet->Put( pFmt->GetHoriOrient() );
455 */ }
457 const SwFmtAnchor& rAnchor =
458 static_cast<const SwFmtAnchor&>( m_pOldSet->Get( RES_ANCHOR, FALSE ) );
459 if( !rAnchor.GetCntntAnchor() )
460 return;
462 xub_StrLen nCntnt = 0;
463 switch( rAnchor.GetAnchorId() )
465 case FLY_IN_CNTNT:
466 case FLY_AUTO_CNTNT:
467 nCntnt = rAnchor.GetCntntAnchor()->nContent.GetIndex();
468 case FLY_AT_CNTNT:
469 case FLY_AT_FLY:
470 m_nNodeIndex = rAnchor.GetCntntAnchor()->nNode.GetIndex();
471 break;
472 default:
473 return;
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
503 return false;
504 // <--
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
517 return false;
518 // <--
521 aNewAnchor.SetAnchor( &aPos );
523 else
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();
543 else
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() )
553 // <--
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 );
564 #ifndef PRODUCT
565 ASSERT( pHnt && pHnt->Which() == RES_TXTATR_FLYCNT,
566 "Missing FlyInCnt-Hint." );
567 ASSERT( pHnt && pHnt->GetFlyCnt().GetFrmFmt() == pFrmFmt,
568 "Wrong TxtFlyCnt-Hint." );
569 #endif
570 const_cast<SwFmtFlyCnt&>(pHnt->GetFlyCnt()).SetFlyFmt();
572 //Die Verbindung ist geloest, jetzt muss noch das Attribut vernichtet
573 //werden.
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;
587 else
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.
613 m_pOldSet->Put(
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() )
627 pFrmFmt->MakeFrms();
629 rIter.pSelFmt = pFrmFmt;
631 // --> OD 2004-10-26 #i35443# - anchor attribute restored.
632 return true;
633 // <--
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 )
644 , m_pOldItem( 0 )
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 );
672 // <--
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 )
698 // reset old values
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();
707 if( pTNd )
709 SwIndex aIdx( pTNd, nSttCntnt );
710 pTNd->DontExpandFmt( aIdx, FALSE );
714 // setze noch den Cursor auf den Undo-Bereich
715 SetPaM( rUndoIter );
718 void SwUndoResetAttr::Redo( SwUndoIter& rUndoIter )
720 // setze Attribut in dem Bereich:
721 SetPaM( rUndoIter );
722 SwDoc& rDoc = rUndoIter.GetDoc();
723 rUndoIter.pLastUndoObj = 0;
724 SvUShortsSort* pIdArr = m_Ids.Count() ? &m_Ids : 0;
726 switch ( m_nFormatId )
728 case RES_CHRFMT:
729 rUndoIter.GetDoc().RstTxtAttrs( *rUndoIter.pAktPam );
730 break;
731 case RES_TXTFMTCOLL:
732 rUndoIter.GetDoc().ResetAttrs( *rUndoIter.pAktPam, FALSE, pIdArr );
733 break;
734 case RES_CONDTXTFMTCOLL:
735 rUndoIter.GetDoc().ResetAttrs( *rUndoIter.pAktPam, TRUE, pIdArr );
737 break;
738 case RES_TXTATR_TOXMARK:
739 // special treatment for TOXMarks
741 SwTOXMarks aArr;
742 SwNodeIndex aIdx( rDoc.GetNodes(), nSttNode );
743 SwPosition aPos( aIdx, SwIndex( aIdx.GetNode().GetCntntNode(),
744 nSttCntnt ));
746 USHORT nCnt = rDoc.GetCurTOXMark( aPos, aArr );
747 if( nCnt )
749 if( 1 < nCnt )
751 // search for the right one
752 SwHistoryHint* pHHint = (GetHistory())[ 0 ];
753 if( pHHint && HSTRY_SETTOXMARKHNT == pHHint->Which() )
755 while( nCnt )
757 if ( static_cast<SwHistorySetTOXMark*>(pHHint)
758 ->IsEqual( *aArr[ --nCnt ] ) )
760 ++nCnt;
761 break;
765 else
766 nCnt = 0;
768 // gefunden, also loeschen
769 if( nCnt-- )
770 rDoc.Delete( aArr[ nCnt ] );
773 break;
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)
783 ->m_nFormatId) ) )
785 return;
788 SvUShortsSort* pIdArr = m_Ids.Count() ? &m_Ids : 0;
789 switch ( m_nFormatId )
791 case RES_CHRFMT:
792 rUndoIter.GetDoc().RstTxtAttrs( *rUndoIter.pAktPam );
793 break;
794 case RES_TXTFMTCOLL:
795 rUndoIter.GetDoc().ResetAttrs( *rUndoIter.pAktPam, FALSE, pIdArr );
796 break;
797 case RES_CONDTXTFMTCOLL:
798 rUndoIter.GetDoc().ResetAttrs( *rUndoIter.pAktPam, TRUE, pIdArr );
799 break;
801 rUndoIter.pLastUndoObj = this;
805 void SwUndoResetAttr::SetAttrs( const SvUShortsSort& rArr )
807 if ( m_Ids.Count() )
809 m_Ids.Remove( 0, m_Ids.Count() );
811 m_Ids.Insert( &rArr );
814 // -----------------------------------------------------
817 SwUndoAttr::SwUndoAttr( const SwPaM& rRange, const SfxPoolItem& rAttr,
818 USHORT nFlags )
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,
831 USHORT nFlags )
832 : SwUndo( UNDO_INSATTR ), SwUndRng( rRange )
833 , m_AttrSet( rSet )
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() );
864 if ( bIsCntnt )
866 m_nNodeIndex = rPam.GetPoint()->nNode.GetIndex();
870 void SwUndoAttr::Undo( SwUndoIter& rUndoIter )
872 SwDoc* pDoc = &rUndoIter.GetDoc();
874 RemoveIdx( *pDoc );
876 if( IDocumentRedlineAccess::IsRedlineOn( GetRedlineMode() ) )
878 SwPaM& rPam = *rUndoIter.pAktPam;
879 if ( ULONG_MAX != m_nNodeIndex )
881 rPam.DeleteMark();
882 rPam.GetPoint()->nNode = m_nNodeIndex;
883 rPam.GetPoint()->nContent.Assign( rPam.GetCntntNode(), nSttCntnt );
884 rPam.SetMark();
885 rPam.GetPoint()->nContent++;
886 pDoc->DeleteRedline( rPam, false, USHRT_MAX );
888 else
890 // alle Format-Redlines entfernen, werden ggfs. neu gesetzt
891 SetPaM( rUndoIter );
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
909 SetPaM( rUndoIter );
912 int lcl_HasEqualItems( const SfxItemSet& rSet1, const SfxItemSet& rSet2 )
914 int nRet = -1;
915 SfxItemIter aIter1( rSet1 ), aIter2( rSet2 );
916 const SfxPoolItem *pI1 = aIter1.FirstItem(), *pI2 = aIter2.FirstItem();
918 while( pI1 && pI2 )
920 if( pI1->Which() != pI2->Which() ||
921 aIter1.IsAtEnd() != aIter2.IsAtEnd() )
923 nRet = 0;
924 break;
926 if( aIter1.IsAtEnd() )
927 break;
928 pI1 = aIter1.NextItem();
929 pI2 = aIter2.NextItem();
931 return nRet;
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 ))
943 return;
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:
967 SetPaM( rUndoIter );
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 )
981 rPam.SetMark();
982 if ( rPam.Move( fnMoveBackward ) )
984 rDoc.AppendRedline( new SwRedline( *m_pRedlineData, rPam ),
985 true);
987 rPam.DeleteMark();
989 else
991 rDoc.AppendRedline( new SwRedline( *m_pRedlineData, rPam ), true);
994 rDoc.SetRedlineMode_intern( eOld );
996 else
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 ))
1008 return ;
1010 SwHistoryHint* pHstHnt;
1011 SwNodes& rNds = rDoc.GetNodes();
1012 for ( USHORT n = 0; n < m_pHistory->Count(); ++n )
1014 xub_StrLen nCntnt = 0;
1015 ULONG nNode = 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();
1029 break;
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();
1044 break;
1049 break;
1051 default:
1052 break;
1055 if( nNode )
1057 SwTxtNode* pTxtNd = rNds[ nNode ]->GetTxtNode();
1058 if( pTxtNd )
1060 SwIndex aIdx( pTxtNd, nCntnt );
1061 SwTxtAttr * pTxtHt = pTxtNd->GetTxtAttr( aIdx, RES_TXTATR_FTN );
1062 if( pTxtHt )
1064 // ok, dann hole mal die Werte
1065 SwTxtFtn* pFtn = static_cast<SwTxtFtn*>(pTxtHt);
1066 RemoveIdxFromSection( rDoc, pFtn->GetStartNode()->GetIndex() );
1067 return ;
1074 // -----------------------------------------------------
1076 SwUndoDefaultAttr::SwUndoDefaultAttr( const SfxItemSet& rSet )
1077 : SwUndo( UNDO_SETDEFTATTR )
1078 , m_pOldSet( 0 )
1079 , m_pTabStop( 0 )
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 ) );
1091 else
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)
1127 Undo( rUndoIter );
1130 // -----------------------------------------------------
1132 SwUndoMoveLeftMargin::SwUndoMoveLeftMargin(
1133 const SwPaM& rPam, BOOL bFlag, BOOL bMod )
1134 : SwUndo( bFlag ? UNDO_INC_LEFTMARGIN : UNDO_DEC_LEFTMARGIN )
1135 , SwUndRng( rPam )
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 );
1156 SetPaM( rIter );
1159 void SwUndoMoveLeftMargin::Redo( SwUndoIter& rIter )
1161 SwDoc* pDoc = &rIter.GetDoc();
1162 SetPaM( rIter );
1163 pDoc->MoveLeftMargin( *rIter.pAktPam, GetId() == UNDO_INC_LEFTMARGIN,
1164 m_bModulus );
1167 void SwUndoMoveLeftMargin::Repeat( SwUndoIter& rIter )
1169 SwDoc* pDoc = &rIter.GetDoc();
1170 pDoc->MoveLeftMargin( *rIter.pAktPam, GetId() == UNDO_INC_LEFTMARGIN,
1171 m_bModulus );
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() )
1182 , m_Text( rTxt )
1183 , m_nNumber( nNum )
1184 , m_bEndNote( bIsEndNote )
1188 SwUndoChangeFootNote::~SwUndoChangeFootNote()
1192 void SwUndoChangeFootNote::Undo( SwUndoIter& rIter )
1194 SwDoc& rDoc = rIter.GetDoc();
1195 SetPaM( rIter );
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();
1205 SetPaM( rIter );
1206 rDoc.DoUndo( bUndo );
1209 void SwUndoChangeFootNote::Redo( SwUndoIter& rIter )
1211 SetPaM( rIter );
1212 rIter.GetDoc().SetCurFtn( *rIter.pAktPam, m_Text, m_nNumber, m_bEndNote );
1213 SetPaM( rIter );
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() );