xref: /trunk/main/sw/source/core/doc/docfmt.cxx (revision dec99bbd)
1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_sw.hxx"
26 
27 
28 #define _ZFORLIST_DECLARE_TABLE
29 #define _SVSTDARR_USHORTSSORT
30 #define _SVSTDARR_USHORTS
31 #include <hintids.hxx>
32 #include <rtl/logfile.hxx>
33 #include <svl/itemiter.hxx>
34 #include <sfx2/app.hxx>
35 #include <editeng/tstpitem.hxx>
36 #include <editeng/eeitem.hxx>
37 #include <editeng/langitem.hxx>
38 #include <editeng/lrspitem.hxx>
39 #include <editeng/brkitem.hxx>
40 #include <svl/whiter.hxx>
41 #ifndef _ZFORLIST_HXX //autogen
42 #define _ZFORLIST_DECLARE_TABLE
43 #include <svl/zforlist.hxx>
44 #endif
45 #include <comphelper/processfactory.hxx>
46 #include <unotools/misccfg.hxx>
47 #include <com/sun/star/i18n/WordType.hdl>
48 #include <fmtpdsc.hxx>
49 #include <fmthdft.hxx>
50 #include <fmtcntnt.hxx>
51 #include <frmatr.hxx>
52 #include <doc.hxx>
53 #include <IDocumentUndoRedo.hxx>
54 #include <rootfrm.hxx>
55 #include <pagefrm.hxx>
56 #include <hints.hxx>			// fuer SwHyphenBug (in SetDefault)
57 #include <ndtxt.hxx>
58 #include <pam.hxx>
59 #include <UndoCore.hxx>
60 #include <UndoAttribute.hxx>
61 #include <ndgrf.hxx>
62 #include <pagedesc.hxx>			// Fuer Sonderbehandlung in InsFrmFmt
63 #include <rolbck.hxx>			// Undo-Attr
64 #include <mvsave.hxx>			// servieren: Veraenderungen erkennen
65 #include <txatbase.hxx>
66 #include <swtable.hxx>
67 #include <swtblfmt.hxx>
68 #include <charfmt.hxx>
69 #include <docary.hxx>
70 #include <paratr.hxx>
71 #include <redline.hxx>
72 #include <reffld.hxx>
73 #include <txtinet.hxx>
74 #include <fmtinfmt.hxx>
75 #include <breakit.hxx>
76 #include <SwStyleNameMapper.hxx>
77 #include <fmtautofmt.hxx>
78 #include <istyleaccess.hxx>
79 #include <SwUndoFmt.hxx>
80 #include <docsh.hxx>
81 
82 using namespace ::com::sun::star::i18n;
83 using namespace ::com::sun::star::lang;
84 using namespace ::com::sun::star::uno;
85 
86 SV_IMPL_PTRARR(SwFrmFmts,SwFrmFmtPtr)
87 SV_IMPL_PTRARR(SwCharFmts,SwCharFmtPtr)
88 
89 //Spezifische Frameformate (Rahmen)
90 SV_IMPL_PTRARR(SwSpzFrmFmts,SwFrmFmtPtr)
91 
92 /*
93  * interne Funktionen
94  */
95 
96 sal_Bool SetTxtFmtCollNext( const SwTxtFmtCollPtr& rpTxtColl, void* pArgs )
97 {
98 	SwTxtFmtColl *pDel = (SwTxtFmtColl*) pArgs;
99 	if ( &rpTxtColl->GetNextTxtFmtColl() == pDel )
100 	{
101 		rpTxtColl->SetNextTxtFmtColl( *rpTxtColl );
102 	}
103 	return sal_True;
104 }
105 
106 /*
107  * Zuruecksetzen der harten Formatierung fuer Text
108  */
109 
110 // Uebergabeparameter fuer _Rst und lcl_SetTxtFmtColl
111 struct ParaRstFmt
112 {
113 	SwFmtColl* pFmtColl;
114 	SwHistory* pHistory;
115 	const SwPosition *pSttNd, *pEndNd;
116 	const SfxItemSet* pDelSet;
117 	sal_uInt16 nWhich;
118     bool bReset;
119     // --> OD 2007-11-06 #i62575#
120     bool bResetListAttrs;
121     // <--
122     bool bResetAll;
123     bool bInclRefToxMark;
124 
125 	bool bKeepOutlineLevelAttr;	//#outline level,add by zhaojianwei
126 
127 	ParaRstFmt( const SwPosition* pStt, const SwPosition* pEnd,
128                 SwHistory* pHst, sal_uInt16 nWhch = 0, const SfxItemSet* pSet = 0 )
129         : pFmtColl(0),
130           pHistory(pHst),
131           pSttNd(pStt),
132           pEndNd(pEnd),
133           pDelSet(pSet),
134           nWhich(nWhch),
135           // --> OD 2007-11-06 #i62675#
136           bReset( false ),
137           bResetListAttrs( false ),
138           // <--
139           bResetAll( true ),
140           bInclRefToxMark( false ),
141 		  bKeepOutlineLevelAttr( false )	//#outline level,add by zhaojianwei
142 	{}
143 
144 	ParaRstFmt( SwHistory* pHst )
145         : pFmtColl(0),
146           pHistory(pHst),
147           pSttNd(0),
148           pEndNd(0),
149           pDelSet(0),
150           nWhich(0),
151           // --> OD 2007-11-06 #i62675#
152           bReset( false ),
153           bResetListAttrs( false ),
154           // <--
155           bResetAll( true ),
156           bInclRefToxMark( false ),
157   		  bKeepOutlineLevelAttr( false )	//#outline level,add by zhaojianwei
158 	{}
159 };
160 
161 /* in pArgs steht die ChrFmtTablle vom Dokument
162  * (wird bei Selectionen am Start/Ende und bei keiner SSelection benoetigt)
163  */
164 
165 sal_Bool lcl_RstTxtAttr( const SwNodePtr& rpNd, void* pArgs )
166 {
167     ParaRstFmt* pPara = (ParaRstFmt*)pArgs;
168     SwTxtNode * pTxtNode = (SwTxtNode*)rpNd->GetTxtNode();
169     if( pTxtNode && pTxtNode->GetpSwpHints() )
170     {
171         SwIndex aSt( pTxtNode, 0 );
172         sal_uInt16 nEnd = pTxtNode->Len();
173 
174         if( &pPara->pSttNd->nNode.GetNode() == pTxtNode &&
175             pPara->pSttNd->nContent.GetIndex() )
176             aSt = pPara->pSttNd->nContent.GetIndex();
177 
178         if( &pPara->pEndNd->nNode.GetNode() == rpNd )
179             nEnd = pPara->pEndNd->nContent.GetIndex();
180 
181         if( pPara->pHistory )
182         {
183             // fuers Undo alle Attribute sichern
184             SwRegHistory aRHst( *pTxtNode, pPara->pHistory );
185             pTxtNode->GetpSwpHints()->Register( &aRHst );
186             pTxtNode->RstTxtAttr( aSt, nEnd - aSt.GetIndex(), pPara->nWhich,
187                                   pPara->pDelSet, pPara->bInclRefToxMark );
188             if( pTxtNode->GetpSwpHints() )
189                 pTxtNode->GetpSwpHints()->DeRegister();
190         }
191         else
192             pTxtNode->RstTxtAttr( aSt, nEnd - aSt.GetIndex(), pPara->nWhich,
193                                   pPara->pDelSet, pPara->bInclRefToxMark );
194     }
195     return sal_True;
196 }
197 
198 sal_Bool lcl_RstAttr( const SwNodePtr& rpNd, void* pArgs )
199 {
200 	ParaRstFmt* pPara = (ParaRstFmt*)pArgs;
201 	SwCntntNode* pNode = (SwCntntNode*)rpNd->GetCntntNode();
202     if( pNode && pNode->HasSwAttrSet() )
203 	{
204         const sal_Bool bLocked = pNode->IsModifyLocked();
205         pNode->LockModify();
206 
207         SwDoc* pDoc = pNode->GetDoc();
208 
209         // --> OD 2008-04-14 #refactorlists#
210         // remove unused attribute RES_LR_SPACE
211         // add list attributes
212 		SfxItemSet aSet( pDoc->GetAttrPool(),
213                          RES_PAGEDESC, RES_BREAK,
214                          RES_PARATR_NUMRULE, RES_PARATR_NUMRULE,
215 						 RES_PARATR_OUTLINELEVEL,RES_PARATR_OUTLINELEVEL,//#outline level,removed by zhaojianwei
216                          RES_PARATR_LIST_BEGIN, RES_PARATR_LIST_END - 1,
217                          0 );
218         const SfxItemSet* pSet = pNode->GetpSwAttrSet();
219 
220         // --> OD 2008-04-15 #refactorlists#
221 //        std::vector<sal_uInt16> aClearWhichIds;
222         SvUShorts aClearWhichIds;
223         // <--
224         // --> OD 2008-04-15 #refactorlists#
225         // restoring all paragraph list attributes
226         {
227             SfxItemSet aListAttrSet( pDoc->GetAttrPool(),
228                                      RES_PARATR_LIST_BEGIN, RES_PARATR_LIST_END - 1,
229                                      0 );
230             aListAttrSet.Set( *pSet );
231             if ( aListAttrSet.Count() )
232             {
233                 aSet.Put( aListAttrSet );
234                 SfxItemIter aIter( aListAttrSet );
235                 const SfxPoolItem* pItem = aIter.GetCurItem();
236                 while( pItem )
237                 {
238                     aClearWhichIds.Insert( pItem->Which(), aClearWhichIds.Count() );
239                     pItem = aIter.NextItem();
240                 }
241             }
242         }
243         // <--
244 
245 		const SfxPoolItem* pItem;
246   //      sal_uInt16 __READONLY_DATA aSavIds[ 3 ] = { RES_PAGEDESC, RES_BREAK,	//#outline level,removed by zhaojianwei
247   //                                              RES_PARATR_NUMRULE };
248 		//for( sal_uInt16 n = 0; n < 3; ++n )
249         sal_uInt16 __READONLY_DATA aSavIds[ 4 ] = { RES_PAGEDESC, RES_BREAK,	//->add by zhaojianwei
250                                                 RES_PARATR_NUMRULE,
251 												RES_PARATR_OUTLINELEVEL };
252 		for( sal_uInt16 n = 0; n < 4; ++n )										//<-end,zhaojianwei
253         {
254 			if( SFX_ITEM_SET == pSet->GetItemState( aSavIds[ n ], sal_False, &pItem ))
255 			{
256                 bool bSave = false;
257 				switch( aSavIds[ n ] )
258 				{
259                     case RES_PAGEDESC:
260                         bSave = 0 != ((SwFmtPageDesc*)pItem)->GetPageDesc();
261                     break;
262                     case RES_BREAK:
263                         bSave = SVX_BREAK_NONE != ((SvxFmtBreakItem*)pItem)->GetBreak();
264                     break;
265                     case RES_PARATR_NUMRULE:
266                     {
267                         bSave = 0 != ((SwNumRuleItem*)pItem)->GetValue().Len();
268                     }
269                     break;
270 					case RES_PARATR_OUTLINELEVEL:				//#outline level,add by zhaojianwei
271 					{
272 						bSave = pPara && pPara->bKeepOutlineLevelAttr;
273 					}
274 					break;										//<-end,zhaojianwei
275 				}
276 				if( bSave )
277 				{
278 					aSet.Put( *pItem );
279                     // --> OD 2008-04-15 #refactorlists#
280 //                    aClearWhichIds.push_back( aSavIds[n] );
281                     aClearWhichIds.Insert( aSavIds[n], aClearWhichIds.Count() );
282                 }
283 			}
284         }
285 
286         // --> OD 2008-04-14 #refactorlists#
287         // do not clear items directly from item set and only clear to be kept
288         // attributes, if no deletion item set is found.
289 //        pNode->ClearItemsFromAttrSet( aClearWhichIds );
290         const bool bKeepAttributes =
291                     !pPara || !pPara->pDelSet || pPara->pDelSet->Count() == 0;
292         if ( bKeepAttributes )
293         {
294             pNode->ResetAttr( aClearWhichIds );
295         }
296         // <--
297 
298         if( !bLocked )
299             pNode->UnlockModify();
300 
301 		if( pPara )
302 		{
303 			SwRegHistory aRegH( pNode, *pNode, pPara->pHistory );
304 
305 			if( pPara->pDelSet && pPara->pDelSet->Count() )
306 			{
307                 // --> OD 2008-04-15 #refactorlists#
308                 ASSERT( !bKeepAttributes,
309                         "<lcl_RstAttr(..)> - certain attributes are kept, but not needed. -> please inform OD" );
310                 // <--
311 				SfxItemIter aIter( *pPara->pDelSet );
312 				pItem = aIter.FirstItem();
313 				while( sal_True )
314 				{
315                     // --> OD 2008-04-14 #refactorlists#
316                     //
317                     if ( ( pItem->Which() != RES_PAGEDESC &&
318                            pItem->Which() != RES_BREAK &&
319                            pItem->Which() != RES_PARATR_NUMRULE ) ||
320                          ( aSet.GetItemState( pItem->Which(), sal_False ) != SFX_ITEM_SET ) )
321                     {
322                         pNode->ResetAttr( pItem->Which() );
323                     }
324                     // <--
325 					if( aIter.IsAtEnd() )
326 						break;
327 					pItem = aIter.NextItem();
328 				}
329 			}
330 			else if( pPara->bResetAll )
331 				pNode->ResetAllAttr();
332 			else
333 				pNode->ResetAttr( RES_PARATR_BEGIN, POOLATTR_END - 1 );
334 		}
335 		else
336 			pNode->ResetAllAttr();
337 
338         // --> OD 2008-04-15 #refactorlists#
339         // only restore saved attributes, if needed
340         if ( bKeepAttributes && aSet.Count() )
341         // <--
342 		{
343             pNode->LockModify();
344 
345             pNode->SetAttr( aSet );
346 
347 			if( !bLocked )
348 				pNode->UnlockModify();
349 		}
350 	}
351 	return sal_True;
352 }
353 
354 void SwDoc::RstTxtAttrs(const SwPaM &rRg, sal_Bool bInclRefToxMark )
355 {
356 	SwHistory* pHst = 0;
357 	SwDataChanged aTmp( rRg, 0 );
358     if (GetIDocumentUndoRedo().DoesUndo())
359     {
360         SwUndoResetAttr* pUndo = new SwUndoResetAttr( rRg, RES_CHRFMT );
361         pHst = &pUndo->GetHistory();
362         GetIDocumentUndoRedo().AppendUndo(pUndo);
363     }
364 	const SwPosition *pStt = rRg.Start(), *pEnd = rRg.End();
365 	ParaRstFmt aPara( pStt, pEnd, pHst );
366     aPara.bInclRefToxMark = ( bInclRefToxMark == sal_True );
367 	GetNodes().ForEach( pStt->nNode.GetIndex(), pEnd->nNode.GetIndex()+1,
368 						lcl_RstTxtAttr, &aPara );
369 	SetModified();
370 }
371 
372 void SwDoc::ResetAttrs( const SwPaM &rRg,
373                         sal_Bool bTxtAttr,
374                         const SvUShortsSort* pAttrs,
375                         // --> OD 2008-11-28 #b96644#
376                         const bool bSendDataChangedEvents )
377                         // <--
378 {
379 	SwPaM* pPam = (SwPaM*)&rRg;
380 	if( !bTxtAttr && pAttrs && pAttrs->Count() &&
381 		RES_TXTATR_END > (*pAttrs)[ 0 ] )
382 		bTxtAttr = sal_True;
383 
384 	if( !rRg.HasMark() )
385 	{
386 		SwTxtNode* pTxtNd = rRg.GetPoint()->nNode.GetNode().GetTxtNode();
387 		if( !pTxtNd )
388 			return ;
389 
390 		pPam = new SwPaM( *rRg.GetPoint() );
391 
392 		SwIndex& rSt = pPam->GetPoint()->nContent;
393 		sal_uInt16 nMkPos, nPtPos = rSt.GetIndex();
394 
395 		// JP 22.08.96: Sonderfall: steht der Crsr in einem URL-Attribut
396 		//				dann wird dessen Bereich genommen
397         SwTxtAttr const*const pURLAttr(
398             pTxtNd->GetTxtAttrAt(rSt.GetIndex(), RES_TXTATR_INETFMT));
399         if (pURLAttr && pURLAttr->GetINetFmt().GetValue().Len())
400 		{
401 			nMkPos = *pURLAttr->GetStart();
402 			nPtPos = *pURLAttr->End();
403 		}
404 		else
405 		{
406 			Boundary aBndry;
407 			if( pBreakIt->GetBreakIter().is() )
408 				aBndry = pBreakIt->GetBreakIter()->getWordBoundary(
409 							pTxtNd->GetTxt(), nPtPos,
410 							pBreakIt->GetLocale( pTxtNd->GetLang( nPtPos ) ),
411 							WordType::ANY_WORD /*ANYWORD_IGNOREWHITESPACES*/,
412 							sal_True );
413 
414 			if( aBndry.startPos < nPtPos && nPtPos < aBndry.endPos )
415 			{
416 				nMkPos = (xub_StrLen)aBndry.startPos;
417 				nPtPos = (xub_StrLen)aBndry.endPos;
418 			}
419 			else
420 			{
421 				nPtPos = nMkPos = rSt.GetIndex();
422 				if( bTxtAttr )
423 					pTxtNd->DontExpandFmt( rSt, sal_True );
424 			}
425 		}
426 
427 		rSt = nMkPos;
428 		pPam->SetMark();
429 		pPam->GetPoint()->nContent = nPtPos;
430 	}
431 
432     // --> OD 2008-11-28 #i96644#
433 //    SwDataChanged aTmp( *pPam, 0 );
434     std::auto_ptr< SwDataChanged > pDataChanged;
435     if ( bSendDataChangedEvents )
436     {
437         pDataChanged.reset( new SwDataChanged( *pPam, 0 ) );
438     }
439     // <--
440 	SwHistory* pHst = 0;
441     if (GetIDocumentUndoRedo().DoesUndo())
442     {
443         SwUndoResetAttr* pUndo = new SwUndoResetAttr( rRg,
444             static_cast<sal_uInt16>(bTxtAttr ? RES_CONDTXTFMTCOLL : RES_TXTFMTCOLL ));
445 		if( pAttrs && pAttrs->Count() )
446         {
447             pUndo->SetAttrs( *pAttrs );
448         }
449         pHst = &pUndo->GetHistory();
450         GetIDocumentUndoRedo().AppendUndo(pUndo);
451     }
452 
453 	const SwPosition *pStt = pPam->Start(), *pEnd = pPam->End();
454 	ParaRstFmt aPara( pStt, pEnd, pHst );
455 
456     // mst: not including META here; it seems attrs with CH_TXTATR are omitted
457     sal_uInt16 __FAR_DATA aResetableSetRange[] = {
458         RES_FRMATR_BEGIN, RES_FRMATR_END-1,
459         RES_CHRATR_BEGIN, RES_CHRATR_END-1,
460         RES_PARATR_BEGIN, RES_PARATR_END-1,
461         // --> OD 2008-02-25 #refactorlists#
462         RES_PARATR_LIST_BEGIN, RES_PARATR_LIST_END-1,
463         // <--
464         RES_TXTATR_INETFMT, RES_TXTATR_INETFMT,
465         RES_TXTATR_CHARFMT, RES_TXTATR_CHARFMT,
466         RES_TXTATR_CJK_RUBY, RES_TXTATR_CJK_RUBY,
467         RES_TXTATR_UNKNOWN_CONTAINER, RES_TXTATR_UNKNOWN_CONTAINER,
468         RES_UNKNOWNATR_BEGIN, RES_UNKNOWNATR_END-1,
469         0
470     };
471 
472 	SfxItemSet aDelSet( GetAttrPool(), aResetableSetRange );
473 	if( pAttrs && pAttrs->Count() )
474 	{
475 		for( sal_uInt16 n = pAttrs->Count(); n; )
476 			if( POOLATTR_END > (*pAttrs)[ --n ] )
477 				aDelSet.Put( *GetDfltAttr( (*pAttrs)[ n ] ));
478 
479 		if( aDelSet.Count() )
480 			aPara.pDelSet = &aDelSet;
481 	}
482 
483 	sal_Bool bAdd = sal_True;
484 	SwNodeIndex aTmpStt( pStt->nNode );
485 	SwNodeIndex aTmpEnd( pEnd->nNode );
486 	if( pStt->nContent.GetIndex() )		// nur ein Teil
487 	{
488 		// dann spaeter aufsetzen und alle CharFmtAttr -> TxtFmtAttr
489 		SwTxtNode* pTNd = aTmpStt.GetNode().GetTxtNode();
490         if( pTNd && pTNd->HasSwAttrSet() && pTNd->GetpSwAttrSet()->Count() )
491 		{
492             if (pHst)
493             {
494                 SwRegHistory history(pTNd, *pTNd, pHst);
495                 pTNd->FmtToTxtAttr(pTNd);
496             }
497             else
498             {
499                 pTNd->FmtToTxtAttr(pTNd);
500             }
501 		}
502 
503 		aTmpStt++;
504 	}
505 	if( pEnd->nContent.GetIndex() == pEnd->nNode.GetNode().GetCntntNode()->Len() )
506 		// dann spaeter aufsetzen und alle CharFmtAttr -> TxtFmtAttr
507 		aTmpEnd++, bAdd = sal_False;
508 	else if( pStt->nNode != pEnd->nNode || !pStt->nContent.GetIndex() )
509 	{
510 		SwTxtNode* pTNd = aTmpEnd.GetNode().GetTxtNode();
511         if( pTNd && pTNd->HasSwAttrSet() && pTNd->GetpSwAttrSet()->Count() )
512 		{
513             if (pHst)
514             {
515                 SwRegHistory history(pTNd, *pTNd, pHst);
516                 pTNd->FmtToTxtAttr(pTNd);
517             }
518             else
519             {
520                 pTNd->FmtToTxtAttr(pTNd);
521             }
522 		}
523 	}
524 
525 	if( aTmpStt < aTmpEnd )
526 		GetNodes().ForEach( pStt->nNode, aTmpEnd, lcl_RstAttr, &aPara );
527 	else if( !rRg.HasMark() )
528 	{
529         aPara.bResetAll = false ;
530 		::lcl_RstAttr( &pStt->nNode.GetNode(), &aPara );
531         aPara.bResetAll = true ;
532 	}
533 
534 	if( bTxtAttr )
535 	{
536 		if( bAdd )
537 			aTmpEnd++;
538 		GetNodes().ForEach( pStt->nNode, aTmpEnd, lcl_RstTxtAttr, &aPara );
539 	}
540 
541 	if( pPam != &rRg )
542 		delete pPam;
543 
544 	SetModified();
545 }
546 
547 #define DELETECHARSETS if ( bDelete ) { delete pCharSet; delete pOtherSet; }
548 
549 // Einfuegen der Hints nach Inhaltsformen;
550 // wird in SwDoc::Insert(..., SwFmtHint &rHt) benutzt
551 
552 static bool lcl_InsAttr(
553     SwDoc *const pDoc,
554     const SwPaM &rRg,
555     const SfxItemSet& rChgSet,
556     const SetAttrMode nFlags,
557     SwUndoAttr *const pUndo,
558     //Modify here for #119405, by easyfan, 2012-05-24
559     const bool bExpandCharToPara=false)
560     //End of modification, by easyfan
561 {
562     // teil die Sets auf (fuer Selektion in Nodes)
563     const SfxItemSet* pCharSet = 0;
564     const SfxItemSet* pOtherSet = 0;
565     bool bDelete = false;
566     bool bCharAttr = false;
567     bool bOtherAttr = false;
568 
569     // Check, if we can work with rChgSet or if we have to create additional SfxItemSets
570     if ( 1 == rChgSet.Count() )
571     {
572         SfxItemIter aIter( rChgSet );
573         const SfxPoolItem* pItem = aIter.FirstItem();
574 
575         if (!IsInvalidItem(pItem))
576         {
577             const sal_uInt16 nWhich = pItem->Which();
578 
579             if ( isCHRATR(nWhich) ||
580                  (RES_TXTATR_CHARFMT == nWhich) ||
581                  (RES_TXTATR_INETFMT == nWhich) ||
582                  (RES_TXTATR_AUTOFMT == nWhich) ||
583                  (RES_TXTATR_UNKNOWN_CONTAINER == nWhich) )
584             {
585                 pCharSet  = &rChgSet;
586                 bCharAttr = true;
587             }
588 
589             if (    isPARATR(nWhich)
590                     || isPARATR_LIST(nWhich)
591                     || isFRMATR(nWhich)
592                     || isGRFATR(nWhich)
593                     || isUNKNOWNATR(nWhich) )
594             {
595                 pOtherSet = &rChgSet;
596                 bOtherAttr = true;
597             }
598         }
599     }
600 
601     // Build new itemset if either
602     // - rChgSet.Count() > 1 or
603     // - The attribute in rChgSet does not belong to one of the above categories
604     if ( !bCharAttr && !bOtherAttr )
605     {
606         SfxItemSet* pTmpCharItemSet = new SfxItemSet( pDoc->GetAttrPool(),
607                                    RES_CHRATR_BEGIN, RES_CHRATR_END-1,
608                                    RES_TXTATR_AUTOFMT, RES_TXTATR_AUTOFMT,
609                                    RES_TXTATR_INETFMT, RES_TXTATR_INETFMT,
610                                    RES_TXTATR_CHARFMT, RES_TXTATR_CHARFMT,
611                RES_TXTATR_UNKNOWN_CONTAINER, RES_TXTATR_UNKNOWN_CONTAINER,
612                                    0 );
613 
614         SfxItemSet* pTmpOtherItemSet = new SfxItemSet( pDoc->GetAttrPool(),
615                                     RES_PARATR_BEGIN, RES_PARATR_END-1,
616                                     RES_PARATR_LIST_BEGIN, RES_PARATR_LIST_END-1,
617                                     RES_FRMATR_BEGIN, RES_FRMATR_END-1,
618                                     RES_GRFATR_BEGIN, RES_GRFATR_END-1,
619                                     RES_UNKNOWNATR_BEGIN, RES_UNKNOWNATR_END-1,
620                                     0 );
621 
622         pTmpCharItemSet->Put( rChgSet );
623         pTmpOtherItemSet->Put( rChgSet );
624 
625         pCharSet = pTmpCharItemSet;
626         pOtherSet = pTmpOtherItemSet;
627 
628         bDelete = true;
629     }
630 
631     SwHistory* pHistory = pUndo ? &pUndo->GetHistory() : 0;
632     bool bRet = false;
633     const SwPosition *pStt = rRg.Start(), *pEnd = rRg.End();
634     SwCntntNode* pNode = pStt->nNode.GetNode().GetCntntNode();
635 
636     if( pNode && pNode->IsTxtNode() )
637     {
638         // -> #i27615#
639         if (rRg.IsInFrontOfLabel())
640         {
641             SwTxtNode * pTxtNd = pNode->GetTxtNode();
642             SwNumRule * pNumRule = pTxtNd->GetNumRule();
643 
644             if ( !pNumRule )
645             {
646                 ASSERT( false,
647                         "<InsAttr(..)> - PaM in front of label, but text node has no numbering rule set. This is a serious defect, please inform OD." );
648                 DELETECHARSETS
649                 return false;
650             }
651 
652             SwNumFmt aNumFmt = pNumRule->Get(static_cast<sal_uInt16>(pTxtNd->GetActualListLevel()));
653             SwCharFmt * pCharFmt =
654                 pDoc->FindCharFmtByName(aNumFmt.GetCharFmtName());
655 
656             if (pCharFmt)
657             {
658                 if (pHistory)
659                     pHistory->Add(pCharFmt->GetAttrSet(), *pCharFmt);
660 
661                 if ( pCharSet )
662                     pCharFmt->SetFmtAttr(*pCharSet);
663             }
664 
665             DELETECHARSETS
666             return true;
667         }
668 
669         const SwIndex& rSt = pStt->nContent;
670 
671         // Attribute ohne Ende haben keinen Bereich
672         if ( !bCharAttr && !bOtherAttr )
673         {
674             SfxItemSet aTxtSet( pDoc->GetAttrPool(),
675                         RES_TXTATR_NOEND_BEGIN, RES_TXTATR_NOEND_END-1 );
676             aTxtSet.Put( rChgSet );
677             if( aTxtSet.Count() )
678             {
679                 SwRegHistory history( pNode, *pNode, pHistory );
680                 bRet = history.InsertItems(
681                     aTxtSet, rSt.GetIndex(), rSt.GetIndex(), nFlags ) || bRet;
682 
683                 if (bRet && (pDoc->IsRedlineOn() || (!pDoc->IsIgnoreRedline()
684                                 && pDoc->GetRedlineTbl().Count())))
685                 {
686                     SwPaM aPam( pStt->nNode, pStt->nContent.GetIndex()-1,
687                                 pStt->nNode, pStt->nContent.GetIndex() );
688 
689                     if( pUndo )
690                         pUndo->SaveRedlineData( aPam, sal_True );
691 
692                     if( pDoc->IsRedlineOn() )
693                         pDoc->AppendRedline( new SwRedline( nsRedlineType_t::REDLINE_INSERT, aPam ), true);
694                     else
695                         pDoc->SplitRedline( aPam );
696                 }
697             }
698         }
699 
700         // TextAttribute mit Ende expandieren nie ihren Bereich
701         if ( !bCharAttr && !bOtherAttr )
702         {
703             // CharFmt wird gesondert behandelt !!!
704             // JP 22.08.96: URL-Attribute auch!!
705             // TEST_TEMP ToDo: AutoFmt!
706             SfxItemSet aTxtSet( pDoc->GetAttrPool(),
707                                 RES_TXTATR_REFMARK, RES_TXTATR_TOXMARK,
708                                 RES_TXTATR_META, RES_TXTATR_METAFIELD,
709                                 RES_TXTATR_CJK_RUBY, RES_TXTATR_CJK_RUBY,
710                                 RES_TXTATR_INPUTFIELD, RES_TXTATR_INPUTFIELD,
711                                 0 );
712 
713             aTxtSet.Put( rChgSet );
714             if( aTxtSet.Count() )
715             {
716                 sal_uInt16 nInsCnt = rSt.GetIndex();
717                 sal_uInt16 nEnd = pStt->nNode == pEnd->nNode
718                                 ? pEnd->nContent.GetIndex()
719                                 : pNode->Len();
720                 SwRegHistory history( pNode, *pNode, pHistory );
721                 bRet = history.InsertItems( aTxtSet, nInsCnt, nEnd, nFlags )
722                        || bRet;
723 
724                 if (bRet && (pDoc->IsRedlineOn() || (!pDoc->IsIgnoreRedline()
725                                 && pDoc->GetRedlineTbl().Count())))
726                 {
727                     // wurde Text-Inhalt eingefuegt? (RefMark/TOXMarks ohne Ende)
728                     sal_Bool bTxtIns = nInsCnt != rSt.GetIndex();
729                     // wurde Inhalt eingefuegt oder ueber die Selektion gesetzt?
730                     SwPaM aPam( pStt->nNode, bTxtIns ? nInsCnt + 1 : nEnd,
731                                 pStt->nNode, nInsCnt );
732                     if( pUndo )
733                         pUndo->SaveRedlineData( aPam, bTxtIns );
734 
735                     if( pDoc->IsRedlineOn() )
736                         pDoc->AppendRedline(
737                             new SwRedline(
738                                 bTxtIns ? nsRedlineType_t::REDLINE_INSERT : nsRedlineType_t::REDLINE_FORMAT, aPam ),
739                                 true);
740                     else if( bTxtIns )
741                         pDoc->SplitRedline( aPam );
742                 }
743             }
744         }
745     }
746 
747     // bei PageDesc's, die am Node gesetzt werden, muss immer das
748     // Auto-Flag gesetzt werden!!
749     if( pOtherSet && pOtherSet->Count() )
750     {
751         SwTableNode* pTblNd;
752         const SwFmtPageDesc* pDesc;
753         if( SFX_ITEM_SET == pOtherSet->GetItemState( RES_PAGEDESC,
754                         sal_False, (const SfxPoolItem**)&pDesc ))
755         {
756             if( pNode )
757             {
758                 // Auto-Flag setzen, nur in Vorlagen ist ohne Auto !
759                 SwFmtPageDesc aNew( *pDesc );
760                 // Bug 38479: AutoFlag wird jetzt in der WrtShell gesetzt
761                 // aNew.SetAuto();
762 
763                 // Tabellen kennen jetzt auch Umbrueche
764                 if( 0 == (nFlags & nsSetAttrMode::SETATTR_APICALL) &&
765                     0 != ( pTblNd = pNode->FindTableNode() ) )
766                 {
767                     SwTableNode* pCurTblNd = pTblNd;
768                     while ( 0 != ( pCurTblNd = pCurTblNd->StartOfSectionNode()->FindTableNode() ) )
769                         pTblNd = pCurTblNd;
770 
771                     // dann am Tabellen Format setzen
772                     SwFrmFmt* pFmt = pTblNd->GetTable().GetFrmFmt();
773                     SwRegHistory aRegH( pFmt, *pTblNd, pHistory );
774                     pFmt->SetFmtAttr( aNew );
775                     bRet = true;
776                 }
777                 else
778                 {
779                     SwRegHistory aRegH( pNode, *pNode, pHistory );
780                     bRet = pNode->SetAttr( aNew ) || bRet;
781                 }
782             }
783 
784             // bOtherAttr = true means that pOtherSet == rChgSet. In this case
785             // we know, that there is only one attribute in pOtherSet. We cannot
786             // perform the following operations, instead we return:
787             if ( bOtherAttr )
788                 return bRet;
789 
790             const_cast<SfxItemSet*>(pOtherSet)->ClearItem( RES_PAGEDESC );
791             if( !pOtherSet->Count() )
792             {
793                 DELETECHARSETS
794                 return bRet;
795             }
796         }
797 
798         // Tabellen kennen jetzt auch Umbrueche
799         const SvxFmtBreakItem* pBreak;
800         if( pNode && 0 == (nFlags & nsSetAttrMode::SETATTR_APICALL) &&
801             0 != (pTblNd = pNode->FindTableNode() ) &&
802             SFX_ITEM_SET == pOtherSet->GetItemState( RES_BREAK,
803                         sal_False, (const SfxPoolItem**)&pBreak ) )
804         {
805             SwTableNode* pCurTblNd = pTblNd;
806             while ( 0 != ( pCurTblNd = pCurTblNd->StartOfSectionNode()->FindTableNode() ) )
807                 pTblNd = pCurTblNd;
808 
809             // dann am Tabellen Format setzen
810             SwFrmFmt* pFmt = pTblNd->GetTable().GetFrmFmt();
811             SwRegHistory aRegH( pFmt, *pTblNd, pHistory );
812             pFmt->SetFmtAttr( *pBreak );
813             bRet = true;
814 
815             // bOtherAttr = true means that pOtherSet == rChgSet. In this case
816             // we know, that there is only one attribute in pOtherSet. We cannot
817             // perform the following operations, instead we return:
818             if ( bOtherAttr )
819                 return bRet;
820 
821             const_cast<SfxItemSet*>(pOtherSet)->ClearItem( RES_BREAK );
822             if( !pOtherSet->Count() )
823             {
824                 DELETECHARSETS
825                 return bRet;
826             }
827         }
828 
829         {
830             // wenns eine PoolNumRule ist, diese ggfs. anlegen
831             const SwNumRuleItem* pRule;
832             sal_uInt16 nPoolId;
833             if( SFX_ITEM_SET == pOtherSet->GetItemState( RES_PARATR_NUMRULE,
834                                 sal_False, (const SfxPoolItem**)&pRule ) &&
835                 !pDoc->FindNumRulePtr( pRule->GetValue() ) &&
836                 USHRT_MAX != (nPoolId = SwStyleNameMapper::GetPoolIdFromUIName ( pRule->GetValue(),
837                                 nsSwGetPoolIdFromName::GET_POOLID_NUMRULE )) )
838                 pDoc->GetNumRuleFromPool( nPoolId );
839         }
840 
841     }
842 
843     if( !rRg.HasMark() )		// kein Bereich
844     {
845         if( !pNode )
846         {
847             DELETECHARSETS
848             return bRet;
849         }
850 
851         if( pNode->IsTxtNode() && pCharSet && pCharSet->Count() )
852         {
853             SwTxtNode* pTxtNd = static_cast<SwTxtNode*>(pNode);
854             const SwIndex& rSt = pStt->nContent;
855             sal_uInt16 nMkPos, nPtPos = rSt.GetIndex();
856             const String& rStr = pTxtNd->GetTxt();
857 
858             // JP 22.08.96: Sonderfall: steht der Crsr in einem URL-Attribut
859             //				dann wird dessen Bereich genommen
860             SwTxtAttr const*const pURLAttr(
861                 pTxtNd->GetTxtAttrAt(rSt.GetIndex(), RES_TXTATR_INETFMT));
862             if (pURLAttr && pURLAttr->GetINetFmt().GetValue().Len())
863             {
864                 nMkPos = *pURLAttr->GetStart();
865                 nPtPos = *pURLAttr->End();
866             }
867             else
868             {
869                 Boundary aBndry;
870                 if( pBreakIt->GetBreakIter().is() )
871                     aBndry = pBreakIt->GetBreakIter()->getWordBoundary(
872                                 pTxtNd->GetTxt(), nPtPos,
873                                 pBreakIt->GetLocale( pTxtNd->GetLang( nPtPos ) ),
874                                 WordType::ANY_WORD /*ANYWORD_IGNOREWHITESPACES*/,
875                                 sal_True );
876 
877                 if( aBndry.startPos < nPtPos && nPtPos < aBndry.endPos )
878                 {
879                     nMkPos = (xub_StrLen)aBndry.startPos;
880                     nPtPos = (xub_StrLen)aBndry.endPos;
881                 }
882                 else
883                     nPtPos = nMkPos = rSt.GetIndex();
884             }
885 
886             // erstmal die zu ueberschreibenden Attribute aus dem
887             // SwpHintsArray entfernen, wenn die Selektion den gesamten
888             // Absatz umspannt. (Diese Attribute werden als FormatAttr.
889             // eingefuegt und verdraengen nie die TextAttr.!)
890             if( !(nFlags & nsSetAttrMode::SETATTR_DONTREPLACE ) &&
891                 pTxtNd->HasHints() && !nMkPos && nPtPos == rStr.Len() )
892             {
893                 SwIndex aSt( pTxtNd );
894                 if( pHistory )
895                 {
896                     // fuers Undo alle Attribute sichern
897                     SwRegHistory aRHst( *pTxtNd, pHistory );
898                     pTxtNd->GetpSwpHints()->Register( &aRHst );
899                     pTxtNd->RstTxtAttr( aSt, nPtPos, 0, pCharSet );
900                     if( pTxtNd->GetpSwpHints() )
901                         pTxtNd->GetpSwpHints()->DeRegister();
902                 }
903                 else
904                     pTxtNd->RstTxtAttr( aSt, nPtPos, 0, pCharSet );
905             }
906 
907             // the SwRegHistory inserts the attribute into the TxtNode!
908             SwRegHistory history( pNode, *pNode, pHistory );
909             bRet = history.InsertItems( *pCharSet, nMkPos, nPtPos, nFlags )
910                 || bRet;
911 
912             if( pDoc->IsRedlineOn() )
913             {
914                 SwPaM aPam( *pNode, nMkPos, *pNode, nPtPos );
915 
916                 if( pUndo )
917                     pUndo->SaveRedlineData( aPam, sal_False );
918                 pDoc->AppendRedline( new SwRedline( nsRedlineType_t::REDLINE_FORMAT, aPam ), true);
919             }
920         }
921         if( pOtherSet && pOtherSet->Count() )
922         {
923             SwRegHistory aRegH( pNode, *pNode, pHistory );
924             bRet = pNode->SetAttr( *pOtherSet ) || bRet;
925         }
926 
927         DELETECHARSETS
928         return bRet;
929     }
930 
931     if( pDoc->IsRedlineOn() && pCharSet && pCharSet->Count() )
932     {
933         if( pUndo )
934             pUndo->SaveRedlineData( rRg, sal_False );
935         pDoc->AppendRedline( new SwRedline( nsRedlineType_t::REDLINE_FORMAT, rRg ), true);
936     }
937 
938     /* jetzt wenn Bereich */
939     sal_uLong nNodes = 0;
940 
941     SwNodeIndex aSt( pDoc->GetNodes() );
942     SwNodeIndex aEnd( pDoc->GetNodes() );
943     SwIndex aCntEnd( pEnd->nContent );
944 
945     if( pNode )
946     {
947         sal_uInt16 nLen = pNode->Len();
948         if( pStt->nNode != pEnd->nNode )
949             aCntEnd.Assign( pNode, nLen );
950 
951         if( pStt->nContent.GetIndex() != 0 || aCntEnd.GetIndex() != nLen )
952         {
953             // the SwRegHistory inserts the attribute into the TxtNode!
954             if( pNode->IsTxtNode() && pCharSet && pCharSet->Count() )
955             {
956                 SwRegHistory history( pNode, *pNode, pHistory );
957                 bRet = history.InsertItems(*pCharSet,
958                         pStt->nContent.GetIndex(), aCntEnd.GetIndex(), nFlags)
959                     || bRet;
960             }
961 
962             if( pOtherSet && pOtherSet->Count() )
963             {
964                 SwRegHistory aRegH( pNode, *pNode, pHistory );
965                 bRet = pNode->SetAttr( *pOtherSet ) || bRet;
966             }
967 
968             // lediglich Selektion in einem Node.
969             if( pStt->nNode == pEnd->nNode )
970             {
971             //Modify here for #119405, by easyfan, 2012-05-24
972             //The data parameter flag: bExpandCharToPara, comes from the data member of SwDoc,
973             //Which is set in SW MS word Binary filter WW8ImplRreader. With this flag on, means that
974             //current setting attribute set is a character range properties set and comes from a MS word
975             //binary file, And the setting range include a paragraph end position (0X0D);
976             //More specifications, as such property inside the character range properties set recorded in
977             //MS word binary file are dealed and inserted into data model (SwDoc) one by one, so we
978             //only dealing the scenario that the char properties set with 1 item inside;
979 
980                     if (bExpandCharToPara && pCharSet && pCharSet->Count() ==1 )
981             {
982                 SwTxtNode* pCurrentNd = pStt->nNode.GetNode().GetTxtNode();
983 
984                 if (pCurrentNd)
985                 {
986                     pCurrentNd->TryCharSetExpandToNum(*pCharSet);
987 
988                 }
989             }
990             //End of modification, by easyfan
991                 DELETECHARSETS
992                 return bRet;
993             }
994             ++nNodes;
995             aSt.Assign( pStt->nNode.GetNode(), +1 );
996         }
997         else
998             aSt = pStt->nNode;
999         aCntEnd = pEnd->nContent; // aEnd wurde veraendert !!
1000     }
1001     else
1002         aSt.Assign( pStt->nNode.GetNode(), +1 );
1003 
1004     // aSt zeigt jetzt auf den ersten vollen Node
1005 
1006     /*
1007      * die Selektion umfasst mehr als einen Node
1008      */
1009     if( pStt->nNode < pEnd->nNode )
1010     {
1011         pNode = pEnd->nNode.GetNode().GetCntntNode();
1012         if(pNode)
1013         {
1014             sal_uInt16 nLen = pNode->Len();
1015             if( aCntEnd.GetIndex() != nLen )
1016             {
1017                 // the SwRegHistory inserts the attribute into the TxtNode!
1018                 if( pNode->IsTxtNode() && pCharSet && pCharSet->Count() )
1019                 {
1020                     SwRegHistory history( pNode, *pNode, pHistory );
1021                     history.InsertItems(*pCharSet,
1022                             0, aCntEnd.GetIndex(), nFlags);
1023                 }
1024 
1025                 if( pOtherSet && pOtherSet->Count() )
1026                 {
1027                     SwRegHistory aRegH( pNode, *pNode, pHistory );
1028                     pNode->SetAttr( *pOtherSet );
1029                 }
1030 
1031                 ++nNodes;
1032                 aEnd = pEnd->nNode;
1033             }
1034             else
1035                 aEnd.Assign( pEnd->nNode.GetNode(), +1 );
1036         }
1037         else
1038             aEnd = pEnd->nNode;
1039     }
1040     else
1041         aEnd.Assign( pEnd->nNode.GetNode(), +1 );
1042 
1043     // aEnd zeigt jetzt HINTER den letzten voll Node
1044 
1045     /* Bearbeitung der vollstaendig selektierten Nodes. */
1046 // alle Attribute aus dem Set zuruecksetzen !!
1047     if( pCharSet && pCharSet->Count() && !( nsSetAttrMode::SETATTR_DONTREPLACE & nFlags ) )
1048     {
1049 
1050         ParaRstFmt aPara( pStt, pEnd, pHistory, 0, pCharSet );
1051         pDoc->GetNodes().ForEach( aSt, aEnd, lcl_RstTxtAttr, &aPara );
1052     }
1053 
1054     sal_Bool bCreateSwpHints = pCharSet && (
1055         SFX_ITEM_SET == pCharSet->GetItemState( RES_TXTATR_CHARFMT, sal_False ) ||
1056         SFX_ITEM_SET == pCharSet->GetItemState( RES_TXTATR_INETFMT, sal_False ) );
1057 
1058     for(; aSt < aEnd; aSt++ )
1059     {
1060         pNode = aSt.GetNode().GetCntntNode();
1061         if( !pNode )
1062             continue;
1063 
1064         SwTxtNode* pTNd = pNode->GetTxtNode();
1065         if( pHistory )
1066         {
1067             SwRegHistory aRegH( pNode, *pNode, pHistory );
1068             SwpHints *pSwpHints;
1069 
1070             if( pTNd && pCharSet && pCharSet->Count() )
1071             {
1072                 pSwpHints = bCreateSwpHints ? &pTNd->GetOrCreateSwpHints()
1073                                             : pTNd->GetpSwpHints();
1074                 if( pSwpHints )
1075                     pSwpHints->Register( &aRegH );
1076 
1077                 pTNd->SetAttr( *pCharSet, 0, pTNd->GetTxt().Len(), nFlags );
1078                 if( pSwpHints )
1079                     pSwpHints->DeRegister();
1080             }
1081             if( pOtherSet && pOtherSet->Count() )
1082                 pNode->SetAttr( *pOtherSet );
1083         }
1084         else
1085         {
1086             if( pTNd && pCharSet && pCharSet->Count() )
1087                 pTNd->SetAttr( *pCharSet, 0, pTNd->GetTxt().Len(), nFlags );
1088             if( pOtherSet && pOtherSet->Count() )
1089                 pNode->SetAttr( *pOtherSet );
1090         }
1091         ++nNodes;
1092     }
1093 
1094     //The data parameter flag: bExpandCharToPara, comes from the data member of SwDoc,
1095     //Which is set in SW MS word Binary filter WW8ImplRreader. With this flag on, means that
1096     //current setting attribute set is a character range properties set and comes from a MS word
1097     //binary file, And the setting range include a paragraph end position (0X0D);
1098     //More specifications, as such property inside the character range properties set recorded in
1099     //MS word binary file are dealed and inserted into data model (SwDoc) one by one, so we
1100     //only dealing the scenario that the char properties set with 1 item inside;
1101     if (bExpandCharToPara && pCharSet && pCharSet->Count() ==1)
1102     {
1103         SwPosition aStartPos (*rRg.Start());
1104         SwPosition aEndPos (*rRg.End());
1105 
1106         if (aEndPos.nNode.GetNode().GetTxtNode() && aEndPos.nContent != aEndPos.nNode.GetNode().GetTxtNode()->Len())
1107             aEndPos.nNode--;
1108 
1109         for (;aStartPos<=aEndPos;aStartPos.nNode++)
1110         {
1111             SwTxtNode* pCurrentNd = aStartPos.nNode.GetNode().GetTxtNode();
1112 
1113             if (pCurrentNd)
1114             {
1115                 pCurrentNd->TryCharSetExpandToNum(*pCharSet);
1116 
1117             }
1118 
1119         }
1120     }
1121 
1122     DELETECHARSETS
1123     return (nNodes != 0) || bRet;
1124 }
1125 
1126 bool SwDoc::InsertPoolItem(
1127     const SwPaM &rRg,
1128     const SfxPoolItem &rHt,
1129     const SetAttrMode nFlags,
1130     const bool bExpandCharToPara)
1131 {
1132     SwDataChanged aTmp( rRg, 0 );
1133     SwUndoAttr* pUndoAttr = 0;
1134     if (GetIDocumentUndoRedo().DoesUndo())
1135     {
1136         GetIDocumentUndoRedo().ClearRedo();
1137         pUndoAttr = new SwUndoAttr( rRg, rHt, nFlags );
1138     }
1139 
1140     SfxItemSet aSet( GetAttrPool(), rHt.Which(), rHt.Which() );
1141     aSet.Put( rHt );
1142     const bool bRet = lcl_InsAttr( this, rRg, aSet, nFlags, pUndoAttr,bExpandCharToPara );
1143 
1144     if (GetIDocumentUndoRedo().DoesUndo())
1145     {
1146         GetIDocumentUndoRedo().AppendUndo( pUndoAttr );
1147     }
1148 
1149     if( bRet )
1150     {
1151         SetModified();
1152     }
1153     return bRet;
1154 }
1155 
1156 bool SwDoc::InsertItemSet ( const SwPaM &rRg, const SfxItemSet &rSet,
1157                             const SetAttrMode nFlags )
1158 {
1159 	SwDataChanged aTmp( rRg, 0 );
1160 	SwUndoAttr* pUndoAttr = 0;
1161     if (GetIDocumentUndoRedo().DoesUndo())
1162     {
1163         GetIDocumentUndoRedo().ClearRedo();
1164         pUndoAttr = new SwUndoAttr( rRg, rSet, nFlags );
1165 	}
1166 
1167     bool bRet = lcl_InsAttr( this, rRg, rSet, nFlags, pUndoAttr );
1168 
1169     if (GetIDocumentUndoRedo().DoesUndo())
1170     {
1171         GetIDocumentUndoRedo().AppendUndo( pUndoAttr );
1172     }
1173 
1174 	if( bRet )
1175 		SetModified();
1176 	return bRet;
1177 }
1178 
1179 
1180 	// Setze das Attribut im angegebenen Format. Ist Undo aktiv, wird
1181 	// das alte in die Undo-History aufgenommen
1182 void SwDoc::SetAttr( const SfxPoolItem& rAttr, SwFmt& rFmt )
1183 {
1184 	SfxItemSet aSet( GetAttrPool(), rAttr.Which(), rAttr.Which() );
1185 	aSet.Put( rAttr );
1186 	SetAttr( aSet, rFmt );
1187 }
1188 
1189 
1190 	// Setze das Attribut im angegebenen Format. Ist Undo aktiv, wird
1191 	// das alte in die Undo-History aufgenommen
1192 void SwDoc::SetAttr( const SfxItemSet& rSet, SwFmt& rFmt )
1193 {
1194     if (GetIDocumentUndoRedo().DoesUndo())
1195     {
1196         SwUndoFmtAttrHelper aTmp( rFmt );
1197         rFmt.SetFmtAttr( rSet );
1198         if ( aTmp.GetUndo() )
1199         {
1200             GetIDocumentUndoRedo().AppendUndo( aTmp.ReleaseUndo() );
1201         }
1202         else
1203         {
1204             GetIDocumentUndoRedo().ClearRedo();
1205         }
1206     }
1207     else
1208     {
1209         rFmt.SetFmtAttr( rSet );
1210     }
1211 	SetModified();
1212 }
1213 
1214 // --> OD 2008-02-12 #newlistlevelattrs#
1215 void SwDoc::ResetAttrAtFormat( const sal_uInt16 nWhichId,
1216                                SwFmt& rChangedFormat )
1217 {
1218     SwUndo *const pUndo = (GetIDocumentUndoRedo().DoesUndo())
1219         ?   new SwUndoFmtResetAttr( rChangedFormat, nWhichId )
1220         :   0;
1221 
1222     const sal_Bool bAttrReset = rChangedFormat.ResetFmtAttr( nWhichId );
1223 
1224     if ( bAttrReset )
1225     {
1226         if ( pUndo )
1227         {
1228             GetIDocumentUndoRedo().AppendUndo( pUndo );
1229         }
1230 
1231         SetModified();
1232     }
1233     else if ( pUndo )
1234         delete pUndo;
1235 }
1236 // <--
1237 
1238 int lcl_SetNewDefTabStops( SwTwips nOldWidth, SwTwips nNewWidth,
1239 								SvxTabStopItem& rChgTabStop )
1240 {
1241 	// dann aender bei allen TabStop die default's auf den neuen Wert
1242 	// !!! Achtung: hier wird immer auf dem PoolAttribut gearbeitet,
1243 	// 				damit nicht in allen Sets die gleiche Berechnung
1244 	//				auf dem gleichen TabStop (gepoolt!) vorgenommen
1245 	//				wird. Als Modify wird ein FmtChg verschickt.
1246 
1247 	sal_uInt16 nOldCnt = rChgTabStop.Count();
1248 	if( !nOldCnt || nOldWidth == nNewWidth )
1249 		return sal_False;
1250 
1251 	// suche den Anfang der Defaults
1252 	SvxTabStop* pTabs = ((SvxTabStop*)rChgTabStop.GetStart())
1253 						+ (nOldCnt-1);
1254 	sal_uInt16 n;
1255 
1256 	for( n = nOldCnt; n ; --n, --pTabs )
1257 		if( SVX_TAB_ADJUST_DEFAULT != pTabs->GetAdjustment() )
1258 			break;
1259 	++n;
1260 	if( n < nOldCnt )	// die DefTabStops loeschen
1261 		rChgTabStop.Remove( n, nOldCnt - n );
1262 	return sal_True;
1263 }
1264 
1265 // Setze das Attribut als neues default Attribut in diesem Dokument.
1266 // Ist Undo aktiv, wird das alte in die Undo-History aufgenommen
1267 void SwDoc::SetDefault( const SfxPoolItem& rAttr )
1268 {
1269 	SfxItemSet aSet( GetAttrPool(), rAttr.Which(), rAttr.Which() );
1270 	aSet.Put( rAttr );
1271 	SetDefault( aSet );
1272 }
1273 
1274 void SwDoc::SetDefault( const SfxItemSet& rSet )
1275 {
1276 	if( !rSet.Count() )
1277 		return;
1278 
1279 	SwModify aCallMod( 0 );
1280 	SwAttrSet aOld( GetAttrPool(), rSet.GetRanges() ),
1281 			aNew( GetAttrPool(), rSet.GetRanges() );
1282 	SfxItemIter aIter( rSet );
1283 	sal_uInt16 nWhich;
1284 	const SfxPoolItem* pItem = aIter.GetCurItem();
1285 	SfxItemPool* pSdrPool = GetAttrPool().GetSecondaryPool();
1286 	while( sal_True )
1287 	{
1288 		sal_Bool bCheckSdrDflt = sal_False;
1289 		nWhich = pItem->Which();
1290 		aOld.Put( GetAttrPool().GetDefaultItem( nWhich ) );
1291 		GetAttrPool().SetPoolDefaultItem( *pItem );
1292 		aNew.Put( GetAttrPool().GetDefaultItem( nWhich ) );
1293 
1294         if (isCHRATR(nWhich) || isTXTATR(nWhich))
1295         {
1296 			aCallMod.Add( pDfltTxtFmtColl );
1297 			aCallMod.Add( pDfltCharFmt );
1298 			bCheckSdrDflt = 0 != pSdrPool;
1299         }
1300         else if ( isPARATR(nWhich) ||
1301                   // --> OD 2008-02-25 #refactorlists#
1302                   isPARATR_LIST(nWhich) )
1303                   // <--
1304         {
1305 			aCallMod.Add( pDfltTxtFmtColl );
1306 			bCheckSdrDflt = 0 != pSdrPool;
1307         }
1308         else if (isGRFATR(nWhich))
1309         {
1310             aCallMod.Add( pDfltGrfFmtColl );
1311         }
1312         else if (isFRMATR(nWhich))
1313         {
1314 			aCallMod.Add( pDfltGrfFmtColl );
1315 			aCallMod.Add( pDfltTxtFmtColl );
1316 			aCallMod.Add( pDfltFrmFmt );
1317         }
1318         else if (isBOXATR(nWhich))
1319         {
1320             aCallMod.Add( pDfltFrmFmt );
1321         }
1322 
1323 		// copy also the defaults
1324 		if( bCheckSdrDflt )
1325 		{
1326 			sal_uInt16 nEdtWhich, nSlotId;
1327 			if( 0 != (nSlotId = GetAttrPool().GetSlotId( nWhich ) ) &&
1328 				nSlotId != nWhich &&
1329 				0 != (nEdtWhich = pSdrPool->GetWhich( nSlotId )) &&
1330 				nSlotId != nEdtWhich )
1331 			{
1332 				SfxPoolItem* pCpy = pItem->Clone();
1333 				pCpy->SetWhich( nEdtWhich );
1334 				pSdrPool->SetPoolDefaultItem( *pCpy );
1335 				delete pCpy;
1336 			}
1337 		}
1338 
1339 		if( aIter.IsAtEnd() )
1340 			break;
1341 		pItem = aIter.NextItem();
1342 	}
1343 
1344 	if( aNew.Count() && aCallMod.GetDepends() )
1345     {
1346         if (GetIDocumentUndoRedo().DoesUndo())
1347         {
1348             GetIDocumentUndoRedo().AppendUndo( new SwUndoDefaultAttr( aOld ) );
1349         }
1350 
1351 		const SfxPoolItem* pTmpItem;
1352 		if( ( SFX_ITEM_SET ==
1353                 aNew.GetItemState( RES_PARATR_TABSTOP, sal_False, &pTmpItem ) ) &&
1354 			((SvxTabStopItem*)pTmpItem)->Count() )
1355 		{
1356 			// TabStop-Aenderungen behandeln wir erstmal anders:
1357 			// dann aender bei allen TabStop die dafault's auf den neuen Wert
1358 			// !!! Achtung: hier wird immer auf dem PoolAttribut gearbeitet,
1359 			// 				damit nicht in allen Sets die gleiche Berechnung
1360 			//				auf dem gleichen TabStop (gepoolt!) vorgenommen
1361 			//				wird. Als Modify wird ein FmtChg verschickt.
1362 			SwTwips nNewWidth = (*(SvxTabStopItem*)pTmpItem)[ 0 ].GetTabPos(),
1363 					nOldWidth = ((SvxTabStopItem&)aOld.Get(RES_PARATR_TABSTOP))[ 0 ].GetTabPos();
1364 
1365 			int bChg = sal_False;
1366 			sal_uInt32 nMaxItems = GetAttrPool().GetItemCount2( RES_PARATR_TABSTOP );
1367 			for( sal_uInt32 n = 0; n < nMaxItems; ++n )
1368 				if( 0 != (pTmpItem = GetAttrPool().GetItem2( RES_PARATR_TABSTOP, n ) ))
1369 					bChg |= lcl_SetNewDefTabStops( nOldWidth, nNewWidth,
1370 												*(SvxTabStopItem*)pTmpItem );
1371 
1372 			aNew.ClearItem( RES_PARATR_TABSTOP );
1373 			aOld.ClearItem( RES_PARATR_TABSTOP );
1374 			if( bChg )
1375 			{
1376 				SwFmtChg aChgFmt( pDfltCharFmt );
1377 				// dann sage mal den Frames bescheid
1378 				aCallMod.ModifyNotification( &aChgFmt, &aChgFmt );
1379 			}
1380 		}
1381 	}
1382 
1383 	if( aNew.Count() && aCallMod.GetDepends() )
1384 	{
1385 		SwAttrSetChg aChgOld( aOld, aOld );
1386 		SwAttrSetChg aChgNew( aNew, aNew );
1387 		aCallMod.ModifyNotification( &aChgOld, &aChgNew );		// alle veraenderten werden verschickt
1388 	}
1389 
1390 	// und die default-Formate wieder beim Object austragen
1391 	SwClient* pDep;
1392 	while( 0 != ( pDep = (SwClient*)aCallMod.GetDepends()) )
1393 		aCallMod.Remove( pDep );
1394 
1395 	SetModified();
1396 }
1397 
1398 	// Erfrage das Default Attribut in diesem Dokument.
1399 const SfxPoolItem& SwDoc::GetDefault( sal_uInt16 nFmtHint ) const
1400 {
1401 	return GetAttrPool().GetDefaultItem( nFmtHint );
1402 }
1403 
1404 /*
1405  * Loeschen der Formate
1406  */
1407 void SwDoc::DelCharFmt(sal_uInt16 nFmt, sal_Bool bBroadcast)
1408 {
1409     SwCharFmt * pDel = (*pCharFmtTbl)[nFmt];
1410 
1411     if (bBroadcast)
1412         BroadcastStyleOperation(pDel->GetName(), SFX_STYLE_FAMILY_CHAR,
1413                                 SFX_STYLESHEET_ERASED);
1414 
1415     if (GetIDocumentUndoRedo().DoesUndo())
1416     {
1417         SwUndo * pUndo =
1418             new SwUndoCharFmtDelete(pDel, this);
1419 
1420         GetIDocumentUndoRedo().AppendUndo(pUndo);
1421     }
1422 
1423 	pCharFmtTbl->DeleteAndDestroy(nFmt);
1424 
1425 	SetModified();
1426 }
1427 
1428 void SwDoc::DelCharFmt( SwCharFmt *pFmt, sal_Bool bBroadcast )
1429 {
1430 	sal_uInt16 nFmt = pCharFmtTbl->GetPos( pFmt );
1431 	ASSERT( USHRT_MAX != nFmt, "Fmt not found," );
1432 
1433 	DelCharFmt( nFmt, bBroadcast );
1434 }
1435 
1436 void SwDoc::DelFrmFmt( SwFrmFmt *pFmt, sal_Bool bBroadcast )
1437 {
1438 	if( pFmt->ISA( SwTableBoxFmt ) || pFmt->ISA( SwTableLineFmt ))
1439 	{
1440 		ASSERT( !this, "Format steht nicht mehr im DocArray, "
1441 					   "kann per delete geloescht werden" );
1442 		delete pFmt;
1443 	}
1444 	else
1445 	{
1446 
1447 		//Das Format muss in einem der beiden Arrays stehen, in welchem
1448 		//werden wir schon merken.
1449 		sal_uInt16 nPos;
1450 		if ( USHRT_MAX != ( nPos = pFrmFmtTbl->GetPos( pFmt )) )
1451         {
1452             if (bBroadcast)
1453                 BroadcastStyleOperation(pFmt->GetName(),
1454                                         SFX_STYLE_FAMILY_FRAME,
1455                                         SFX_STYLESHEET_ERASED);
1456 
1457             if (GetIDocumentUndoRedo().DoesUndo())
1458             {
1459                 SwUndo * pUndo = new SwUndoFrmFmtDelete(pFmt, this);
1460 
1461                 GetIDocumentUndoRedo().AppendUndo(pUndo);
1462             }
1463 
1464 			pFrmFmtTbl->DeleteAndDestroy( nPos );
1465         }
1466 		else
1467 		{
1468 			nPos = GetSpzFrmFmts()->GetPos( pFmt );
1469 			ASSERT( nPos != USHRT_MAX, "FrmFmt not found." );
1470 			if( USHRT_MAX != nPos )
1471 				GetSpzFrmFmts()->DeleteAndDestroy( nPos );
1472 		}
1473 	}
1474 }
1475 
1476 void SwDoc::DelTblFrmFmt( SwTableFmt *pFmt )
1477 {
1478 	sal_uInt16 nPos = pTblFrmFmtTbl->GetPos( pFmt );
1479 	ASSERT( USHRT_MAX != nPos, "Fmt not found," );
1480 	pTblFrmFmtTbl->DeleteAndDestroy( nPos );
1481 }
1482 
1483 /*
1484  * Erzeugen der Formate
1485  */
1486 SwFlyFrmFmt *SwDoc::MakeFlyFrmFmt( const String &rFmtName,
1487 									SwFrmFmt *pDerivedFrom )
1488 {
1489 	SwFlyFrmFmt *pFmt = new SwFlyFrmFmt( GetAttrPool(), rFmtName, pDerivedFrom );
1490 	GetSpzFrmFmts()->Insert(pFmt, GetSpzFrmFmts()->Count());
1491 	SetModified();
1492 	return pFmt;
1493 }
1494 
1495 SwDrawFrmFmt *SwDoc::MakeDrawFrmFmt( const String &rFmtName,
1496 									 SwFrmFmt *pDerivedFrom )
1497 {
1498 	SwDrawFrmFmt *pFmt = new SwDrawFrmFmt( GetAttrPool(), rFmtName, pDerivedFrom);
1499 	GetSpzFrmFmts()->Insert(pFmt,GetSpzFrmFmts()->Count());
1500 	SetModified();
1501 	return pFmt;
1502 }
1503 
1504 
1505 sal_uInt16 SwDoc::GetTblFrmFmtCount(sal_Bool bUsed) const
1506 {
1507 	sal_uInt16 nCount = pTblFrmFmtTbl->Count();
1508 	if(bUsed)
1509 	{
1510         SwAutoFmtGetDocNode aGetHt( &GetNodes() );
1511 		for ( sal_uInt16 i = nCount; i; )
1512 		{
1513 			if((*pTblFrmFmtTbl)[--i]->GetInfo( aGetHt ))
1514 
1515 				--nCount;
1516 		}
1517 	}
1518 
1519 	return nCount;
1520 }
1521 
1522 
1523 SwFrmFmt& SwDoc::GetTblFrmFmt(sal_uInt16 nFmt, sal_Bool bUsed ) const
1524 {
1525 	sal_uInt16 nRemoved = 0;
1526 	if(bUsed)
1527 	{
1528         SwAutoFmtGetDocNode aGetHt( &GetNodes() );
1529 		for ( sal_uInt16 i = 0; i <= nFmt; i++ )
1530 		{
1531 			while ( (*pTblFrmFmtTbl)[ i + nRemoved]->GetInfo( aGetHt ))
1532 			{
1533 				nRemoved++;
1534 			}
1535 		}
1536 	}
1537 	return *((*pTblFrmFmtTbl)[nRemoved + nFmt]);
1538 }
1539 
1540 SwTableFmt* SwDoc::MakeTblFrmFmt( const String &rFmtName,
1541 									SwFrmFmt *pDerivedFrom )
1542 {
1543 	SwTableFmt* pFmt = new SwTableFmt( GetAttrPool(), rFmtName, pDerivedFrom );
1544 	pTblFrmFmtTbl->Insert( pFmt, pTblFrmFmtTbl->Count() );
1545 	SetModified();
1546 
1547 	return pFmt;
1548 }
1549 
1550 SwFrmFmt *SwDoc::MakeFrmFmt(const String &rFmtName,
1551 							SwFrmFmt *pDerivedFrom,
1552                             sal_Bool bBroadcast, sal_Bool bAuto)
1553 {
1554 
1555 	SwFrmFmt *pFmt = new SwFrmFmt( GetAttrPool(), rFmtName, pDerivedFrom );
1556 
1557     pFmt->SetAuto(bAuto);
1558 	pFrmFmtTbl->Insert( pFmt, pFrmFmtTbl->Count());
1559 	SetModified();
1560 
1561     if (bBroadcast)
1562     {
1563         BroadcastStyleOperation(rFmtName, SFX_STYLE_FAMILY_PARA,
1564                                 SFX_STYLESHEET_CREATED);
1565 
1566         if (GetIDocumentUndoRedo().DoesUndo())
1567         {
1568             SwUndo * pUndo = new SwUndoFrmFmtCreate(pFmt, pDerivedFrom, this);
1569 
1570             GetIDocumentUndoRedo().AppendUndo(pUndo);
1571         }
1572     }
1573 
1574 	return pFmt;
1575 }
1576 
1577 SwFmt *SwDoc::_MakeFrmFmt(const String &rFmtName,
1578 							SwFmt *pDerivedFrom,
1579                             sal_Bool bBroadcast, sal_Bool bAuto)
1580 {
1581     SwFrmFmt *pFrmFmt = dynamic_cast<SwFrmFmt*>(pDerivedFrom);
1582     pFrmFmt = MakeFrmFmt( rFmtName, pFrmFmt, bBroadcast, bAuto );
1583     return dynamic_cast<SwFmt*>(pFrmFmt);
1584 }
1585 
1586 
1587 // --> OD 2005-01-13 #i40550# - add parameter <bAuto> - not relevant
1588 SwCharFmt *SwDoc::MakeCharFmt( const String &rFmtName,
1589                                SwCharFmt *pDerivedFrom,
1590                                sal_Bool bBroadcast,
1591                                sal_Bool )
1592 // <--
1593 {
1594 	SwCharFmt *pFmt = new SwCharFmt( GetAttrPool(), rFmtName, pDerivedFrom );
1595     pCharFmtTbl->Insert( pFmt, pCharFmtTbl->Count() );
1596     pFmt->SetAuto( sal_False );
1597     SetModified();
1598 
1599     if (GetIDocumentUndoRedo().DoesUndo())
1600     {
1601         SwUndo * pUndo = new SwUndoCharFmtCreate(pFmt, pDerivedFrom, this);
1602 
1603         GetIDocumentUndoRedo().AppendUndo(pUndo);
1604     }
1605 
1606     if (bBroadcast)
1607     {
1608         BroadcastStyleOperation(rFmtName, SFX_STYLE_FAMILY_CHAR,
1609                                 SFX_STYLESHEET_CREATED);
1610     }
1611 
1612 	return pFmt;
1613 }
1614 
1615 SwFmt *SwDoc::_MakeCharFmt(const String &rFmtName,
1616 							SwFmt *pDerivedFrom,
1617                             sal_Bool bBroadcast, sal_Bool bAuto)
1618 {
1619     SwCharFmt *pCharFmt = dynamic_cast<SwCharFmt*>(pDerivedFrom);
1620     pCharFmt = MakeCharFmt( rFmtName, pCharFmt, bBroadcast, bAuto );
1621     return dynamic_cast<SwFmt*>(pCharFmt);
1622 }
1623 
1624 
1625 /*
1626  * Erzeugen der FormatCollections
1627  */
1628 // TXT
1629 // --> OD 2005-01-13 #i40550# - add parameter <bAuto> - not relevant
1630 SwTxtFmtColl* SwDoc::MakeTxtFmtColl( const String &rFmtName,
1631 									 SwTxtFmtColl *pDerivedFrom,
1632                                      sal_Bool bBroadcast,
1633                                      sal_Bool )
1634 // <--
1635 {
1636 	SwTxtFmtColl *pFmtColl = new SwTxtFmtColl( GetAttrPool(), rFmtName,
1637 												pDerivedFrom );
1638 	pTxtFmtCollTbl->Insert(pFmtColl, pTxtFmtCollTbl->Count());
1639 	pFmtColl->SetAuto( sal_False );
1640 	SetModified();
1641 
1642     if (GetIDocumentUndoRedo().DoesUndo())
1643     {
1644         SwUndo * pUndo = new SwUndoTxtFmtCollCreate(pFmtColl, pDerivedFrom,
1645                                                     this);
1646         GetIDocumentUndoRedo().AppendUndo(pUndo);
1647     }
1648 
1649     if (bBroadcast)
1650         BroadcastStyleOperation(rFmtName, SFX_STYLE_FAMILY_PARA,
1651                                 SFX_STYLESHEET_CREATED);
1652 
1653 	return pFmtColl;
1654 }
1655 
1656 SwFmt *SwDoc::_MakeTxtFmtColl(const String &rFmtName,
1657 							SwFmt *pDerivedFrom,
1658                             sal_Bool bBroadcast, sal_Bool bAuto)
1659 {
1660     SwTxtFmtColl *pTxtFmtColl = dynamic_cast<SwTxtFmtColl*>(pDerivedFrom);
1661     pTxtFmtColl = MakeTxtFmtColl( rFmtName, pTxtFmtColl, bBroadcast, bAuto );
1662     return dynamic_cast<SwFmt*>(pTxtFmtColl);
1663 }
1664 
1665 
1666 //FEATURE::CONDCOLL
1667 SwConditionTxtFmtColl* SwDoc::MakeCondTxtFmtColl( const String &rFmtName,
1668                                                   SwTxtFmtColl *pDerivedFrom,
1669                                                   sal_Bool bBroadcast)
1670 {
1671 	SwConditionTxtFmtColl*pFmtColl = new SwConditionTxtFmtColl( GetAttrPool(),
1672 													rFmtName, pDerivedFrom );
1673 	pTxtFmtCollTbl->Insert(pFmtColl, pTxtFmtCollTbl->Count());
1674 	pFmtColl->SetAuto( sal_False );
1675 	SetModified();
1676 
1677     if (bBroadcast)
1678         BroadcastStyleOperation(rFmtName, SFX_STYLE_FAMILY_PARA,
1679                                 SFX_STYLESHEET_CREATED);
1680 
1681 	return pFmtColl;
1682 }
1683 //FEATURE::CONDCOLL
1684 
1685 // GRF
1686 
1687 SwGrfFmtColl* SwDoc::MakeGrfFmtColl( const String &rFmtName,
1688 									 SwGrfFmtColl *pDerivedFrom )
1689 {
1690 	SwGrfFmtColl *pFmtColl = new SwGrfFmtColl( GetAttrPool(), rFmtName,
1691 												pDerivedFrom );
1692 	pGrfFmtCollTbl->Insert( pFmtColl, pGrfFmtCollTbl->Count() );
1693 	pFmtColl->SetAuto( sal_False );
1694 	SetModified();
1695 	return pFmtColl;
1696 }
1697 
1698 void SwDoc::DelTxtFmtColl(sal_uInt16 nFmtColl, sal_Bool bBroadcast)
1699 {
1700 	ASSERT( nFmtColl, "Remove fuer Coll 0." );
1701 
1702 	// Wer hat die zu loeschende als Next
1703 	SwTxtFmtColl *pDel = (*pTxtFmtCollTbl)[nFmtColl];
1704 	if( pDfltTxtFmtColl == pDel )
1705 		return;		// default nie loeschen !!
1706 
1707     if (bBroadcast)
1708         BroadcastStyleOperation(pDel->GetName(), SFX_STYLE_FAMILY_PARA,
1709                                 SFX_STYLESHEET_ERASED);
1710 
1711     if (GetIDocumentUndoRedo().DoesUndo())
1712     {
1713         SwUndoTxtFmtCollDelete * pUndo =
1714             new SwUndoTxtFmtCollDelete(pDel, this);
1715 
1716         GetIDocumentUndoRedo().AppendUndo(pUndo);
1717     }
1718 
1719 	// Die FmtColl austragen
1720 	pTxtFmtCollTbl->Remove(nFmtColl);
1721 	// Next korrigieren
1722 	pTxtFmtCollTbl->ForEach( 1, pTxtFmtCollTbl->Count(),
1723 							&SetTxtFmtCollNext, pDel );
1724 	delete pDel;
1725 	SetModified();
1726 }
1727 
1728 void SwDoc::DelTxtFmtColl( SwTxtFmtColl *pColl, sal_Bool bBroadcast )
1729 {
1730 	sal_uInt16 nFmt = pTxtFmtCollTbl->GetPos( pColl );
1731 	ASSERT( USHRT_MAX != nFmt, "Collection not found," );
1732 	DelTxtFmtColl( nFmt, bBroadcast );
1733 }
1734 
1735 sal_Bool lcl_SetTxtFmtColl( const SwNodePtr& rpNode, void* pArgs )
1736 {
1737 	// ParaSetFmtColl * pPara = (ParaSetFmtColl*)pArgs;
1738 	SwCntntNode* pCNd = (SwCntntNode*)rpNode->GetTxtNode();
1739 	if( pCNd )
1740 	{
1741 		ParaRstFmt* pPara = (ParaRstFmt*)pArgs;
1742 
1743         SwTxtFmtColl* pFmt = static_cast<SwTxtFmtColl*>(pPara->pFmtColl);
1744 		if ( pPara->bReset )
1745         {
1746 
1747 			if( pFmt->GetAttrOutlineLevel() == 0 && pPara )
1748 				pPara->bKeepOutlineLevelAttr = true;
1749 
1750 			lcl_RstAttr( pCNd, pPara );
1751 
1752             // --> OD 2007-11-06 #i62675#
1753             // --> OD 2008-04-15 #refactorlists#
1754             // check, if paragraph style has changed
1755             if ( pPara->bResetListAttrs &&
1756                  pFmt != pCNd->GetFmtColl() &&
1757                  pFmt->GetItemState( RES_PARATR_NUMRULE ) == SFX_ITEM_SET )
1758             {
1759                 // --> OD 2009-09-07 #b6876367#
1760                 // Check, if the list style of the paragraph will change.
1761                 bool bChangeOfListStyleAtParagraph( true );
1762                 SwTxtNode* pTNd( dynamic_cast<SwTxtNode*>(pCNd) );
1763                 ASSERT( pTNd,
1764                         "<lcl_SetTxtFmtColl(..)> - text node expected -> crash" );
1765                 {
1766                     SwNumRule* pNumRuleAtParagraph( pTNd->GetNumRule() );
1767                     if ( pNumRuleAtParagraph )
1768                     {
1769                         const SwNumRuleItem& rNumRuleItemAtParagraphStyle =
1770                                                             pFmt->GetNumRule();
1771                         if ( rNumRuleItemAtParagraphStyle.GetValue() ==
1772                                                 pNumRuleAtParagraph->GetName() )
1773                         {
1774                             bChangeOfListStyleAtParagraph = false;
1775                         }
1776                     }
1777                 }
1778 
1779                 if ( bChangeOfListStyleAtParagraph )
1780                 {
1781                     // --> OD 2008-04-08 #refactorlists#
1782                     std::auto_ptr< SwRegHistory > pRegH;
1783                     if ( pPara->pHistory )
1784                     {
1785                         pRegH.reset( new SwRegHistory( pTNd, *pTNd, pPara->pHistory ) );
1786                     }
1787 
1788                     pCNd->ResetAttr( RES_PARATR_NUMRULE );
1789 
1790                     // reset all list attributes
1791                     pCNd->ResetAttr( RES_PARATR_LIST_LEVEL );
1792                     pCNd->ResetAttr( RES_PARATR_LIST_ISRESTART );
1793                     pCNd->ResetAttr( RES_PARATR_LIST_RESTARTVALUE );
1794                     pCNd->ResetAttr( RES_PARATR_LIST_ISCOUNTED );
1795                     pCNd->ResetAttr( RES_PARATR_LIST_ID );
1796                 }
1797                 // <--
1798             }
1799             // <--
1800         }
1801 
1802         // erst in die History aufnehmen, damit ggfs. alte Daten
1803 		// gesichert werden koennen
1804 		if( pPara->pHistory )
1805 			pPara->pHistory->Add( pCNd->GetFmtColl(), pCNd->GetIndex(),
1806 									ND_TEXTNODE );
1807 
1808         pCNd->ChgFmtColl( pFmt );
1809 
1810 		pPara->nWhich++;
1811 	}
1812 	return sal_True;
1813 }
1814 
1815 sal_Bool SwDoc::SetTxtFmtColl( const SwPaM &rRg,
1816                            SwTxtFmtColl *pFmt,
1817                            bool bReset,
1818                            bool bResetListAttrs )
1819 {
1820 	SwDataChanged aTmp( rRg, 0 );
1821 	const SwPosition *pStt = rRg.Start(), *pEnd = rRg.End();
1822 	SwHistory* pHst = 0;
1823 	sal_Bool bRet = sal_True;
1824 
1825     if (GetIDocumentUndoRedo().DoesUndo())
1826     {
1827         // --> OD 2008-04-15 #refactorlists#
1828         SwUndoFmtColl* pUndo = new SwUndoFmtColl( rRg, pFmt,
1829                                                   bReset,
1830                                                   bResetListAttrs );
1831         // <--
1832 		pHst = pUndo->GetHistory();
1833         GetIDocumentUndoRedo().AppendUndo(pUndo);
1834     }
1835 
1836     ParaRstFmt aPara( pStt, pEnd, pHst );
1837     aPara.pFmtColl = pFmt;
1838     aPara.bReset = bReset;
1839     // --> OD 2007-11-06 #i62675#
1840     aPara.bResetListAttrs = bResetListAttrs;
1841     // <--
1842 
1843     GetNodes().ForEach( pStt->nNode.GetIndex(), pEnd->nNode.GetIndex()+1,
1844                         lcl_SetTxtFmtColl, &aPara );
1845     if( !aPara.nWhich )
1846         bRet = sal_False;           // keinen gueltigen Node gefunden
1847 
1848 	if( bRet )
1849 		SetModified();
1850 	return bRet;
1851 }
1852 
1853 
1854 // ---- Kopiere die Formate in sich selbst (SwDoc) ----------------------
1855 
1856 SwFmt* SwDoc::CopyFmt( const SwFmt& rFmt,
1857 						const SvPtrarr& rFmtArr,
1858 						FNCopyFmt fnCopyFmt, const SwFmt& rDfltFmt )
1859 {
1860 	//  kein-Autoformat || default Format || Collection-Format
1861 	// dann suche danach.
1862 	if( !rFmt.IsAuto() || !rFmt.GetRegisteredIn() )
1863 		for( sal_uInt16 n = 0; n < rFmtArr.Count(); n++ )
1864 		{
1865 			// ist die Vorlage schon im Doc vorhanden ??
1866 			if( ((SwFmt*)rFmtArr[n])->GetName().Equals( rFmt.GetName() ))
1867 				return (SwFmt*)rFmtArr[n];
1868 		}
1869 
1870 	// suche erstmal nach dem "Parent"
1871 	SwFmt* pParent = (SwFmt*)&rDfltFmt;
1872 	if( rFmt.DerivedFrom() && pParent != rFmt.DerivedFrom() )
1873 		pParent = CopyFmt( *rFmt.DerivedFrom(), rFmtArr,
1874 								fnCopyFmt, rDfltFmt );
1875 
1876 	// erzeuge das Format und kopiere die Attribute
1877     // --> OD 2005-01-13 #i40550#
1878     SwFmt* pNewFmt = (this->*fnCopyFmt)( rFmt.GetName(), pParent, sal_False, sal_True );
1879     // <--
1880     pNewFmt->SetAuto( rFmt.IsAuto() );
1881 	pNewFmt->CopyAttrs( rFmt, sal_True );			// kopiere Attribute
1882 
1883 	pNewFmt->SetPoolFmtId( rFmt.GetPoolFmtId() );
1884 	pNewFmt->SetPoolHelpId( rFmt.GetPoolHelpId() );
1885 
1886 	// HelpFile-Id immer auf dflt setzen !!
1887 	pNewFmt->SetPoolHlpFileId( UCHAR_MAX );
1888 
1889 	return pNewFmt;
1890 }
1891 
1892 
1893 // ---- kopiere das Frame-Format --------
1894 SwFrmFmt* SwDoc::CopyFrmFmt( const SwFrmFmt& rFmt )
1895 {
1896 
1897     return (SwFrmFmt*)CopyFmt( rFmt, *GetFrmFmts(), &SwDoc::_MakeFrmFmt,
1898                                 *GetDfltFrmFmt() );
1899 }
1900 
1901 // ---- kopiere das Char-Format --------
1902 SwCharFmt* SwDoc::CopyCharFmt( const SwCharFmt& rFmt )
1903 {
1904 	return (SwCharFmt*)CopyFmt( rFmt, *GetCharFmts(),
1905 								&SwDoc::_MakeCharFmt,
1906 								*GetDfltCharFmt() );
1907 }
1908 
1909 
1910 // --- Kopiere TextNodes ----
1911 
1912 SwTxtFmtColl* SwDoc::CopyTxtColl( const SwTxtFmtColl& rColl )
1913 {
1914 	SwTxtFmtColl* pNewColl = FindTxtFmtCollByName( rColl.GetName() );
1915 	if( pNewColl )
1916 		return pNewColl;
1917 
1918 	// suche erstmal nach dem "Parent"
1919 	SwTxtFmtColl* pParent = pDfltTxtFmtColl;
1920 	if( pParent != rColl.DerivedFrom() )
1921 		pParent = CopyTxtColl( *(SwTxtFmtColl*)rColl.DerivedFrom() );
1922 
1923 
1924 //FEATURE::CONDCOLL
1925 	if( RES_CONDTXTFMTCOLL == rColl.Which() )
1926 	{
1927 		pNewColl = new SwConditionTxtFmtColl( GetAttrPool(), rColl.GetName(),
1928 												pParent);
1929 		pTxtFmtCollTbl->Insert( pNewColl, pTxtFmtCollTbl->Count() );
1930 		pNewColl->SetAuto( sal_False );
1931 		SetModified();
1932 
1933 		// Kopiere noch die Bedingungen
1934 		((SwConditionTxtFmtColl*)pNewColl)->SetConditions(
1935 							((SwConditionTxtFmtColl&)rColl).GetCondColls() );
1936 	}
1937 	else
1938 //FEATURE::CONDCOLL
1939 		pNewColl = MakeTxtFmtColl( rColl.GetName(), pParent );
1940 
1941 	// kopiere jetzt noch die Auto-Formate oder kopiere die Attribute
1942 	pNewColl->CopyAttrs( rColl, sal_True );
1943 
1944 	// setze noch den Outline-Level
1945     //if( NO_NUMBERING != rColl.GetOutlineLevel() )	//#outline level,zhaojianwei
1946 	//	pNewColl->SetOutlineLevel( rColl.GetOutlineLevel() );
1947     if(rColl.IsAssignedToListLevelOfOutlineStyle())
1948 		pNewColl->AssignToListLevelOfOutlineStyle(rColl.GetAssignedOutlineStyleLevel());//<-end,zhaojianwei
1949 	//<-end
1950 	pNewColl->SetPoolFmtId( rColl.GetPoolFmtId() );
1951 	pNewColl->SetPoolHelpId( rColl.GetPoolHelpId() );
1952 
1953 	// HelpFile-Id immer auf dflt setzen !!
1954 	pNewColl->SetPoolHlpFileId( UCHAR_MAX );
1955 
1956 	if( &rColl.GetNextTxtFmtColl() != &rColl )
1957 		pNewColl->SetNextTxtFmtColl( *CopyTxtColl( rColl.GetNextTxtFmtColl() ));
1958 
1959 	// ggfs. die NumRule erzeugen
1960 	if( this != rColl.GetDoc() )
1961 	{
1962 		const SfxPoolItem* pItem;
1963 		if( SFX_ITEM_SET == pNewColl->GetItemState( RES_PARATR_NUMRULE,
1964 			sal_False, &pItem ))
1965 		{
1966 			const SwNumRule* pRule;
1967 			const String& rName = ((SwNumRuleItem*)pItem)->GetValue();
1968 			if( rName.Len() &&
1969 				0 != ( pRule = rColl.GetDoc()->FindNumRulePtr( rName )) &&
1970 				!pRule->IsAutoRule() )
1971 			{
1972 				SwNumRule* pDestRule = FindNumRulePtr( rName );
1973 				if( pDestRule )
1974 					pDestRule->SetInvalidRule( sal_True );
1975 				else
1976 					MakeNumRule( rName, pRule );
1977 			}
1978 		}
1979 	}
1980 	return pNewColl;
1981 }
1982 
1983 // --- Kopiere GrafikNodes ----
1984 
1985 SwGrfFmtColl* SwDoc::CopyGrfColl( const SwGrfFmtColl& rColl )
1986 {
1987 	SwGrfFmtColl* pNewColl = FindGrfFmtCollByName( rColl.GetName() );
1988 	if( pNewColl )
1989 		return pNewColl;
1990 
1991 	// suche erstmal nach dem "Parent"
1992 	SwGrfFmtColl* pParent = pDfltGrfFmtColl;
1993 	if( pParent != rColl.DerivedFrom() )
1994 		pParent = CopyGrfColl( *(SwGrfFmtColl*)rColl.DerivedFrom() );
1995 
1996 	// falls nicht, so kopiere sie
1997 	pNewColl = MakeGrfFmtColl( rColl.GetName(), pParent );
1998 
1999 	// noch die Attribute kopieren
2000 	pNewColl->CopyAttrs( rColl );
2001 
2002 	pNewColl->SetPoolFmtId( rColl.GetPoolFmtId() );
2003 	pNewColl->SetPoolHelpId( rColl.GetPoolHelpId() );
2004 
2005 	// HelpFile-Id immer auf dflt setzen !!
2006 	pNewColl->SetPoolHlpFileId( UCHAR_MAX );
2007 
2008 	return pNewColl;
2009 }
2010 
2011 SwPageDesc* lcl_FindPageDesc( const SwPageDescs& rArr, const String& rName )
2012 {
2013 	for( sal_uInt16 n = rArr.Count(); n; )
2014 	{
2015 		SwPageDesc* pDesc = rArr[ --n ];
2016 		if( pDesc->GetName() == rName )
2017 			return pDesc;
2018 	}
2019 	return 0;
2020 }
2021 
2022 void SwDoc::CopyFmtArr( const SvPtrarr& rSourceArr,
2023 						SvPtrarr& rDestArr,
2024 						FNCopyFmt fnCopyFmt,
2025 						SwFmt& rDfltFmt )
2026 {
2027 	sal_uInt16 nSrc;
2028 	SwFmt* pSrc, *pDest;
2029 
2030 	// 1. Schritt alle Formate anlegen (das 0. ueberspringen - Default!)
2031 	for( nSrc = rSourceArr.Count(); nSrc > 1; )
2032 	{
2033 		pSrc = (SwFmt*)rSourceArr[ --nSrc ];
2034 		if( pSrc->IsDefault() || pSrc->IsAuto() )
2035 			continue;
2036 
2037 		if( 0 == FindFmtByName( rDestArr, pSrc->GetName() ) )
2038 		{
2039 			if( RES_CONDTXTFMTCOLL == pSrc->Which() )
2040                 MakeCondTxtFmtColl( pSrc->GetName(), (SwTxtFmtColl*)&rDfltFmt );
2041 			else
2042                 // --> OD 2005-01-13 #i40550#
2043                 (this->*fnCopyFmt)( pSrc->GetName(), &rDfltFmt, sal_False, sal_True );
2044                 // <--
2045 		}
2046 	}
2047 
2048 	// 2. Schritt alle Attribute kopieren, richtige Parents setzen
2049 	for( nSrc = rSourceArr.Count(); nSrc > 1; )
2050 	{
2051 		pSrc = (SwFmt*)rSourceArr[ --nSrc ];
2052 		if( pSrc->IsDefault() || pSrc->IsAuto() )
2053 			continue;
2054 
2055 		pDest = FindFmtByName( rDestArr, pSrc->GetName() );
2056 		pDest->SetAuto( sal_False );
2057 		pDest->DelDiffs( *pSrc );
2058 
2059         // #i94285#: existing <SwFmtPageDesc> instance, before copying attributes
2060         const SfxPoolItem* pItem;
2061         if( &GetAttrPool() != pSrc->GetAttrSet().GetPool() &&
2062             SFX_ITEM_SET == pSrc->GetAttrSet().GetItemState(
2063             RES_PAGEDESC, sal_False, &pItem ) &&
2064             ((SwFmtPageDesc*)pItem)->GetPageDesc() )
2065         {
2066             SwFmtPageDesc aPageDesc( *(SwFmtPageDesc*)pItem );
2067             const String& rNm = aPageDesc.GetPageDesc()->GetName();
2068             SwPageDesc* pPageDesc = ::lcl_FindPageDesc( aPageDescs, rNm );
2069             if( !pPageDesc )
2070             {
2071                 pPageDesc = aPageDescs[ MakePageDesc( rNm ) ];
2072             }
2073             aPageDesc.RegisterToPageDesc( *pPageDesc );
2074             SwAttrSet aTmpAttrSet( pSrc->GetAttrSet() );
2075             aTmpAttrSet.Put( aPageDesc );
2076             pDest->SetFmtAttr( aTmpAttrSet );
2077         }
2078         else
2079         {
2080             pDest->SetFmtAttr( pSrc->GetAttrSet() );
2081         }
2082 
2083 		pDest->SetPoolFmtId( pSrc->GetPoolFmtId() );
2084 		pDest->SetPoolHelpId( pSrc->GetPoolHelpId() );
2085 
2086 		// HelpFile-Id immer auf dflt setzen !!
2087 		pDest->SetPoolHlpFileId( UCHAR_MAX );
2088 
2089 		if( pSrc->DerivedFrom() )
2090 			pDest->SetDerivedFrom( FindFmtByName( rDestArr,
2091 										pSrc->DerivedFrom()->GetName() ) );
2092 		if( RES_TXTFMTCOLL == pSrc->Which() ||
2093 			RES_CONDTXTFMTCOLL == pSrc->Which() )
2094 		{
2095 			SwTxtFmtColl* pSrcColl = (SwTxtFmtColl*)pSrc,
2096 						* pDstColl = (SwTxtFmtColl*)pDest;
2097 			if( &pSrcColl->GetNextTxtFmtColl() != pSrcColl )
2098 				pDstColl->SetNextTxtFmtColl( *(SwTxtFmtColl*)FindFmtByName(
2099 					rDestArr, pSrcColl->GetNextTxtFmtColl().GetName() ) );
2100 
2101 			// setze noch den Outline-Level
2102 			//if( NO_NUMBERING != pSrcColl->GetOutlineLevel() )	//#outline level,zhaojianwei
2103 			//	pDstColl->SetOutlineLevel( pSrcColl->GetOutlineLevel() );
2104 			if(pSrcColl->IsAssignedToListLevelOfOutlineStyle())
2105 				pDstColl->AssignToListLevelOfOutlineStyle(pSrcColl->GetAssignedOutlineStyleLevel());//<-end,zhaojianwei
2106 			//<-end
2107 
2108 //FEATURE::CONDCOLL
2109 			if( RES_CONDTXTFMTCOLL == pSrc->Which() )
2110 				// Kopiere noch die Bedingungen
2111 				// aber erst die alten loeschen!
2112 				((SwConditionTxtFmtColl*)pDstColl)->SetConditions(
2113 							((SwConditionTxtFmtColl*)pSrc)->GetCondColls() );
2114 //FEATURE::CONDCOLL
2115 		}
2116 	}
2117 }
2118 
2119 void SwDoc::CopyPageDescHeaderFooterImpl( bool bCpyHeader,
2120 								const SwFrmFmt& rSrcFmt, SwFrmFmt& rDestFmt )
2121 {
2122 	// jetzt noch Header-/Footer-Attribute richtig behandeln
2123 	// Contenten Nodes Dokumentuebergreifend kopieren!
2124 	sal_uInt16 nAttr = static_cast<sal_uInt16>( bCpyHeader ? RES_HEADER : RES_FOOTER );
2125 	const SfxPoolItem* pItem;
2126 	if( SFX_ITEM_SET != rSrcFmt.GetAttrSet().GetItemState( nAttr, sal_False, &pItem ))
2127 		return ;
2128 
2129 	// Im Header steht noch der Verweis auf das Format aus dem
2130 	// anderen Document!!
2131 	SfxPoolItem* pNewItem = pItem->Clone();
2132 
2133 	SwFrmFmt* pOldFmt;
2134 	if( bCpyHeader )
2135 		 pOldFmt = ((SwFmtHeader*)pNewItem)->GetHeaderFmt();
2136 	else
2137 		 pOldFmt = ((SwFmtFooter*)pNewItem)->GetFooterFmt();
2138 
2139 	if( pOldFmt )
2140 	{
2141 		SwFrmFmt* pNewFmt = new SwFrmFmt( GetAttrPool(), "CpyDesc",
2142 											GetDfltFrmFmt() );
2143 		pNewFmt->CopyAttrs( *pOldFmt, sal_True );
2144 
2145 		if( SFX_ITEM_SET == pNewFmt->GetAttrSet().GetItemState(
2146 			RES_CNTNT, sal_False, &pItem ))
2147 		{
2148 			SwFmtCntnt* pCntnt = (SwFmtCntnt*)pItem;
2149 			if( pCntnt->GetCntntIdx() )
2150 			{
2151 				SwNodeIndex aTmpIdx( GetNodes().GetEndOfAutotext() );
2152 				const SwNodes& rSrcNds = rSrcFmt.GetDoc()->GetNodes();
2153 				SwStartNode* pSttNd = GetNodes().MakeEmptySection( aTmpIdx,
2154 												bCpyHeader
2155 													? SwHeaderStartNode
2156 													: SwFooterStartNode );
2157 				const SwNode& rCSttNd = pCntnt->GetCntntIdx()->GetNode();
2158 				SwNodeRange aRg( rCSttNd, 0, *rCSttNd.EndOfSectionNode() );
2159 				aTmpIdx = *pSttNd->EndOfSectionNode();
2160 				rSrcNds._Copy( aRg, aTmpIdx );
2161 				aTmpIdx = *pSttNd;
2162                 rSrcFmt.GetDoc()->CopyFlyInFlyImpl( aRg, 0, aTmpIdx );
2163                 pNewFmt->SetFmtAttr( SwFmtCntnt( pSttNd ));
2164 			}
2165 			else
2166                 pNewFmt->ResetFmtAttr( RES_CNTNT );
2167 		}
2168 		if( bCpyHeader )
2169             ((SwFmtHeader*)pNewItem)->RegisterToFormat(*pNewFmt);
2170 		else
2171             ((SwFmtFooter*)pNewItem)->RegisterToFormat(*pNewFmt);
2172         rDestFmt.SetFmtAttr( *pNewItem );
2173 	}
2174 	delete pNewItem;
2175 }
2176 
2177 void SwDoc::CopyPageDesc( const SwPageDesc& rSrcDesc, SwPageDesc& rDstDesc,
2178 							sal_Bool bCopyPoolIds )
2179 {
2180 	sal_Bool bNotifyLayout = sal_False;
2181 	SwRootFrm* pTmpRoot = GetCurrentLayout();//swmod 080219
2182 
2183 	rDstDesc.SetLandscape( rSrcDesc.GetLandscape() );
2184 	rDstDesc.SetNumType( rSrcDesc.GetNumType() );
2185 	if( rDstDesc.ReadUseOn() != rSrcDesc.ReadUseOn() )
2186 	{
2187 		rDstDesc.WriteUseOn( rSrcDesc.ReadUseOn() );
2188 		bNotifyLayout = sal_True;
2189 	}
2190 
2191 	if( bCopyPoolIds )
2192 	{
2193 		rDstDesc.SetPoolFmtId( rSrcDesc.GetPoolFmtId() );
2194 		rDstDesc.SetPoolHelpId( rSrcDesc.GetPoolHelpId() );
2195 		// HelpFile-Id immer auf dflt setzen !!
2196 		rDstDesc.SetPoolHlpFileId( UCHAR_MAX );
2197 	}
2198 
2199 	if( rSrcDesc.GetFollow() != &rSrcDesc )
2200 	{
2201 		SwPageDesc* pFollow = ::lcl_FindPageDesc( aPageDescs,
2202 									rSrcDesc.GetFollow()->GetName() );
2203 		if( !pFollow )
2204 		{
2205 			// dann mal kopieren
2206 			sal_uInt16 nPos = MakePageDesc( rSrcDesc.GetFollow()->GetName() );
2207 			pFollow = aPageDescs[ nPos ];
2208 			CopyPageDesc( *rSrcDesc.GetFollow(), *pFollow );
2209 		}
2210 		rDstDesc.SetFollow( pFollow );
2211 		bNotifyLayout = sal_True;
2212 	}
2213 
2214 	// die Header/Footer-Attribute werden gesondert kopiert, die Content-
2215 	// Sections muessen vollstaendig mitgenommen werden!
2216 	{
2217 		SfxItemSet aAttrSet( rSrcDesc.GetMaster().GetAttrSet() );
2218 		aAttrSet.ClearItem( RES_HEADER );
2219 		aAttrSet.ClearItem( RES_FOOTER );
2220 
2221 		rDstDesc.GetMaster().DelDiffs( aAttrSet );
2222         rDstDesc.GetMaster().SetFmtAttr( aAttrSet );
2223 
2224 		aAttrSet.ClearItem();
2225 		aAttrSet.Put( rSrcDesc.GetLeft().GetAttrSet() );
2226 		aAttrSet.ClearItem( RES_HEADER );
2227 		aAttrSet.ClearItem( RES_FOOTER );
2228 
2229 		rDstDesc.GetLeft().DelDiffs( aAttrSet );
2230         rDstDesc.GetLeft().SetFmtAttr( aAttrSet );
2231 	}
2232 
2233 	CopyHeader( rSrcDesc.GetMaster(), rDstDesc.GetMaster() );
2234 	CopyFooter( rSrcDesc.GetMaster(), rDstDesc.GetMaster() );
2235 	if( !rDstDesc.IsHeaderShared() )
2236 		CopyHeader( rSrcDesc.GetLeft(), rDstDesc.GetLeft() );
2237 	else
2238         rDstDesc.GetLeft().SetFmtAttr( rDstDesc.GetMaster().GetHeader() );
2239 
2240 	if( !rDstDesc.IsFooterShared() )
2241 		CopyFooter( rSrcDesc.GetLeft(), rDstDesc.GetLeft() );
2242 	else
2243         rDstDesc.GetLeft().SetFmtAttr( rDstDesc.GetMaster().GetFooter() );
2244 
2245 	if( bNotifyLayout && pTmpRoot )
2246 	{
2247 		std::set<SwRootFrm*> aAllLayouts = GetAllLayouts();//swmod 080225
2248 		std::for_each( aAllLayouts.begin(), aAllLayouts.end(),std::mem_fun(&SwRootFrm::AllCheckPageDescs));//swmod 080226
2249 	}
2250 
2251 	//Wenn sich FussnotenInfo veraendert, so werden die Seiten
2252 	//angetriggert.
2253 	if( !(rDstDesc.GetFtnInfo() == rSrcDesc.GetFtnInfo()) )
2254 	{
2255 		rDstDesc.SetFtnInfo( rSrcDesc.GetFtnInfo() );
2256 		SwMsgPoolItem  aInfo( RES_PAGEDESC_FTNINFO );
2257 		{
2258             rDstDesc.GetMaster().ModifyBroadcast( &aInfo, 0, TYPE(SwFrm) );
2259 		}
2260 		{
2261             rDstDesc.GetLeft().ModifyBroadcast( &aInfo, 0, TYPE(SwFrm) );
2262 		}
2263 	}
2264 }
2265 
2266 void SwDoc::ReplaceStyles( SwDoc& rSource )
2267 {
2268     ::sw::UndoGuard const undoGuard(GetIDocumentUndoRedo());
2269 
2270 	CopyFmtArr( *rSource.pCharFmtTbl, *pCharFmtTbl,
2271 				&SwDoc::_MakeCharFmt, *pDfltCharFmt );
2272 	CopyFmtArr( *rSource.pFrmFmtTbl, *pFrmFmtTbl,
2273 				&SwDoc::_MakeFrmFmt, *pDfltFrmFmt );
2274 	CopyFmtArr( *rSource.pTxtFmtCollTbl, *pTxtFmtCollTbl,
2275 				&SwDoc::_MakeTxtFmtColl, *pDfltTxtFmtColl );
2276 
2277 	// und jetzt noch die Seiten-Vorlagen
2278 	sal_uInt16 nCnt = rSource.aPageDescs.Count();
2279 	if( nCnt )
2280 	{
2281 		// ein anderes Doc -> Numberformatter muessen gemergt werden
2282 		SwTblNumFmtMerge aTNFM( rSource, *this );
2283 
2284 		// 1. Schritt alle Formate anlegen (das 0. ueberspringen - Default!)
2285 		while( nCnt )
2286 		{
2287 			SwPageDesc *pSrc = rSource.aPageDescs[ --nCnt ];
2288 			if( 0 == ::lcl_FindPageDesc( aPageDescs, pSrc->GetName() ) )
2289 				MakePageDesc( pSrc->GetName() );
2290 		}
2291 
2292 		// 2. Schritt alle Attribute kopieren, richtige Parents setzen
2293 		for( nCnt = rSource.aPageDescs.Count(); nCnt; )
2294 		{
2295 			SwPageDesc *pSrc = rSource.aPageDescs[ --nCnt ];
2296 			CopyPageDesc( *pSrc, *::lcl_FindPageDesc( aPageDescs, pSrc->GetName() ));
2297 		}
2298 	}
2299 
2300 	//JP 08.04.99: und dann sind da noch die Numerierungs-Vorlagen
2301 	nCnt = rSource.GetNumRuleTbl().Count();
2302 	if( nCnt )
2303 	{
2304 		const SwNumRuleTbl& rArr = rSource.GetNumRuleTbl();
2305 		for( sal_uInt16 n = 0; n < nCnt; ++n )
2306 		{
2307 			const SwNumRule& rR = *rArr[ n ];
2308 			if( !rR.IsAutoRule() )
2309 			{
2310 				SwNumRule* pNew = FindNumRulePtr( rR.GetName());
2311 				if( pNew )
2312 					pNew->CopyNumRule( this, rR );
2313 				else
2314 					MakeNumRule( rR.GetName(), &rR );
2315 			}
2316 		}
2317 	}
2318 
2319     if (undoGuard.UndoWasEnabled())
2320     {
2321         // nodes array was modified!
2322         GetIDocumentUndoRedo().DelAllUndoObj();
2323     }
2324 
2325 	SetModified();
2326 }
2327 
2328 SwFmt* SwDoc::FindFmtByName( const SvPtrarr& rFmtArr,
2329 									const String& rName ) const
2330 {
2331 	SwFmt* pFnd = 0;
2332 	for( sal_uInt16 n = 0; n < rFmtArr.Count(); n++ )
2333 	{
2334 		// ist die Vorlage schon im Doc vorhanden ??
2335 		if( ((SwFmt*)rFmtArr[n])->GetName() == rName )
2336 		{
2337 			pFnd = (SwFmt*)rFmtArr[n];
2338 			break;
2339 		}
2340 	}
2341 	return pFnd;
2342 }
2343 
2344 void SwDoc::MoveLeftMargin( const SwPaM& rPam, sal_Bool bRight, sal_Bool bModulus )
2345 {
2346 	SwHistory* pHistory = 0;
2347     if (GetIDocumentUndoRedo().DoesUndo())
2348     {
2349 		SwUndoMoveLeftMargin* pUndo = new SwUndoMoveLeftMargin( rPam, bRight,
2350 																bModulus );
2351         pHistory = &pUndo->GetHistory();
2352         GetIDocumentUndoRedo().AppendUndo( pUndo );
2353     }
2354 
2355 	const SvxTabStopItem& rTabItem = (SvxTabStopItem&)GetDefault( RES_PARATR_TABSTOP );
2356 	sal_uInt16 nDefDist = rTabItem.Count() ?
2357         static_cast<sal_uInt16>(rTabItem[0].GetTabPos()) : 1134;
2358 	const SwPosition &rStt = *rPam.Start(), &rEnd = *rPam.End();
2359 	SwNodeIndex aIdx( rStt.nNode );
2360 	while( aIdx <= rEnd.nNode )
2361 	{
2362 		SwTxtNode* pTNd = aIdx.GetNode().GetTxtNode();
2363 		if( pTNd )
2364 		{
2365             SvxLRSpaceItem aLS( (SvxLRSpaceItem&)pTNd->SwCntntNode::GetAttr( RES_LR_SPACE ) );
2366 
2367             // --> FME 2008-09-16 #i93873# See also lcl_MergeListLevelIndentAsLRSpaceItem in thints.cxx
2368             if ( pTNd->AreListLevelIndentsApplicable() )
2369             {
2370                 const SwNumRule* pRule = pTNd->GetNumRule();
2371                 if ( pRule )
2372                 {
2373                     const int nListLevel = pTNd->GetActualListLevel();
2374                     if ( nListLevel >= 0 )
2375                     {
2376                         const SwNumFmt& rFmt = pRule->Get(static_cast<sal_uInt16>(nListLevel));
2377                         if ( rFmt.GetPositionAndSpaceMode() == SvxNumberFormat::LABEL_ALIGNMENT )
2378                         {
2379                             aLS.SetTxtLeft( rFmt.GetIndentAt() );
2380                             aLS.SetTxtFirstLineOfst( static_cast<short>(rFmt.GetFirstLineIndent()) );
2381                         }
2382                     }
2383                 }
2384             }
2385 
2386 			long nNext = aLS.GetTxtLeft();
2387 			if( bModulus )
2388 				nNext = ( nNext / nDefDist ) * nDefDist;
2389 
2390 			if( bRight )
2391 				nNext += nDefDist;
2392 			else
2393 				nNext -= nDefDist;
2394 
2395             aLS.SetTxtLeft( nNext );
2396 
2397 			SwRegHistory aRegH( pTNd, *pTNd, pHistory );
2398             pTNd->SetAttr( aLS );
2399 		}
2400 		aIdx++;
2401 	}
2402 	SetModified();
2403 }
2404 
2405 sal_Bool SwDoc::DontExpandFmt( const SwPosition& rPos, sal_Bool bFlag )
2406 {
2407 	sal_Bool bRet = sal_False;
2408 	SwTxtNode* pTxtNd = rPos.nNode.GetNode().GetTxtNode();
2409 	if( pTxtNd )
2410 	{
2411 		bRet = pTxtNd->DontExpandFmt( rPos.nContent, bFlag );
2412         if( bRet && GetIDocumentUndoRedo().DoesUndo() )
2413         {
2414             GetIDocumentUndoRedo().AppendUndo( new SwUndoDontExpandFmt(rPos) );
2415         }
2416     }
2417 	return bRet;
2418 }
2419 
2420 SwTableBoxFmt* SwDoc::MakeTableBoxFmt()
2421 {
2422 	SwTableBoxFmt* pFmt = new SwTableBoxFmt( GetAttrPool(), aEmptyStr,
2423 												pDfltFrmFmt );
2424 	SetModified();
2425 	return pFmt;
2426 }
2427 
2428 SwTableLineFmt* SwDoc::MakeTableLineFmt()
2429 {
2430 	SwTableLineFmt* pFmt = new SwTableLineFmt( GetAttrPool(), aEmptyStr,
2431 												pDfltFrmFmt );
2432 	SetModified();
2433 	return pFmt;
2434 }
2435 
2436 void SwDoc::_CreateNumberFormatter()
2437 {
2438 	RTL_LOGFILE_CONTEXT_AUTHOR( aLog, "SW", "JP93722",  "SwDoc::_CreateNumberFormatter" );
2439 
2440 	ASSERT( !pNumberFormatter, "ist doch schon vorhanden" );
2441 
2442 
2443 	LanguageType eLang = LANGUAGE_SYSTEM; //System::GetLanguage();
2444 /*				((const SvxLanguageItem&)GetAttrPool().
2445 					GetDefaultItem( RES_CHRATR_LANGUAGE )).GetLanguage();
2446 */
2447 	Reference< XMultiServiceFactory > xMSF = ::comphelper::getProcessServiceFactory();
2448 	pNumberFormatter = new SvNumberFormatter( xMSF, eLang );
2449 	pNumberFormatter->SetEvalDateFormat( NF_EVALDATEFORMAT_FORMAT_INTL );
2450 	pNumberFormatter->SetYear2000(static_cast<sal_uInt16>(::utl::MiscCfg().GetYear2000()));
2451 
2452 }
2453 
2454 SwTblNumFmtMerge::SwTblNumFmtMerge( const SwDoc& rSrc, SwDoc& rDest )
2455 	: pNFmt( 0 )
2456 {
2457 	// ein anderes Doc -> Numberformatter muessen gemergt werden
2458 	SvNumberFormatter* pN;
2459 	if( &rSrc != &rDest && 0 != ( pN = ((SwDoc&)rSrc).GetNumberFormatter( sal_False ) ))
2460 		( pNFmt = rDest.GetNumberFormatter( sal_True ))->MergeFormatter( *pN );
2461 
2462 	if( &rSrc != &rDest )
2463 		((SwGetRefFieldType*)rSrc.GetSysFldType( RES_GETREFFLD ))->
2464 			MergeWithOtherDoc( rDest );
2465 }
2466 
2467 SwTblNumFmtMerge::~SwTblNumFmtMerge()
2468 {
2469 	if( pNFmt )
2470 		pNFmt->ClearMergeTable();
2471 }
2472 
2473 
2474 void SwDoc::SetTxtFmtCollByAutoFmt( const SwPosition& rPos, sal_uInt16 nPoolId,
2475 									const SfxItemSet* pSet )
2476 {
2477 	SwPaM aPam( rPos );
2478 	SwTxtNode* pTNd = rPos.nNode.GetNode().GetTxtNode();
2479 
2480 	if( mbIsAutoFmtRedline && pTNd )
2481 	{
2482 		// dann das Redline Object anlegen
2483 		const SwTxtFmtColl& rColl = *pTNd->GetTxtColl();
2484 		SwRedline* pRedl = new SwRedline( nsRedlineType_t::REDLINE_FMTCOLL, aPam );
2485 		pRedl->SetMark();
2486 
2487 		// interressant sind nur die Items, die vom Set NICHT wieder
2488 		// in den Node gesetzt werden. Also muss man die Differenz nehmen
2489 		SwRedlineExtraData_FmtColl aExtraData( rColl.GetName(),
2490 												rColl.GetPoolFmtId() );
2491         if( pSet && pTNd->HasSwAttrSet() )
2492 		{
2493 			SfxItemSet aTmp( *pTNd->GetpSwAttrSet() );
2494 			aTmp.Differentiate( *pSet );
2495 			// das Adjust Item behalten wir extra
2496 			const SfxPoolItem* pItem;
2497 			if( SFX_ITEM_SET == pTNd->GetpSwAttrSet()->GetItemState(
2498 					RES_PARATR_ADJUST, sal_False, &pItem ))
2499 				aTmp.Put( *pItem );
2500 			aExtraData.SetItemSet( aTmp );
2501 		}
2502 		pRedl->SetExtraData( &aExtraData );
2503 
2504 // !!!!!!!!! Undo fehlt noch !!!!!!!!!!!!!!!!!!
2505 		AppendRedline( pRedl, true );
2506 	}
2507 
2508 	SetTxtFmtColl( aPam, GetTxtCollFromPool( nPoolId ) );
2509 
2510 	if( pSet && pTNd && pSet->Count() )
2511 	{
2512 		aPam.SetMark();
2513 		aPam.GetMark()->nContent.Assign( pTNd, pTNd->GetTxt().Len() );
2514         InsertItemSet( aPam, *pSet, 0 );
2515     }
2516 }
2517 
2518 void SwDoc::SetFmtItemByAutoFmt( const SwPaM& rPam, const SfxItemSet& rSet )
2519 {
2520 	SwTxtNode* pTNd = rPam.GetPoint()->nNode.GetNode().GetTxtNode();
2521 
2522 	RedlineMode_t eOld = GetRedlineMode();
2523 
2524 	if( mbIsAutoFmtRedline && pTNd )
2525 	{
2526 		// dann das Redline Object anlegen
2527 		SwRedline* pRedl = new SwRedline( nsRedlineType_t::REDLINE_FORMAT, rPam );
2528 		if( !pRedl->HasMark() )
2529 			pRedl->SetMark();
2530 
2531 		// interressant sind nur die Items, die vom Set NICHT wieder
2532 		// in den Node gesetzt werden. Also muss man die Differenz nehmen
2533 		SwRedlineExtraData_Format aExtraData( rSet );
2534 
2535 /*
2536         if( pSet && pTNd->HasSwAttrSet() )
2537 		{
2538 			SfxItemSet aTmp( *pTNd->GetpSwAttrSet() );
2539 			aTmp.Differentiate( *pSet );
2540 			// das Adjust Item behalten wir extra
2541 			const SfxPoolItem* pItem;
2542 			if( SFX_ITEM_SET == pTNd->GetpSwAttrSet()->GetItemState(
2543 					RES_PARATR_ADJUST, sal_False, &pItem ))
2544 				aTmp.Put( *pItem );
2545 			aExtraData.SetItemSet( aTmp );
2546 		}
2547 */
2548 		pRedl->SetExtraData( &aExtraData );
2549 
2550 // !!!!!!!!! Undo fehlt noch !!!!!!!!!!!!!!!!!!
2551 		AppendRedline( pRedl, true );
2552 
2553 		SetRedlineMode_intern( (RedlineMode_t)(eOld | nsRedlineMode_t::REDLINE_IGNORE));
2554 	}
2555 
2556     InsertItemSet( rPam, rSet, nsSetAttrMode::SETATTR_DONTEXPAND );
2557 	SetRedlineMode_intern( eOld );
2558 }
2559 
2560 void SwDoc::ChgFmt(SwFmt & rFmt, const SfxItemSet & rSet)
2561 {
2562     if (GetIDocumentUndoRedo().DoesUndo())
2563     {
2564         // copying <rSet> to <aSet>
2565         SfxItemSet aSet(rSet);
2566         // remove from <aSet> all items, which are already set at the format
2567         aSet.Differentiate(rFmt.GetAttrSet());
2568         // <aSet> contains now all *new* items for the format
2569 
2570         // copying current format item set to <aOldSet>
2571         SfxItemSet aOldSet(rFmt.GetAttrSet());
2572         // insert new items into <aOldSet>
2573         aOldSet.Put(aSet);
2574         // invalidate all new items in <aOldSet> in order to clear these items,
2575         // if the undo action is triggered.
2576         {
2577             SfxItemIter aIter(aSet);
2578 
2579             const SfxPoolItem * pItem = aIter.FirstItem();
2580             while (pItem != NULL)
2581             {
2582                 aOldSet.InvalidateItem(pItem->Which());
2583 
2584                 pItem = aIter.NextItem();
2585             }
2586         }
2587 
2588         SwUndo * pUndo = new SwUndoFmtAttr(aOldSet, rFmt);
2589 
2590         GetIDocumentUndoRedo().AppendUndo(pUndo);
2591     }
2592 
2593     rFmt.SetFmtAttr(rSet);
2594 }
2595 
2596 void SwDoc::RenameFmt(SwFmt & rFmt, const String & sNewName,
2597                       sal_Bool bBroadcast)
2598 {
2599     SfxStyleFamily eFamily = SFX_STYLE_FAMILY_ALL;
2600 
2601     if (GetIDocumentUndoRedo().DoesUndo())
2602     {
2603         SwUndo * pUndo = NULL;
2604 
2605         switch (rFmt.Which())
2606         {
2607         case RES_CHRFMT:
2608             pUndo = new SwUndoRenameCharFmt(rFmt.GetName(), sNewName, this);
2609             eFamily = SFX_STYLE_FAMILY_PARA;
2610             break;
2611         case RES_TXTFMTCOLL:
2612             pUndo = new SwUndoRenameFmtColl(rFmt.GetName(), sNewName, this);
2613             eFamily = SFX_STYLE_FAMILY_CHAR;
2614             break;
2615         case RES_FRMFMT:
2616             pUndo = new SwUndoRenameFrmFmt(rFmt.GetName(), sNewName, this);
2617             eFamily = SFX_STYLE_FAMILY_FRAME;
2618             break;
2619 
2620         default:
2621             break;
2622         }
2623 
2624         if (pUndo)
2625         {
2626             GetIDocumentUndoRedo().AppendUndo(pUndo);
2627         }
2628     }
2629 
2630     rFmt.SetName(sNewName);
2631 
2632     if (bBroadcast)
2633         BroadcastStyleOperation(sNewName, eFamily, SFX_STYLESHEET_MODIFIED);
2634 }
2635 
2636 // --> OD 2006-09-27 #i69627#
2637 namespace docfunc
2638 {
2639     bool HasOutlineStyleToBeWrittenAsNormalListStyle( SwDoc& rDoc )
2640     {
2641         // If a parent paragraph style of one of the parargraph styles, which
2642         // are assigned to the list levels of the outline style, has a list style
2643         // set or inherits a list style from its parent style, the outline style
2644         // has to be written as a normal list style to the OpenDocument file
2645         // format or the OpenOffice.org file format.
2646         bool bRet( false );
2647 
2648         const SwTxtFmtColls* pTxtFmtColls( rDoc.GetTxtFmtColls() );
2649         if ( pTxtFmtColls )
2650         {
2651             const sal_uInt16 nCount = pTxtFmtColls->Count();
2652             for ( sal_uInt16 i = 0; i < nCount; ++i )
2653             {
2654                 SwTxtFmtColl* pTxtFmtColl = (*pTxtFmtColls)[i];
2655 
2656                 if ( pTxtFmtColl->IsDefault() ||
2657                    //  pTxtFmtColl->GetOutlineLevel() == NO_NUMBERING ) //#outline level,zhaojianwei
2658 					! pTxtFmtColl->IsAssignedToListLevelOfOutlineStyle() )	//<-end,zhaojianwei
2659                 {
2660                     continue;
2661                 }
2662 
2663                 const SwTxtFmtColl* pParentTxtFmtColl =
2664                    dynamic_cast<const SwTxtFmtColl*>( pTxtFmtColl->DerivedFrom());
2665                 if ( !pParentTxtFmtColl )
2666                     continue;
2667 
2668                 if ( SFX_ITEM_SET == pParentTxtFmtColl->GetItemState( RES_PARATR_NUMRULE ) )
2669                 {
2670                     // --> OD 2009-11-12 #i106218#
2671                     // consider that the outline style is set
2672                     const SwNumRuleItem& rDirectItem = pParentTxtFmtColl->GetNumRule();
2673                     if ( rDirectItem.GetValue() != rDoc.GetOutlineNumRule()->GetName() )
2674                     {
2675                         bRet = true;
2676                         break;
2677                     }
2678                     // <--
2679                 }
2680             }
2681 
2682         }
2683         return bRet;
2684     }
2685 }
2686 // <--
2687