summaryrefslogtreecommitdiff
path: root/sw/source/core/doc/docnum.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'sw/source/core/doc/docnum.cxx')
-rw-r--r--sw/source/core/doc/docnum.cxx2969
1 files changed, 2969 insertions, 0 deletions
diff --git a/sw/source/core/doc/docnum.cxx b/sw/source/core/doc/docnum.cxx
new file mode 100644
index 000000000000..1224c5c02f17
--- /dev/null
+++ b/sw/source/core/doc/docnum.cxx
@@ -0,0 +1,2969 @@
+/*************************************************************************
+ *
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * Copyright 2000, 2010 Oracle and/or its affiliates.
+ *
+ * OpenOffice.org - a multi-platform office productivity suite
+ *
+ * This file is part of OpenOffice.org.
+ *
+ * OpenOffice.org is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License version 3
+ * only, as published by the Free Software Foundation.
+ *
+ * OpenOffice.org is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Lesser General Public License version 3 for more details
+ * (a copy is included in the LICENSE file that accompanied this code).
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * version 3 along with OpenOffice.org. If not, see
+ * <http://www.openoffice.org/license.html>
+ * for a copy of the LGPLv3 License.
+ *
+ ************************************************************************/
+
+// MARKER(update_precomp.py): autogen include statement, do not remove
+#include "precompiled_sw.hxx"
+#include <hintids.hxx>
+#include <tools/date.hxx>
+#include <tools/time.hxx>
+#include <tools/resid.hxx>
+#include <editeng/lrspitem.hxx>
+#include <ftninfo.hxx>
+#include <ftnidx.hxx>
+#include <doc.hxx>
+#include <pam.hxx>
+#include <ndtxt.hxx>
+#include <doctxm.hxx> // pTOXBaseRing
+#include <poolfmt.hxx>
+#include <undobj.hxx>
+#include <swundo.hxx>
+#include <SwUndoFmt.hxx>
+#include <rolbck.hxx>
+#include <paratr.hxx>
+#include <docary.hxx>
+#include <mvsave.hxx>
+#include <txtfrm.hxx>
+#include <pamtyp.hxx>
+#include <redline.hxx>
+#ifndef _COMCORE_HRC
+#include <comcore.hrc>
+#endif
+#include <editeng/adjitem.hxx>
+#include <editeng/frmdiritem.hxx>
+#include <frmatr.hxx>
+#include <SwStyleNameMapper.hxx>
+#include <SwNodeNum.hxx>
+// --> OD 2008-03-13 #refactorlists#
+#include <list.hxx>
+#include <listfunc.hxx>
+// <--
+
+#include <map>
+
+inline BYTE GetUpperLvlChg( BYTE nCurLvl, BYTE nLevel, USHORT nMask )
+{
+ if( 1 < nLevel )
+ {
+ if( nCurLvl + 1 >= nLevel )
+ nCurLvl -= nLevel - 1;
+ else
+ nCurLvl = 0;
+ }
+ return static_cast<BYTE>((nMask - 1) & ~(( 1 << nCurLvl ) - 1));
+}
+
+void SwDoc::SetOutlineNumRule( const SwNumRule& rRule )
+{
+ if( pOutlineRule )
+ (*pOutlineRule) = rRule;
+ else
+ {
+ pOutlineRule = new SwNumRule( rRule );
+
+ AddNumRule(pOutlineRule); // #i36749#
+ }
+
+ pOutlineRule->SetRuleType( OUTLINE_RULE );
+ // --> OD 2008-07-08 #i91400#
+ pOutlineRule->SetName( String::CreateFromAscii(
+ SwNumRule::GetOutlineRuleName() ),
+ *this);
+ // <--
+ // --> OD 2006-09-21 #i69522#
+ // assure that the outline numbering rule is an automatic rule
+ pOutlineRule->SetAutoRule( TRUE );
+ // <--
+
+ // teste ob die evt. gesetzen CharFormate in diesem Document
+ // definiert sind
+ pOutlineRule->CheckCharFmts( this );
+
+ // --> OD 2008-05-13 #refactorlists#
+ // notify text nodes, which are registered at the outline style, about the
+ // changed outline style
+ SwNumRule::tTxtNodeList aTxtNodeList;
+ pOutlineRule->GetTxtNodeList( aTxtNodeList );
+ for ( SwNumRule::tTxtNodeList::iterator aIter = aTxtNodeList.begin();
+ aIter != aTxtNodeList.end(); ++aIter )
+ {
+ SwTxtNode* pTxtNd = *aIter;
+ pTxtNd->NumRuleChgd();
+ // --> OD 2009-01-20 #i94152#
+ // assure that list level corresponds to outline level
+ if ( pTxtNd->GetTxtColl()->IsAssignedToListLevelOfOutlineStyle() &&
+ pTxtNd->GetAttrListLevel() != pTxtNd->GetTxtColl()->GetAssignedOutlineStyleLevel() )
+ {
+ pTxtNd->SetAttrListLevel( pTxtNd->GetTxtColl()->GetAssignedOutlineStyleLevel() );
+ }
+ // <--
+ }
+ // <--
+
+ PropagateOutlineRule();
+ pOutlineRule->SetInvalidRule(TRUE);
+ UpdateNumRule();
+
+ // gibt es Fussnoten && gilt Kapitelweises Nummerieren, dann updaten
+ if( GetFtnIdxs().Count() && FTNNUM_CHAPTER == GetFtnInfo().eNum )
+ GetFtnIdxs().UpdateAllFtn();
+
+ UpdateExpFlds(NULL, true);
+
+ SetModified();
+}
+
+void SwDoc::PropagateOutlineRule()
+{
+ for (USHORT n = 0; n < pTxtFmtCollTbl->Count(); n++)
+ {
+ SwTxtFmtColl *pColl = (*pTxtFmtCollTbl)[n];
+
+ // if (NO_NUMBERING != pColl->GetOutlineLevel())//#outline level,zhaojianwei
+ if(pColl->IsAssignedToListLevelOfOutlineStyle())//<-end,zhaojianwei
+ {
+ SwClientIter aIter(*pColl);
+
+ // --> OD 2006-11-20 #i71764#
+ // Check only the list style, which is set at the paragraph style
+ const SwNumRuleItem & rCollRuleItem = pColl->GetNumRule( FALSE );
+ // <--
+
+ // --> OD 2006-11-20 #i71764#
+ // Check on document setting OUTLINE_LEVEL_YIELDS_OUTLINE_RULE no longer needed.
+ if ( rCollRuleItem.GetValue().Len() == 0 )
+ // <--
+ {
+ SwNumRule * pMyOutlineRule = GetOutlineNumRule();
+
+ if (pMyOutlineRule)
+ {
+ SwNumRuleItem aNumItem( pMyOutlineRule->GetName() );
+
+ pColl->SetFmtAttr(aNumItem);
+ }
+ }
+ }
+ }
+}
+
+ // Hoch-/Runterstufen
+BOOL SwDoc::OutlineUpDown( const SwPaM& rPam, short nOffset )
+{
+ if( !GetNodes().GetOutLineNds().Count() || !nOffset )
+ return FALSE;
+
+ // den Bereich feststellen
+ const SwOutlineNodes& rOutlNds = GetNodes().GetOutLineNds();
+ const SwNodePtr pSttNd = (SwNodePtr)&rPam.Start()->nNode.GetNode();
+ const SwNodePtr pEndNd = (SwNodePtr)&rPam.End()->nNode.GetNode();
+ USHORT nSttPos, nEndPos;
+
+ if( !rOutlNds.Seek_Entry( pSttNd, &nSttPos ) &&
+ !nSttPos-- )
+ // wir stehen in keiner "Outline-Section"
+ return FALSE;
+
+ if( rOutlNds.Seek_Entry( pEndNd, &nEndPos ) )
+ ++nEndPos;
+
+ // jetzt haben wir unseren Bereich im OutlineNodes-Array
+ // dann prufe ersmal, ob nicht unterebenen aufgehoben werden
+ // (Stufung ueber die Grenzen)
+ USHORT n;
+
+ // so, dann koennen wir:
+ // 1. Vorlagen-Array anlegen
+ SwTxtFmtColl* aCollArr[ MAXLEVEL ];
+ memset( aCollArr, 0, sizeof( SwTxtFmtColl* ) * MAXLEVEL );
+
+ for( n = 0; n < pTxtFmtCollTbl->Count(); ++n )
+ {
+ //BYTE nLevel = (*pTxtFmtCollTbl)[ n ]->GetOutlineLevel();//#outline level,zhaojianwei
+ //if( nLevel < MAXLEVEL )
+ // aCollArr[ nLevel ] = (*pTxtFmtCollTbl)[ n ];
+ if((*pTxtFmtCollTbl)[ n ]->IsAssignedToListLevelOfOutlineStyle())
+ {
+ const int nLevel = (*pTxtFmtCollTbl)[ n ]->GetAssignedOutlineStyleLevel();
+ aCollArr[ nLevel ] = (*pTxtFmtCollTbl)[ n ];
+ }//<-end,zhaojianwei
+ }
+
+ /* --> #111107# */
+ /* Find the last occupied level (backward). */
+ for (n = MAXLEVEL - 1; n > 0; n--)
+ {
+ if (aCollArr[n] != 0)
+ break;
+ }
+
+ /* If an occupied level is found, choose next level (which IS
+ unoccupied) until a valid level is found. If no occupied level
+ was found n is 0 and aCollArr[0] is 0. In this case no demoting
+ is possible. */
+ if (aCollArr[n] != 0)
+ {
+ while (n < MAXLEVEL - 1)
+ {
+ n++;
+
+ SwTxtFmtColl *aTmpColl =
+ GetTxtCollFromPool(static_cast<sal_uInt16>(RES_POOLCOLL_HEADLINE1 + n));
+
+ //if (aTmpColl->GetOutlineLevel() == n)//#outline level,zhaojianwei
+ if( aTmpColl->IsAssignedToListLevelOfOutlineStyle() &&
+ aTmpColl->GetAssignedOutlineStyleLevel() == n )//<-end,zhaojianwei
+ {
+ aCollArr[n] = aTmpColl;
+ break;
+ }
+ }
+ }
+
+ /* Find the first occupied level (forward). */
+ for (n = 0; n < MAXLEVEL - 1; n++)
+ {
+ if (aCollArr[n] != 0)
+ break;
+ }
+
+ /* If an occupied level is found, choose previous level (which IS
+ unoccupied) until a valid level is found. If no occupied level
+ was found n is MAXLEVEL - 1 and aCollArr[MAXLEVEL - 1] is 0. In
+ this case no demoting is possible. */
+ if (aCollArr[n] != 0)
+ {
+ while (n > 0)
+ {
+ n--;
+
+ SwTxtFmtColl *aTmpColl =
+ GetTxtCollFromPool(static_cast<sal_uInt16>(RES_POOLCOLL_HEADLINE1 + n));
+
+ //if (aTmpColl->GetOutlineLevel() == n)//#outline level,zhaojianwei
+ if( aTmpColl->IsAssignedToListLevelOfOutlineStyle() &&
+ aTmpColl->GetAssignedOutlineStyleLevel() == n )//<-end,zhaojianwei
+ {
+ aCollArr[n] = aTmpColl;
+ break;
+ }
+ }
+ }
+ /* <-- #111107# */
+
+ /* --> #i13747#
+
+ Build a move table that states from which level an outline will
+
+ be moved to which other level. */
+
+ /* the move table
+
+ aMoveArr[n] = m: replace aCollArr[n] with aCollArr[m]
+ */
+ int aMoveArr[MAXLEVEL];
+ int nStep; // step size for searching in aCollArr: -1 or 1
+ int nNum; // amount of steps for stepping in aCollArr
+
+ if (nOffset < 0)
+ {
+ nStep = -1;
+ nNum = -nOffset;
+ }
+ else
+ {
+ nStep = 1;
+ nNum = nOffset;
+ }
+
+ /* traverse aCollArr */
+ for (n = 0; n < MAXLEVEL; n++)
+ {
+ /* If outline level n has an assigned paragraph style step
+ nNum steps forwards (nStep == 1) or backwards (nStep ==
+ -1). One step is to go to the next non-null entry in
+ aCollArr in the selected direction. If nNum steps were
+ possible write the index of the entry found to aCollArr[n],
+ i.e. outline level n will be replaced by outline level
+ aCollArr[n].
+
+ If outline level n has no assigned paragraph style
+ aMoveArr[n] is set to -1.
+ */
+ if (aCollArr[n] != NULL)
+ {
+ USHORT m = n;
+ int nCount = nNum;
+
+ while (nCount > 0 && m + nStep >= 0 && m + nStep < MAXLEVEL)
+ {
+ m = static_cast<USHORT>(m + nStep);
+
+ if (aCollArr[m] != NULL)
+ nCount--;
+ }
+
+ if (nCount == 0)
+ aMoveArr[n] = m;
+ else
+ aMoveArr[n] = -1;
+
+ }
+ else
+ aMoveArr[n] = -1;
+ }
+
+ /* If moving of the outline levels is applicable, i.e. for all
+ outline levels occuring in the document there has to be a valid
+ target outline level implied by aMoveArr. */
+ bool bMoveApplicable = true;
+ for (n = nSttPos; n < nEndPos; n++)
+ {
+ SwTxtNode* pTxtNd = rOutlNds[ n ]->GetTxtNode();
+ SwTxtFmtColl* pColl = pTxtNd->GetTxtColl();
+// int nLevel = pColl->GetOutlineLevel();//#outline level,zhaojianwei
+// if (aMoveArr[nLevel] == -1)
+// bMoveApplicable = false;
+ if( pColl->IsAssignedToListLevelOfOutlineStyle() )
+ {
+ const int nLevel = pColl->GetAssignedOutlineStyleLevel();
+ if (aMoveArr[nLevel] == -1)
+ bMoveApplicable = false;
+ }//<-end,zhaojianwei
+ // --> OD 2008-12-16 #i70748#
+ // Check on outline level attribute of text node, if text node is
+ // not an outline via a to outline style assigned paragraph style.
+ else
+ {
+ const int nNewOutlineLevel = pTxtNd->GetAttrOutlineLevel() + nOffset;
+ if ( nNewOutlineLevel < 1 || nNewOutlineLevel > MAXLEVEL )
+ {
+ bMoveApplicable = false;
+ }
+ }
+ // <--
+ }
+
+ if (! bMoveApplicable )
+ return FALSE;
+
+ /* <-- #i13747 # */
+ if( DoesUndo() )
+ {
+ ClearRedo();
+ StartUndo(UNDO_OUTLINE_LR, NULL);
+ AppendUndo( new SwUndoOutlineLeftRight( rPam, nOffset ) );
+ }
+
+ // 2. allen Nodes die neue Vorlage zuweisen
+
+ n = nSttPos;
+ while( n < nEndPos)
+ {
+ SwTxtNode* pTxtNd = rOutlNds[ n ]->GetTxtNode();
+ SwTxtFmtColl* pColl = pTxtNd->GetTxtColl();
+
+ if( pColl->IsAssignedToListLevelOfOutlineStyle() )
+ {
+ // ASSERT(pColl->GetOutlineLevel() < MAXLEVEL, //#outline level,removed by zhaojianwei
+ // "non outline node in outline nodes?");
+ //int nLevel = pColl->GetOutlineLevel();
+ const int nLevel = pColl->GetAssignedOutlineStyleLevel();//#outline level,add by zhaojianwei
+
+ ASSERT(aMoveArr[nLevel] >= 0,
+ "move table: current TxtColl not found when building table!");
+
+
+ if (nLevel < MAXLEVEL && aMoveArr[nLevel] >= 0)
+ {
+ pColl = aCollArr[ aMoveArr[nLevel] ];
+
+ if (pColl != NULL)
+ pColl = (SwTxtFmtColl*)pTxtNd->ChgFmtColl( pColl );
+ }
+
+ }
+ else if( pTxtNd->GetAttrOutlineLevel() > 0) //#outline level,add by zhaojianwei
+ {
+ int nLevel = pTxtNd->GetAttrOutlineLevel() + nOffset;
+ if( 0 <= nLevel && nLevel <= MAXLEVEL)
+ pTxtNd->SetAttrOutlineLevel( nLevel );
+
+ }//<-end,zhaojianwei
+
+ n++;
+ // Undo ???
+ }
+ if (DoesUndo())
+ EndUndo(UNDO_OUTLINE_LR, NULL);
+
+ ChkCondColls();
+ SetModified();
+
+ return TRUE;
+}
+
+
+
+ // Hoch-/Runter - Verschieben !
+BOOL SwDoc::MoveOutlinePara( const SwPaM& rPam, short nOffset )
+{
+ // kein Verschiebung in den Sonderbereichen
+ const SwPosition& rStt = *rPam.Start(),
+ & rEnd = &rStt == rPam.GetPoint() ? *rPam.GetMark()
+ : *rPam.GetPoint();
+ if( !GetNodes().GetOutLineNds().Count() || !nOffset ||
+ rStt.nNode.GetIndex() < aNodes.GetEndOfExtras().GetIndex() ||
+ rEnd.nNode.GetIndex() < aNodes.GetEndOfExtras().GetIndex() )
+ return FALSE;
+
+ USHORT nAktPos = 0;
+ SwNodeIndex aSttRg( rStt.nNode ), aEndRg( rEnd.nNode );
+
+ //BYTE nOutLineLevel = NO_NUMBERING; //#outline level,zhaojianwei
+ int nOutLineLevel = MAXLEVEL; //<-end,zhaojianwei
+ SwNode* pSrch = &aSttRg.GetNode();
+ //if( pSrch->IsTxtNode() ) //#outline level,zhaojianwei
+ // nOutLineLevel = static_cast<BYTE>(((SwTxtNode*)pSrch)->GetOutlineLevel());
+ if( pSrch->IsTxtNode())
+ nOutLineLevel = static_cast<BYTE>(((SwTxtNode*)pSrch)->GetAttrOutlineLevel()-1);//<-end,zhaojianwei
+ SwNode* pEndSrch = &aEndRg.GetNode();
+ if( !GetNodes().GetOutLineNds().Seek_Entry( pSrch, &nAktPos ) )
+ {
+ if( !nAktPos )
+ return FALSE; // Promoting or demoting before the first outline => no.
+ if( --nAktPos )
+ aSttRg = *GetNodes().GetOutLineNds()[ nAktPos ];
+ else if( 0 > nOffset )
+ return FALSE; // Promoting at the top of document?!
+ else
+ aSttRg = *GetNodes().GetEndOfContent().StartOfSectionNode();
+ }
+ USHORT nTmpPos = 0;
+ // If the given range ends at an outlined text node we have to decide if it has to be a part of
+ // the moving range or not. Normally it will be a sub outline of our chapter
+ // and has to be moved, too. But if the chapter ends with a table(or a section end),
+ // the next text node will be choosen and this could be the next outline of the same level.
+ // The criteria has to be the outline level: sub level => incorporate, same/higher level => no.
+ if( GetNodes().GetOutLineNds().Seek_Entry( pEndSrch, &nTmpPos ) )
+ {
+ if( !pEndSrch->IsTxtNode() || pEndSrch == pSrch ||
+ //nOutLineLevel < ((SwTxtNode*)pEndSrch)->GetOutlineLevel() )//#outline level,zhaojianwei
+ nOutLineLevel < ((SwTxtNode*)pEndSrch)->GetAttrOutlineLevel()-1 )//<-end,zhaojianwei
+ ++nTmpPos; // For sub outlines only!
+ }
+
+ aEndRg = nTmpPos < GetNodes().GetOutLineNds().Count()
+ ? *GetNodes().GetOutLineNds()[ nTmpPos ]
+ : GetNodes().GetEndOfContent();
+ if( nOffset >= 0 )
+ nAktPos = nTmpPos;
+ if( aEndRg == aSttRg )
+ {
+ ASSERT( false, "Moving outlines: Surprising selection" );
+ aEndRg++;
+ }
+
+ const SwNode* pNd;
+ // The following code corrects the range to handle sections (start/end nodes)
+ // The range will be extended if the least node before the range is a start node
+ // which ends inside the range => The complete section will be moved.
+ // The range will be shrinked if the last position is a start node.
+ // The range will be shrinked if the last node is an end node which starts before the range.
+ aSttRg--;
+ while( aSttRg.GetNode().IsStartNode() )
+ {
+ pNd = aSttRg.GetNode().EndOfSectionNode();
+ if( pNd->GetIndex() >= aEndRg.GetIndex() )
+ break;
+ aSttRg--;
+ }
+ aSttRg++;
+
+ aEndRg--;
+ while( aEndRg.GetNode().IsStartNode() )
+ aEndRg--;
+ while( aEndRg.GetNode().IsEndNode() )
+ {
+ pNd = aEndRg.GetNode().StartOfSectionNode();
+ if( pNd->GetIndex() >= aSttRg.GetIndex() )
+ break;
+ aEndRg--;
+ }
+ aEndRg++;
+
+ // calculation of the new position
+ if( nOffset < 0 && nAktPos < USHORT(-nOffset) )
+ pNd = GetNodes().GetEndOfContent().StartOfSectionNode();
+ else if( nAktPos + nOffset >= GetNodes().GetOutLineNds().Count() )
+ pNd = &GetNodes().GetEndOfContent();
+ else
+ pNd = GetNodes().GetOutLineNds()[ nAktPos + nOffset ];
+
+ ULONG nNewPos = pNd->GetIndex();
+
+ // And now a correction of the insert position if necessary...
+ SwNodeIndex aInsertPos( *pNd, -1 );
+ while( aInsertPos.GetNode().IsStartNode() )
+ {
+ // Just before the insert position starts a section:
+ // when I'm moving forward I do not want to enter the section,
+ // when I'm moving backward I want to stay in the section if I'm already a part of,
+ // I want to stay outside if I was outside before.
+ if( nOffset < 0 )
+ {
+ pNd = aInsertPos.GetNode().EndOfSectionNode();
+ if( pNd->GetIndex() >= aEndRg.GetIndex() )
+ break;
+ }
+ aInsertPos--;
+ --nNewPos;
+ }
+ if( nOffset >= 0 )
+ {
+ // When just before the insert position a section ends, it is okay when I'm moving backward
+ // because I want to stay outside the section.
+ // When moving forward I've to check if I started inside or outside the section
+ // because I don't want to enter of leave such a section
+ while( aInsertPos.GetNode().IsEndNode() )
+ {
+ pNd = aInsertPos.GetNode().StartOfSectionNode();
+ if( pNd->GetIndex() >= aSttRg.GetIndex() )
+ break;
+ aInsertPos--;
+ --nNewPos;
+ }
+ }
+ // We do not want to move into tables (at the moment)
+ aInsertPos++;
+ pNd = &aInsertPos.GetNode();
+ if( pNd->IsTableNode() )
+ pNd = pNd->StartOfSectionNode();
+ if( pNd->FindTableNode() )
+ return FALSE;
+
+ ASSERT( aSttRg.GetIndex() > nNewPos || nNewPos >= aEndRg.GetIndex(),
+ "Position liegt im MoveBereich" );
+
+ // wurde ein Position in den Sonderbereichen errechnet, dann
+ // setze die Position auf den Dokumentanfang.
+ // Sollten da Bereiche oder Tabellen stehen, so werden sie nach
+ // hinten verschoben.
+ nNewPos = Max( nNewPos, aNodes.GetEndOfExtras().GetIndex() + 2 );
+
+ long nOffs = nNewPos - ( 0 < nOffset ? aEndRg.GetIndex() : aSttRg.GetIndex());
+ SwPaM aPam( aSttRg, aEndRg, 0, -1 );
+ return MoveParagraph( aPam, nOffs, TRUE );
+}
+
+
+USHORT lcl_FindOutlineName( const SwNodes& rNds, const String& rName,
+ BOOL bExact )
+{
+ USHORT nSavePos = USHRT_MAX;
+ const SwOutlineNodes& rOutlNds = rNds.GetOutLineNds();
+ for( USHORT n = 0; n < rOutlNds.Count(); ++n )
+ {
+ SwTxtNode* pTxtNd = rOutlNds[ n ]->GetTxtNode();
+ String sTxt( pTxtNd->GetExpandTxt() );
+ if( sTxt.Equals( rName ) )
+ {
+ // "exact" gefunden, setze Pos auf den Node
+ nSavePos = n;
+ break;
+ }
+ else if( !bExact && USHRT_MAX == nSavePos &&
+ COMPARE_EQUAL == sTxt.CompareTo( rName, rName.Len()) )
+ {
+ // dann vielleicht nur den den 1.Teil vom Text gefunden
+ nSavePos = n;
+ }
+ }
+
+ return nSavePos;
+}
+
+
+
+USHORT lcl_FindOutlineNum( const SwNodes& rNds, String& rName )
+{
+ // Gueltig Nummern sind (immer nur Offsets!!!):
+ // ([Nummer]+\.)+ (als regulaerer Ausdruck!)
+ // (Nummer gefolgt von Punkt, zum 5 Wiederholungen)
+ // also: "1.1.", "1.", "1.1.1."
+ xub_StrLen nPos = 0;
+ String sNum = rName.GetToken( 0, '.', nPos );
+ if( STRING_NOTFOUND == nPos )
+ return USHRT_MAX; // ungueltige Nummer!!!
+
+ USHORT nLevelVal[ MAXLEVEL ]; // Nummern aller Levels
+ memset( nLevelVal, 0, MAXLEVEL * sizeof( nLevelVal[0] ));
+ BYTE nLevel = 0;
+ String sName( rName );
+
+ while( STRING_NOTFOUND != nPos )
+ {
+ USHORT nVal = 0;
+ sal_Unicode c;
+ for( USHORT n = 0; n < sNum.Len(); ++n )
+ if( '0' <= ( c = sNum.GetChar( n )) && c <= '9' )
+ {
+ nVal *= 10; nVal += c - '0';
+ }
+ else if( nLevel )
+ break; // "fast" gueltige Nummer
+ else
+ return USHRT_MAX; // ungueltige Nummer!!!
+
+ if( MAXLEVEL > nLevel )
+ nLevelVal[ nLevel++ ] = nVal;
+
+ sName.Erase( 0, nPos );
+ nPos = 0;
+ sNum = sName.GetToken( 0, '.', nPos );
+ // #i4533# without this check all parts delimited by a dot are treated as outline numbers
+ if(!ByteString(sNum, gsl_getSystemTextEncoding()).IsNumericAscii())
+ nPos = STRING_NOTFOUND;
+ }
+ rName = sName; // das ist der nachfolgende Text.
+
+ // alle Levels gelesen, dann suche mal im Document nach dieser
+ // Gliederung:
+ const SwOutlineNodes& rOutlNds = rNds.GetOutLineNds();
+ // OS: ohne OutlineNodes lohnt die Suche nicht
+ // und man spart sich einen Absturz #42958#
+ if(!rOutlNds.Count())
+ return USHRT_MAX;
+ SwTxtNode* pNd;
+ nPos = 0;
+ //search in the existing outline nodes for the required outline num array
+ for( ; nPos < rOutlNds.Count(); ++nPos )
+ {
+ pNd = rOutlNds[ nPos ]->GetTxtNode();
+ //BYTE nLvl = pNd->GetTxtColl()->GetOutlineLevel(); //#outline level,zhaojianwei
+ const int nLvl = pNd->GetAttrOutlineLevel()-1; //<-end,zhaojianwei
+ if( nLvl == nLevel - 1)
+ {
+ // check for the outline num
+ // --> OD 2005-11-02 #i51089 - TUNING#
+ // --> OD 2006-09-22 #i68289#
+ // Assure, that text node has the correct numbering level. Otherwise,
+ // its number vector will not fit to the searched level.
+// if ( pNd->GetNum() )
+ if ( pNd->GetNum() &&
+ pNd->GetActualListLevel() == ( nLevel - 1 ) )
+ // <--
+ {
+ const SwNodeNum & rNdNum = *(pNd->GetNum());
+ SwNumberTree::tNumberVector aLevelVal = rNdNum.GetNumberVector();
+ //now compare with the one searched for
+ bool bEqual = true;
+ for( BYTE n = 0; (n < nLevel) && bEqual; ++n )
+ {
+ bEqual = aLevelVal[n] == nLevelVal[n];
+ }
+ if(bEqual)
+ {
+ break;
+ }
+ }
+ else
+ {
+ // --> OD 2006-01-12 #126588#
+ // A text node, which has an outline paragraph style applied and
+ // has as hard attribute 'no numbering' set, has an outline level,
+ // but no numbering tree node. Thus, consider this situation in
+ // the assertion condition.
+ ASSERT( !pNd->GetNumRule(),
+ "<lcl_FindOutlineNum(..)> - text node with outline level and numbering rule, but without numbering tree node. This is a serious defect -> inform OD" );
+ }
+ }
+ }
+ if( nPos >= rOutlNds.Count() )
+ nPos = USHRT_MAX;
+ return nPos;
+}
+
+ // zu diesem Gliederungspunkt
+
+
+ // JP 13.06.96:
+ // im Namen kann eine Nummer oder/und der Text stehen.
+ // zuerst wird ueber die Nummer versucht den richtigen Eintrag zu finden.
+ // Gibt es diesen, dann wird ueber den Text verglichen, od es der
+ // gewuenschte ist. Ist das nicht der Fall, wird noch mal nur ueber den
+ // Text gesucht. Wird dieser gefunden ist es der Eintrag. Ansonsten der,
+ // der ueber die Nummer gefunden wurde.
+ // Ist keine Nummer angegeben, dann nur den Text suchen.
+
+BOOL SwDoc::GotoOutline( SwPosition& rPos, const String& rName ) const
+{
+ if( rName.Len() )
+ {
+ const SwOutlineNodes& rOutlNds = GetNodes().GetOutLineNds();
+
+ // 1. Schritt: ueber die Nummer:
+ String sName( rName );
+ USHORT nFndPos = ::lcl_FindOutlineNum( GetNodes(), sName );
+ if( USHRT_MAX != nFndPos )
+ {
+ SwTxtNode* pNd = rOutlNds[ nFndPos ]->GetTxtNode();
+ String sExpandedText = pNd->GetExpandTxt();
+ //#i4533# leading numbers followed by a dot have been remove while
+ //searching for the outline position
+ //to compensate this they must be removed from the paragraphs text content, too
+ USHORT nPos = 0;
+ String sTempNum;
+ while(sExpandedText.Len() && (sTempNum = sExpandedText.GetToken(0, '.', nPos)).Len() &&
+ STRING_NOTFOUND != nPos &&
+ ByteString(sTempNum, gsl_getSystemTextEncoding()).IsNumericAscii())
+ {
+ sExpandedText.Erase(0, nPos);
+ nPos = 0;
+ }
+
+ if( !sExpandedText.Equals( sName ) )
+ {
+ USHORT nTmp = ::lcl_FindOutlineName( GetNodes(), sName, TRUE );
+ if( USHRT_MAX != nTmp ) // ueber den Namen gefunden
+ {
+ nFndPos = nTmp;
+ pNd = rOutlNds[ nFndPos ]->GetTxtNode();
+ }
+ }
+ rPos.nNode = *pNd;
+ rPos.nContent.Assign( pNd, 0 );
+ return TRUE;
+ }
+
+ nFndPos = ::lcl_FindOutlineName( GetNodes(), rName, FALSE );
+ if( USHRT_MAX != nFndPos )
+ {
+ SwTxtNode* pNd = rOutlNds[ nFndPos ]->GetTxtNode();
+ rPos.nNode = *pNd;
+ rPos.nContent.Assign( pNd, 0 );
+ return TRUE;
+ }
+
+ // --> OD 2006-09-22 #i68289#
+ // additional search on hyperlink URL without its outline numbering part
+ if ( !sName.Equals( rName ) )
+ {
+ nFndPos = ::lcl_FindOutlineName( GetNodes(), sName, FALSE );
+ if( USHRT_MAX != nFndPos )
+ {
+ SwTxtNode* pNd = rOutlNds[ nFndPos ]->GetTxtNode();
+ rPos.nNode = *pNd;
+ rPos.nContent.Assign( pNd, 0 );
+ return TRUE;
+ }
+ }
+ // <--
+ }
+ return FALSE;
+}
+
+/* */
+
+// --- Nummerierung -----------------------------------------
+
+// --> OD 2008-02-19 #refactorlists#
+//void SwNumRuleInfo::MakeList( SwDoc& rDoc, BOOL )
+//{
+// SwNumRule* pRule = rDoc.FindNumRulePtr(rName);
+
+// // no rule, no fun.
+// if ( !pRule )
+// return;
+
+// //
+// // 1. Case: Information already available at pRule:
+// //
+// if (pRule->GetTxtNodeList())
+// {
+// // copy list to own pList pointer:
+// aList = *pRule->GetTxtNodeList();
+// return;
+// }
+
+// //
+// // 2. Case: Information has to be generated from scratch:
+// //
+
+// if (pRule->IsOutlineRule())
+// {
+// const SwOutlineNodes & rOutlineNodes = rDoc.GetNodes().GetOutLineNds();
+
+// for (USHORT i = 0; i < rOutlineNodes.Count(); ++i)
+// {
+// SwTxtNode & aNode = *((SwTxtNode *) rOutlineNodes[i]);
+
+// if (pRule == aNode.GetNumRule())
+// AddNode(aNode);
+// }
+// }
+// {
+// SwModify* pMod;
+// const SfxPoolItem* pItem;
+// USHORT i, nMaxItems = rDoc.GetAttrPool().GetItemCount
+// ( RES_PARATR_NUMRULE);
+// for( i = 0; i < nMaxItems; ++i )
+// {
+// pItem = rDoc.GetAttrPool().GetItem( RES_PARATR_NUMRULE, i );
+// if( 0 != pItem)
+// {
+// pMod = (SwModify*)((SwNumRuleItem*)pItem)->GetDefinedIn();
+// if (0 != pMod &&
+// ((SwNumRuleItem*)pItem)->GetValue().Len() &&
+// ((SwNumRuleItem*)pItem)->GetValue() == rName )
+// {
+// if( pMod->IsA( TYPE( SwFmt )) )
+// pMod->GetInfo( *this );
+// else
+// {
+// SwTxtNode* pModTxtNode = (SwTxtNode*)pMod;
+
+// // #115901#
+// if( pModTxtNode->GetNodes().IsDocNodes())
+// {
+// AddNode( *pModTxtNode );
+// }
+// }
+// }
+// }
+// }
+// }
+
+// // --> FME 2004-11-03 #i36571# The numrule and this info structure should
+// // have different instances of the list:
+// // --> OD 2006-09-12 #i69145#
+// // method <SwNumRule::SetList(..)> copies content of list provided by the parameter
+// pRule->SetTxtNodeList( aList );
+// // <--
+//}
+// <--
+
+
+void lcl_ChgNumRule( SwDoc& rDoc, const SwNumRule& rRule )
+{
+ SwNumRule* pOld = rDoc.FindNumRulePtr( rRule.GetName() );
+ ASSERT( pOld, "ohne die alte NumRule geht gar nichts" );
+
+ USHORT nChgFmtLevel = 0, nMask = 1;
+ BYTE n;
+
+ for( n = 0; n < MAXLEVEL; ++n, nMask <<= 1 )
+ {
+ const SwNumFmt& rOldFmt = pOld->Get( n ),
+ & rNewFmt = rRule.Get( n );
+
+ if( rOldFmt != rNewFmt )
+ {
+ nChgFmtLevel |= nMask;
+ }
+ else if( SVX_NUM_NUMBER_NONE > rNewFmt.GetNumberingType() && 1 < rNewFmt.GetIncludeUpperLevels() &&
+ 0 != (nChgFmtLevel & GetUpperLvlChg( n, rNewFmt.GetIncludeUpperLevels(),nMask )) )
+ nChgFmtLevel |= nMask;
+ }
+
+ if( !nChgFmtLevel ) // es wurde nichts veraendert?
+ {
+ // --> OD 2006-04-27 #i64311#
+ const bool bInvalidateNumRule( pOld->IsContinusNum() != rRule.IsContinusNum() );
+ // <--
+ pOld->CheckCharFmts( &rDoc );
+ pOld->SetContinusNum( rRule.IsContinusNum() );
+ // --> OD 2008-06-17 #i87166#
+ // Do NOT change list style type
+// pOld->SetRuleType( rRule.GetRuleType() );
+ // <--
+ // --> OD 2006-04-27 #i64311#
+ if ( bInvalidateNumRule )
+ {
+ pOld->SetInvalidRule(TRUE);
+ }
+ // <--
+ return ;
+ }
+
+ // --> OD 2008-02-19 #refactorlists#
+// SwNumRuleInfo* pUpd = new SwNumRuleInfo( rRule.GetName() );
+// pUpd->MakeList( rDoc );
+
+// BYTE nLvl;
+// for( ULONG nFirst = 0, nLast = pUpd->GetList().Count();
+// nFirst < nLast; ++nFirst )
+// {
+// SwTxtNode* pTxtNd = pUpd->GetList().GetObject( nFirst );
+// nLvl = static_cast<BYTE>(pTxtNd->GetLevel());
+
+// if( nLvl < MAXLEVEL )
+// {
+// if( nChgFmtLevel & ( 1 << nLvl ))
+// {
+// pTxtNd->NumRuleChgd();
+// }
+// }
+// }
+ SwNumRule::tTxtNodeList aTxtNodeList;
+ pOld->GetTxtNodeList( aTxtNodeList );
+ BYTE nLvl( 0 );
+ for ( SwNumRule::tTxtNodeList::iterator aIter = aTxtNodeList.begin();
+ aIter != aTxtNodeList.end(); ++aIter )
+ {
+ SwTxtNode* pTxtNd = *aIter;
+ nLvl = static_cast<BYTE>(pTxtNd->GetActualListLevel());
+
+ if( nLvl < MAXLEVEL )
+ {
+ if( nChgFmtLevel & ( 1 << nLvl ))
+ {
+ pTxtNd->NumRuleChgd();
+ }
+ }
+ }
+ // <--
+
+ for( n = 0; n < MAXLEVEL; ++n )
+ if( nChgFmtLevel & ( 1 << n ))
+ pOld->Set( n, rRule.GetNumFmt( n ));
+
+ pOld->CheckCharFmts( &rDoc );
+ pOld->SetInvalidRule(TRUE);
+ pOld->SetContinusNum( rRule.IsContinusNum() );
+ // --> OD 2008-06-17 #i87166#
+ // Do NOT change list style type
+// pOld->SetRuleType( rRule.GetRuleType() );
+ // <--
+
+ // --> OD 2008-02-19 #refactorlists#
+// delete pUpd;
+ // <--
+
+ rDoc.UpdateNumRule();
+}
+
+// OD 2008-02-08 #newlistlevelattrs# - add handling of parameter <bResetIndentAttrs>
+// --> OD 2008-03-17 #refactorlists#
+void SwDoc::SetNumRule( const SwPaM& rPam,
+ const SwNumRule& rRule,
+ const bool bCreateNewList,
+ const String sContinuedListId,
+ sal_Bool bSetItem,
+ const bool bResetIndentAttrs )
+{
+ SwUndoInsNum * pUndo = NULL;
+ if (DoesUndo())
+ {
+ ClearRedo();
+ StartUndo( UNDO_INSNUM, NULL ); // Klammerung fuer die Attribute!
+ AppendUndo( pUndo = new SwUndoInsNum( rPam, rRule ) );
+ }
+
+ SwNumRule * pNew = FindNumRulePtr( rRule.GetName() );
+ bool bUpdateRule = false;
+
+ if( !pNew )
+ {
+ pNew = (*pNumRuleTbl)[ MakeNumRule( rRule.GetName(), &rRule ) ];
+ }
+ else if (rRule != *pNew)
+ {
+ bUpdateRule = true;
+ }
+
+ if (bUpdateRule)
+ {
+ if( pUndo )
+ {
+ pUndo->SaveOldNumRule( *pNew );
+ ::lcl_ChgNumRule( *this, rRule );
+ pUndo->SetLRSpaceEndPos();
+ }
+ else
+ {
+ ::lcl_ChgNumRule( *this, rRule );
+ }
+ }
+
+ // --> OD 2008-03-17 #refactorlists#
+ if ( bSetItem )
+ {
+ if ( bCreateNewList )
+ {
+ String sListId;
+ if ( !bUpdateRule )
+ {
+ // apply list id of list, which has been created for the new list style
+ sListId = pNew->GetDefaultListId();
+ }
+ else
+ {
+ // create new list and apply its list id
+ SwList* pNewList = createList( String(), pNew->GetName() );
+ ASSERT( pNewList,
+ "<SwDoc::SetNumRule(..)> - could not create new list. Serious defect -> please inform OD." );
+ sListId = pNewList->GetListId();
+ }
+ InsertPoolItem( rPam,
+ SfxStringItem( RES_PARATR_LIST_ID, sListId ), 0 );
+ }
+ else if ( sContinuedListId.Len() > 0 )
+ {
+ // apply given list id
+ InsertPoolItem( rPam,
+ SfxStringItem( RES_PARATR_LIST_ID, sContinuedListId ), 0 );
+ }
+ }
+ // <--
+
+ if ( ! rPam.HasMark())
+ {
+ SwTxtNode * pTxtNd = rPam.GetPoint()->nNode.GetNode().GetTxtNode();
+ // --> OD 2006-10-19 #134160#
+ // consider case that the PaM doesn't denote a text node - e.g. it denotes a graphic node
+ if ( pTxtNd )
+ {
+ SwNumRule * pRule = pTxtNd->GetNumRule();
+
+ if (pRule && pRule->GetName() == pNew->GetName())
+ {
+ bSetItem = sal_False;
+ // --> OD 2008-06-02 #refactorlists#
+ if ( !pTxtNd->IsInList() )
+ {
+ pTxtNd->AddToList();
+ }
+ // <--
+ }
+ // --> OD 2005-10-26 #b6340308# - only clear numbering attribute at
+ // text node, if at paragraph style the new numbering rule is found.
+ else if ( !pRule )
+ {
+ SwTxtFmtColl* pColl = pTxtNd->GetTxtColl();
+ if ( pColl )
+ {
+ SwNumRule* pCollRule = FindNumRulePtr(pColl->GetNumRule().GetValue());
+ if ( pCollRule && pCollRule->GetName() == pNew->GetName() )
+ {
+ pTxtNd->ResetAttr( RES_PARATR_NUMRULE );
+ bSetItem = sal_False;
+ }
+ }
+ }
+ // <--
+ }
+ // <--
+ }
+
+ // --> OD 2009-08-18 #i103817#
+ if ( bSetItem )
+ // <--
+ {
+ InsertPoolItem( rPam, SwNumRuleItem( pNew->GetName() ), 0 );
+ }
+
+ // --> OD 2008-02-08 #newlistlevelattrs#
+ if ( bResetIndentAttrs &&
+ pNew && pNew->Get( 0 ).GetPositionAndSpaceMode() == SvxNumberFormat::LABEL_ALIGNMENT )
+ {
+ SvUShortsSort aResetAttrsArray;
+ aResetAttrsArray.Insert( RES_LR_SPACE );
+ ResetAttrs( rPam, sal_True, &aResetAttrsArray );
+ }
+ // <--
+
+ if (DoesUndo())
+ EndUndo( UNDO_INSNUM, NULL );
+
+ SetModified();
+}
+
+void SwDoc::SetCounted(const SwPaM & rPam, bool bCounted)
+{
+ // --> OD 2008-04-03 #refactorlists#
+// ULONG nStartPos = rPam.Start()->nNode.GetIndex();
+// ULONG nEndPos = rPam.End()->nNode.GetIndex();
+
+// for (ULONG n = nStartPos; n <= nEndPos; n++)
+// {
+// SwTxtNode * pNd = GetNodes()[n]->GetTxtNode();
+
+// if (pNd)
+// pNd->SetCountedInList(bCounted);
+// }
+ if ( bCounted )
+ {
+ SvUShortsSort aResetAttrsArray;
+ aResetAttrsArray.Insert( RES_PARATR_LIST_ISCOUNTED );
+ ResetAttrs( rPam, sal_True, &aResetAttrsArray );
+ }
+ else
+ {
+ InsertPoolItem( rPam,
+ SfxBoolItem( RES_PARATR_LIST_ISCOUNTED, FALSE ), 0 );
+ }
+}
+
+//void SwDoc::ReplaceNumRule(const SwPaM & rPaM, const SwNumRule & rNumRule)
+//{
+// if (DoesUndo())
+// StartUndo(UNDO_START, NULL);
+
+// ULONG nStt = rPaM.Start()->nNode.GetIndex();
+// ULONG nEnd = rPaM.End()->nNode.GetIndex();
+
+// for (ULONG n = nStt; n <= nEnd; n++)
+// {
+// SwTxtNode * pCNd = GetNodes()[n]->GetTxtNode();
+
+// if (pCNd && NULL != pCNd->GetNumRule())
+// {
+// SwPaM aPam(*pCNd);
+
+// InsertPoolItem(aPam, SwNumRuleItem(rNumRule.GetName()), 0);
+// }
+// }
+
+// if (DoesUndo())
+// EndUndo(UNDO_START, NULL);
+//}
+
+void SwDoc::SetNumRuleStart( const SwPosition& rPos, BOOL bFlag )
+{
+ SwTxtNode* pTxtNd = rPos.nNode.GetNode().GetTxtNode();
+
+ if (pTxtNd)
+ {
+ const SwNumRule* pRule = pTxtNd->GetNumRule();
+ if( pRule && !bFlag != !pTxtNd->IsListRestart())
+ {
+ if( DoesUndo() )
+ {
+ ClearRedo();
+ AppendUndo( new SwUndoNumRuleStart( rPos, bFlag ));
+ }
+
+ pTxtNd->SetListRestart(bFlag ? true : false);
+
+ SetModified();
+ }
+ }
+}
+
+void SwDoc::SetNodeNumStart( const SwPosition& rPos, USHORT nStt )
+{
+ SwTxtNode* pTxtNd = rPos.nNode.GetNode().GetTxtNode();
+
+ if (pTxtNd)
+ {
+ // --> OD 2008-02-27 #refactorlists#
+// const SwNumRule* pRule = pTxtNd->GetNumRule();
+// if( pRule && nStt != pTxtNd->GetListRestartValue() )
+// {
+// if( DoesUndo() )
+// {
+// ClearRedo();
+// AppendUndo( new SwUndoNumRuleStart( rPos, nStt ));
+// }
+// }
+// pTxtNd->SetListRestartValue(nStt);
+
+// SetModified();
+ if ( !pTxtNd->HasAttrListRestartValue() ||
+ pTxtNd->GetAttrListRestartValue() != nStt )
+ {
+ if( DoesUndo() )
+ {
+ ClearRedo();
+ AppendUndo( new SwUndoNumRuleStart( rPos, nStt ));
+ }
+ pTxtNd->SetAttrListRestartValue( nStt );
+
+ SetModified();
+ }
+ // <--
+ }
+}
+
+ // loeschen geht nur, wenn die Rule niemand benutzt!
+BOOL SwDoc::DelNumRule( const String& rName, BOOL bBroadcast )
+{
+ USHORT nPos = FindNumRule( rName );
+
+ // --> OD 2007-12-17 #151213#
+ if ( (*pNumRuleTbl)[ nPos ] == GetOutlineNumRule() )
+ {
+ ASSERT( false,
+ "<SwDoc::DelNumRule(..)> - No deletion of outline list style. This is serious defect - please inform OD" );
+ return FALSE;
+ }
+ // <--
+
+ if( USHRT_MAX != nPos && !IsUsed( *(*pNumRuleTbl)[ nPos ] ))
+ {
+ if (DoesUndo())
+ {
+ SwUndo * pUndo =
+ new SwUndoNumruleDelete(*(*pNumRuleTbl)[nPos], this);
+
+ AppendUndo(pUndo);
+ }
+
+ if (bBroadcast)
+ BroadcastStyleOperation(rName, SFX_STYLE_FAMILY_PSEUDO,
+ SFX_STYLESHEET_ERASED);
+
+ // --> OD 2008-04-02 #refactorlists#
+ deleteListForListStyle( rName );
+ {
+ // delete further list, which have the deleted list style as default list style
+ std::vector< SwList* > aListsForDeletion;
+ tHashMapForLists::iterator aListIter = maLists.begin();
+ while ( aListIter != maLists.end() )
+ {
+ SwList* pList = (*aListIter).second;
+ if ( pList->GetDefaultListStyleName() == rName )
+ {
+ aListsForDeletion.push_back( pList );
+ }
+
+ ++aListIter;
+ }
+ while ( aListsForDeletion.size() > 0 )
+ {
+ SwList* pList = aListsForDeletion.back();
+ aListsForDeletion.pop_back();
+ deleteList( pList->GetListId() );
+ }
+ }
+ // <--
+ // --> FME 2004-11-02 #i34097# DeleteAndDestroy deletes rName if
+ // rName is directly taken from the numrule.
+ const String aTmpName( rName );
+ // <--
+ pNumRuleTbl->DeleteAndDestroy( nPos );
+ maNumRuleMap.erase(aTmpName);
+
+ SetModified();
+ return TRUE;
+ }
+ return FALSE;
+}
+
+// #106897#
+void SwDoc::ChgNumRuleFmts( const SwNumRule& rRule, const String * pName )
+{
+ // #106897#
+ SwNumRule* pRule = FindNumRulePtr( pName ? *pName : rRule.GetName() );
+ if( pRule )
+ {
+ SwUndoInsNum* pUndo = 0;
+ if( DoesUndo() )
+ {
+ ClearRedo();
+ pUndo = new SwUndoInsNum( *pRule, rRule );
+ pUndo->GetHistory();
+ AppendUndo( pUndo );
+ }
+ ::lcl_ChgNumRule( *this, rRule );
+
+ if( pUndo )
+ pUndo->SetLRSpaceEndPos();
+
+ SetModified();
+ }
+}
+
+sal_Bool SwDoc::RenameNumRule(const String & rOldName, const String & rNewName,
+ BOOL bBroadcast)
+{
+ sal_Bool bResult = sal_False;
+ SwNumRule * pNumRule = FindNumRulePtr(rOldName);
+
+ if (pNumRule)
+ {
+ if (DoesUndo())
+ {
+ SwUndo * pUndo = new SwUndoNumruleRename(rOldName, rNewName, this);
+
+ AppendUndo(pUndo);
+ }
+
+ // --> OD 2008-02-19 #refactorlists#
+// SwNumRuleInfo aInfo(rOldName);
+// aInfo.MakeList(*this);
+ SwNumRule::tTxtNodeList aTxtNodeList;
+ pNumRule->GetTxtNodeList( aTxtNodeList );
+ // <--
+
+ // --> OD 2008-07-08 #i91400#
+ pNumRule->SetName( rNewName, *this );
+ // <--
+
+ SwNumRuleItem aItem(rNewName);
+ // --> OD 2008-02-19 #refactorlists#
+// for (ULONG nI = 0; nI < aInfo.GetList().Count(); ++nI)
+// {
+// SwTxtNode * pTxtNd = aInfo.GetList().GetObject(nI);
+// pTxtNd->SwCntntNode::SetAttr(aItem);
+// }
+ for ( SwNumRule::tTxtNodeList::iterator aIter = aTxtNodeList.begin();
+ aIter != aTxtNodeList.end(); ++aIter )
+ {
+ SwTxtNode * pTxtNd = *aIter;
+ pTxtNd->SetAttr(aItem);
+ }
+ // <--
+
+ bResult = sal_True;
+
+ if (bBroadcast)
+ BroadcastStyleOperation(rOldName, SFX_STYLE_FAMILY_PSEUDO,
+ SFX_STYLESHEET_MODIFIED);
+ }
+
+ return bResult;
+}
+
+void SwDoc::StopNumRuleAnimations( OutputDevice* pOut )
+{
+ for( USHORT n = GetNumRuleTbl().Count(); n; )
+ {
+ // --> OD 2008-02-19 #refactorlists#
+// SwNumRuleInfo aUpd( GetNumRuleTbl()[ --n ]->GetName() );
+// aUpd.MakeList( *this );
+
+// for( ULONG nFirst = 0, nLast = aUpd.GetList().Count();
+// nFirst < nLast; ++nFirst )
+// {
+// SwTxtNode* pTNd = aUpd.GetList().GetObject( nFirst );
+// SwClientIter aIter( *pTNd );
+// for( SwFrm* pFrm = (SwFrm*)aIter.First( TYPE(SwFrm) );
+// pFrm; pFrm = (SwFrm*)aIter.Next() )
+// if( ((SwTxtFrm*)pFrm)->HasAnimation() )
+// ((SwTxtFrm*)pFrm)->StopAnimation( pOut );
+// }
+ SwNumRule::tTxtNodeList aTxtNodeList;
+ GetNumRuleTbl()[ --n ]->GetTxtNodeList( aTxtNodeList );
+ for ( SwNumRule::tTxtNodeList::iterator aTxtNodeIter = aTxtNodeList.begin();
+ aTxtNodeIter != aTxtNodeList.end(); ++aTxtNodeIter )
+ {
+ SwTxtNode* pTNd = *aTxtNodeIter;
+ SwClientIter aIter( *pTNd );
+ for( SwFrm* pFrm = (SwFrm*)aIter.First( TYPE(SwFrm) );
+ pFrm; pFrm = (SwFrm*)aIter.Next() )
+ if( ((SwTxtFrm*)pFrm)->HasAnimation() )
+ ((SwTxtFrm*)pFrm)->StopAnimation( pOut );
+ }
+ // <--
+ }
+}
+
+BOOL SwDoc::ReplaceNumRule( const SwPosition& rPos,
+ const String& rOldRule, const String& rNewRule )
+{
+ BOOL bRet = FALSE;
+ SwNumRule *pOldRule = FindNumRulePtr( rOldRule ),
+ *pNewRule = FindNumRulePtr( rNewRule );
+ if( pOldRule && pNewRule && pOldRule != pNewRule )
+ {
+ // --> OD 2008-02-19 #refactorlists#
+ SwUndoInsNum* pUndo = 0;
+ if( DoesUndo() )
+ {
+ ClearRedo();
+ StartUndo( UNDO_START, NULL ); // Klammerung fuer die Attribute!
+ AppendUndo( pUndo = new SwUndoInsNum( rPos, *pNewRule, rOldRule ) );
+ }
+
+ // --> OD 2008-02-19 #refactorlists#
+ // apply new list style <pNewRule> to all text nodes, which have the
+ // old list style <pOldNRule> applied and belong to the same list as
+ // the text node of the given <SwPosition>.
+// SwNumRuleInfo aUpd( rOldRule );
+// aUpd.MakeList( *this );
+
+// if (aUpd.GetList().Count() > 0) // #106897#
+ SwNumRule::tTxtNodeList aTxtNodeList;
+ pOldRule->GetTxtNodeList( aTxtNodeList );
+ if ( aTxtNodeList.size() > 0 )
+ {
+// // Position suchen und bestimme ob ein Node davor oder dahinter
+// // einen Start erzwingt
+// SwTxtNode* pTxtNd;
+// ULONG nFndPos, nFirst, nLast;
+
+// if( TABLE_ENTRY_NOTFOUND != aUpd.GetList().SearchKey(
+// rPos.nNode.GetIndex(), &nFndPos ))
+// ++nFndPos;
+
+// for( nLast = nFndPos; nLast < aUpd.GetList().Count(); ++nLast )
+// {
+// pTxtNd = aUpd.GetList().GetObject( nLast );
+// if(pTxtNd->IsRestart())
+// break;
+// }
+// for( nFirst = nFndPos; nFirst; )
+// {
+// pTxtNd = aUpd.GetList().GetObject( --nFirst );
+// if( pTxtNd->IsRestart() )
+// break;
+// }
+// // dann neue Numerierung ueber diesen Bereich
+// // definieren und den Start am Anfang/Ende zurueck setzen
+// pTxtNd = aUpd.GetList().GetObject( nFirst );
+// if( pTxtNd->IsRestart() )
+// {
+// pTxtNd->SetRestart(false);
+// if( pUndo )
+// pUndo->SetSttNum( pTxtNd->GetIndex() );
+// }
+
+ SwRegHistory aRegH( pUndo ? pUndo->GetHistory() : 0 );
+ USHORT nChgFmtLevel = 0;
+ for( BYTE n = 0; n < MAXLEVEL; ++n )
+ {
+ const SwNumFmt& rOldFmt = pOldRule->Get( n ),
+ & rNewFmt = pNewRule->Get( n );
+
+ if( rOldFmt.GetAbsLSpace() != rNewFmt.GetAbsLSpace() ||
+ rOldFmt.GetFirstLineOffset() != rNewFmt.GetFirstLineOffset() )
+ nChgFmtLevel |= ( 1 << n );
+ }
+
+ const SwTxtNode* pGivenTxtNode = rPos.nNode.GetNode().GetTxtNode();
+ SwNumRuleItem aRule( rNewRule );
+// for( ; nFirst < nLast; ++nFirst )
+// {
+// pTxtNd = aUpd.GetList().GetObject( nFirst );
+
+// aRegH.RegisterInModify( pTxtNd, *pTxtNd );
+
+// pTxtNd->SwCntntNode::SetAttr( aRule );
+// pTxtNd->NumRuleChgd();
+// }
+ for ( SwNumRule::tTxtNodeList::iterator aIter = aTxtNodeList.begin();
+ aIter != aTxtNodeList.end(); ++aIter )
+ {
+ SwTxtNode* pTxtNd = *aIter;
+
+ if ( pGivenTxtNode &&
+ pGivenTxtNode->GetListId() == pTxtNd->GetListId() )
+ {
+ aRegH.RegisterInModify( pTxtNd, *pTxtNd );
+
+ pTxtNd->SetAttr( aRule );
+ pTxtNd->NumRuleChgd();
+ }
+ }
+ EndUndo( UNDO_END, NULL );
+ SetModified();
+
+ bRet = TRUE; // #106897#
+ }
+ }
+
+ return bRet;
+}
+
+// --> OD 2008-03-18 #refactorlists#
+namespace
+{
+ struct ListStyleData
+ {
+ SwNumRule* pReplaceNumRule;
+ bool bCreateNewList;
+ String sListId;
+
+ ListStyleData()
+ : pReplaceNumRule( 0 ),
+ bCreateNewList( false ),
+ sListId()
+ {}
+ };
+}
+// <--
+
+void SwDoc::MakeUniqueNumRules(const SwPaM & rPaM)
+{
+ ASSERT( rPaM.GetDoc() == this, "need same doc" );
+
+ // --> OD 2008-03-18 #refactorlists#
+// map<SwNumRule *, SwNumRule *> aMyNumRuleMap;
+ ::std::map<SwNumRule *, ListStyleData> aMyNumRuleMap;
+ // <--
+
+ ULONG nStt = rPaM.Start()->nNode.GetIndex();
+ ULONG nEnd = rPaM.End()->nNode.GetIndex();
+
+ bool bFirst = true;
+
+ for (ULONG n = nStt; n <= nEnd; n++)
+ {
+ SwTxtNode * pCNd = GetNodes()[n]->GetTxtNode();
+
+ if (pCNd)
+ {
+ SwNumRule * pRule = pCNd->GetNumRule();
+
+ if (pRule && pRule->IsAutoRule() && ! pRule->IsOutlineRule())
+ {
+ // --> OD 2008-03-18 #refactorlists#
+// SwNumRule * pReplaceNumRule = aMyNumRuleMap[pRule];
+ ListStyleData aListStyleData = aMyNumRuleMap[pRule];
+
+// if (! pReplaceNumRule)
+ if ( aListStyleData.pReplaceNumRule == 0 )
+ {
+ if (bFirst)
+ {
+ SwPosition aPos(*pCNd);
+ aListStyleData.pReplaceNumRule =
+ const_cast<SwNumRule *>
+ (SearchNumRule( aPos, false, pCNd->HasNumber(),
+ false, 0,
+ aListStyleData.sListId, true ));
+ }
+
+// if (! pReplaceNumRule)
+ if ( aListStyleData.pReplaceNumRule == 0 )
+ {
+// pReplaceNumRule = new SwNumRule(*pRule);
+// pReplaceNumRule->SetName(GetUniqueNumRuleName());
+ aListStyleData.pReplaceNumRule = new SwNumRule(*pRule);
+ // --> OD 2008-07-08 #i91400#
+ aListStyleData.pReplaceNumRule->SetName(
+ GetUniqueNumRuleName(), *this );
+ // <--
+ aListStyleData.bCreateNewList = true;
+ }
+
+// aMyNumRuleMap[pRule] = pReplaceNumRule;
+ aMyNumRuleMap[pRule] = aListStyleData;
+ }
+
+ SwPaM aPam(*pCNd);
+
+ SetNumRule( aPam, *aListStyleData.pReplaceNumRule,
+ aListStyleData.bCreateNewList,
+ aListStyleData.sListId );
+ if ( aListStyleData.bCreateNewList )
+ {
+ aListStyleData.bCreateNewList = false;
+ aListStyleData.sListId = pCNd->GetListId();
+ aMyNumRuleMap[pRule] = aListStyleData;
+ }
+ // <--
+
+ bFirst = false;
+ }
+ }
+ }
+}
+
+BOOL SwDoc::NoNum( const SwPaM& rPam )
+{
+
+ BOOL bRet = SplitNode( *rPam.GetPoint(), false );
+ // ist ueberhaupt Nummerierung im Spiel ?
+ if( bRet )
+ {
+ // NoNum setzen und Upaten
+ const SwNodeIndex& rIdx = rPam.GetPoint()->nNode;
+ SwTxtNode* pNd = rIdx.GetNode().GetTxtNode();
+ const SwNumRule* pRule = pNd->GetNumRule();
+ if( pRule )
+ {
+ pNd->SetCountedInList(false);
+
+ SetModified();
+ }
+ else
+ bRet = FALSE; // keine Nummerierung , ?? oder immer TRUE ??
+ }
+ return bRet;
+}
+
+void SwDoc::DelNumRules( const SwPaM& rPam )
+{
+ ULONG nStt = rPam.GetPoint()->nNode.GetIndex(),
+ nEnd = rPam.GetMark()->nNode.GetIndex();
+ if( nStt > nEnd )
+ {
+ ULONG nTmp = nStt; nStt = nEnd; nEnd = nTmp;
+ }
+
+ SwUndoDelNum* pUndo;
+ if( DoesUndo() )
+ {
+ ClearRedo();
+ AppendUndo( pUndo = new SwUndoDelNum( rPam ) );
+ }
+ else
+ pUndo = 0;
+
+ SwRegHistory aRegH( pUndo ? pUndo->GetHistory() : 0 );
+
+ SwNumRuleItem aEmptyRule( aEmptyStr );
+ const SwNode* pOutlNd = 0;
+ for( ; nStt <= nEnd; ++nStt )
+ {
+ SwTxtNode* pTNd = GetNodes()[ nStt ]->GetTxtNode();
+ // --> OD 2008-03-13 #refactorlists#
+// if( pTNd && 0 != ( pItem = pTNd->GetNoCondAttr(
+// RES_PARATR_NUMRULE, TRUE ) ) &&
+// ( pName = &((SwNumRuleItem*)pItem)->GetValue())->Len() )
+ SwNumRule* pNumRuleOfTxtNode = pTNd ? pTNd->GetNumRule() : 0;
+ if ( pTNd && pNumRuleOfTxtNode )
+ // <--
+ {
+ // recognize changes of attribute for undo
+ aRegH.RegisterInModify( pTNd, *pTNd );
+
+ if( pUndo )
+ pUndo->AddNode( *pTNd, FALSE );
+
+ // directly set list style attribute is reset, otherwise empty
+ // list style is applied
+ const SfxItemSet* pAttrSet = pTNd->GetpSwAttrSet();
+ if ( pAttrSet &&
+ pAttrSet->GetItemState( RES_PARATR_NUMRULE, FALSE ) == SFX_ITEM_SET )
+ pTNd->ResetAttr( RES_PARATR_NUMRULE );
+ else
+ pTNd->SetAttr( aEmptyRule );
+
+ // --> OD 2008-03-26 #refactorlists#
+ pTNd->ResetAttr( RES_PARATR_LIST_ID );
+ pTNd->ResetAttr( RES_PARATR_LIST_LEVEL );
+ pTNd->ResetAttr( RES_PARATR_LIST_ISRESTART );
+ pTNd->ResetAttr( RES_PARATR_LIST_RESTARTVALUE );
+ pTNd->ResetAttr( RES_PARATR_LIST_ISCOUNTED );
+ // <--
+
+ if( RES_CONDTXTFMTCOLL == pTNd->GetFmtColl()->Which() )
+ pTNd->ChkCondColl();
+ //else if( !pOutlNd && NO_NUMBERING != //#outline level,zhaojianwei
+ // ((SwTxtFmtColl*)pTNd->GetFmtColl())->GetOutlineLevel() )
+ else if( !pOutlNd &&
+ ((SwTxtFmtColl*)pTNd->GetFmtColl())->IsAssignedToListLevelOfOutlineStyle() )//<-end,zhaojianwei
+ pOutlNd = pTNd;
+ }
+ }
+
+ // dann noch alle Updaten
+ UpdateNumRule();
+
+ if( pOutlNd )
+ GetNodes().UpdtOutlineIdx( *pOutlNd );
+}
+
+void SwDoc::InvalidateNumRules()
+{
+ for (USHORT n = 0; n < pNumRuleTbl->Count(); ++n)
+ (*pNumRuleTbl)[n]->SetInvalidRule(TRUE);
+}
+
+ // zum naechsten/vorhergehenden Punkt auf gleicher Ebene
+
+BOOL lcl_IsNumOk( BYTE nSrchNum, BYTE& rLower, BYTE& rUpper,
+ BOOL bOverUpper, BYTE nNumber )
+{
+ // --> OD 2008-04-02 #refactorlists#
+ ASSERT( nNumber < MAXLEVEL,
+ "<lcl_IsNumOk(..)> - misusage of method" );
+ // <--
+
+ BOOL bRet = FALSE;
+ {
+ if( bOverUpper ? nSrchNum == nNumber : nSrchNum >= nNumber )
+ bRet = TRUE;
+ else if( nNumber > rLower )
+ rLower = nNumber;
+ else if( nNumber < rUpper )
+ rUpper = nNumber;
+ }
+ return bRet;
+}
+
+BOOL lcl_IsValidPrevNextNumNode( const SwNodeIndex& rIdx )
+{
+ BOOL bRet = FALSE;
+ const SwNode& rNd = rIdx.GetNode();
+ switch( rNd.GetNodeType() )
+ {
+ case ND_ENDNODE:
+ bRet = SwTableBoxStartNode == rNd.StartOfSectionNode()->GetStartNodeType() ||
+ rNd.StartOfSectionNode()->IsSectionNode();
+ break;
+
+ case ND_STARTNODE:
+ bRet = SwTableBoxStartNode == ((SwStartNode&)rNd).GetStartNodeType();
+ break;
+
+ case ND_SECTIONNODE: // der ist erlaubt, also weiter
+ bRet = TRUE;
+ break;
+ }
+ return bRet;
+}
+
+BOOL lcl_GotoNextPrevNum( SwPosition& rPos, BOOL bNext,
+ BOOL bOverUpper, BYTE* pUpper, BYTE* pLower )
+{
+ const SwTxtNode* pNd = rPos.nNode.GetNode().GetTxtNode();
+ const SwNumRule* pRule;
+ if( !pNd || 0 == ( pRule = pNd->GetNumRule()))
+ return FALSE;
+
+ BYTE nSrchNum = static_cast<BYTE>(pNd->GetActualListLevel());
+
+ SwNodeIndex aIdx( rPos.nNode );
+ if( ! pNd->IsCountedInList() )
+ {
+ // falls gerade mal NO_NUMLEVEL an ist, so such den vorherigen Node
+ // mit Nummerierung
+ BOOL bError = FALSE;
+ do {
+ aIdx--;
+ if( aIdx.GetNode().IsTxtNode() )
+ {
+ pNd = aIdx.GetNode().GetTxtNode();
+ pRule = pNd->GetNumRule();
+
+ BYTE nTmpNum;
+
+ if( pRule )
+ {
+ nTmpNum = static_cast<BYTE>(pNd->GetActualListLevel());
+ if( !( ! pNd->IsCountedInList() &&
+ (nTmpNum >= nSrchNum )) )
+ break; // gefunden
+ }
+ else
+ bError = TRUE;
+ }
+ else
+ bError = !lcl_IsValidPrevNextNumNode( aIdx );
+
+ } while( !bError );
+ if( bError )
+ return FALSE;
+ }
+
+ BYTE nLower = nSrchNum, nUpper = nSrchNum;
+ BOOL bRet = FALSE;
+
+ const SwTxtNode* pLast;
+ if( bNext )
+ aIdx++, pLast = pNd;
+ else
+ aIdx--, pLast = 0;
+
+ while( bNext ? ( aIdx.GetIndex() < aIdx.GetNodes().Count() - 1 )
+ : aIdx.GetIndex() )
+ {
+ if( aIdx.GetNode().IsTxtNode() )
+ {
+ pNd = aIdx.GetNode().GetTxtNode();
+ pRule = pNd->GetNumRule();
+ if( pRule )
+ {
+ if( ::lcl_IsNumOk( nSrchNum, nLower, nUpper, bOverUpper,
+ static_cast<BYTE>(pNd->GetActualListLevel()) ))
+ {
+ rPos.nNode = aIdx;
+ rPos.nContent.Assign( (SwTxtNode*)pNd, 0 );
+ bRet = TRUE;
+ break;
+ }
+ else
+ pLast = pNd;
+ }
+ else
+ break;
+ }
+ else if( !lcl_IsValidPrevNextNumNode( aIdx ))
+ break;
+
+ if( bNext )
+ aIdx++;
+ else
+ aIdx--;
+ }
+
+ if( !bRet && !bOverUpper && pLast ) // nicht ueber hoehere Nummmern, aber bis Ende
+ {
+ if( bNext )
+ {
+ rPos.nNode = aIdx;
+ if( aIdx.GetNode().IsCntntNode() )
+ rPos.nContent.Assign( aIdx.GetNode().GetCntntNode(), 0 );
+ }
+ else
+ {
+ rPos.nNode.Assign( *pLast );
+ rPos.nContent.Assign( (SwTxtNode*)pLast, 0 );
+ }
+ bRet = TRUE;
+ }
+
+ if( bRet )
+ {
+ if( pUpper )
+ *pUpper = nUpper;
+ if( pLower )
+ *pLower = nLower;
+ }
+ return bRet;
+}
+
+BOOL SwDoc::GotoNextNum( SwPosition& rPos, BOOL bOverUpper,
+ BYTE* pUpper, BYTE* pLower )
+{
+ return ::lcl_GotoNextPrevNum( rPos, TRUE, bOverUpper, pUpper, pLower );
+}
+
+// -> #i23731#
+// --> OD 2008-03-18 #refactorlists# - add output parameter <sListId>
+const SwNumRule * SwDoc::SearchNumRule(const SwPosition & rPos,
+ const bool bForward,
+ const bool bNum,
+ const bool bOutline,
+ int nNonEmptyAllowed,
+ String& sListId,
+ const bool bInvestigateStartNode)
+{
+ const SwNumRule * pResult = NULL;
+ SwTxtNode * pTxtNd = rPos.nNode.GetNode().GetTxtNode();
+ SwNode * pStartFromNode = pTxtNd;
+
+ if (pTxtNd)
+ {
+ SwNodeIndex aIdx(rPos.nNode);
+
+ // --> OD 2005-10-20 #i55391#
+ // - the start node has also been investigated, if requested.
+ const SwNode * pNode = NULL;
+ do
+ {
+ // --> OD 2005-10-20 #i55391#
+ if ( !bInvestigateStartNode )
+ {
+ if (bForward)
+ aIdx++;
+ else
+ aIdx--;
+ }
+ // <--
+ if (aIdx.GetNode().IsTxtNode())
+ {
+ pTxtNd = aIdx.GetNode().GetTxtNode();
+
+ const SwNumRule * pNumRule = pTxtNd->GetNumRule();
+ if (pNumRule)
+ {
+ if ( ( pNumRule->IsOutlineRule() == ( bOutline ? TRUE : FALSE ) ) && // #115901#
+ ( ( bNum && pNumRule->Get(0).IsEnumeration()) ||
+ ( !bNum && pNumRule->Get(0).IsItemize() ) ) ) // #i22362#, #i29560#
+ {
+ pResult = pTxtNd->GetNumRule();
+ // --> OD 2008-03-18 #refactorlists#
+ // provide also the list id, to which the text node belongs.
+ sListId = pTxtNd->GetListId();
+ }
+
+ break;
+ }
+ else if (pTxtNd->Len() > 0 || NULL != pTxtNd->GetNumRule())
+ {
+ if (nNonEmptyAllowed == 0)
+ break;
+
+ nNonEmptyAllowed--;
+
+ if (nNonEmptyAllowed < 0)
+ nNonEmptyAllowed = -1;
+ }
+ }
+
+ // --> OD 2005-10-20 #i55391#
+ if ( bInvestigateStartNode )
+ {
+ if (bForward)
+ aIdx++;
+ else
+ aIdx--;
+ }
+ // <--
+
+ pNode = &aIdx.GetNode();
+ }
+ while (! (pNode == aNodes.DocumentSectionStartNode(pStartFromNode) ||
+ pNode == aNodes.DocumentSectionEndNode(pStartFromNode)));
+ // <--
+ }
+
+ return pResult;
+}
+// <- #i23731#
+
+BOOL SwDoc::GotoPrevNum( SwPosition& rPos, BOOL bOverUpper,
+ BYTE* pUpper, BYTE* pLower )
+{
+ return ::lcl_GotoNextPrevNum( rPos, FALSE, bOverUpper, pUpper, pLower );
+}
+
+BOOL SwDoc::NumUpDown( const SwPaM& rPam, BOOL bDown )
+{
+ ULONG nStt = rPam.GetPoint()->nNode.GetIndex(),
+ nEnd = rPam.GetMark()->nNode.GetIndex();
+ if( nStt > nEnd )
+ {
+ ULONG nTmp = nStt; nStt = nEnd; nEnd = nTmp;
+ }
+
+ // -> #115901# outline nodes are promoted or demoted differently
+ bool bOnlyOutline = true;
+ bool bOnlyNonOutline = true;
+ for (ULONG n = nStt; n <= nEnd; n++)
+ {
+ SwTxtNode * pTxtNd = GetNodes()[n]->GetTxtNode();
+
+ if (pTxtNd)
+ {
+ SwNumRule * pRule = pTxtNd->GetNumRule();
+
+ if (pRule)
+ {
+ if (pRule->IsOutlineRule())
+ bOnlyNonOutline = false;
+ else
+ bOnlyOutline = false;
+ }
+ }
+ }
+ // <- #115901#
+
+ BOOL bRet = TRUE;
+ char nDiff = bDown ? 1 : -1;
+
+ // ->#115901#
+ if (bOnlyOutline)
+ bRet = OutlineUpDown(rPam, nDiff);
+ else if (bOnlyNonOutline)
+ {
+ /* --> #i24560#
+
+ Only promote or demote if all selected paragraphs are
+ promotable resp. demotable.
+
+ */
+ for (ULONG nTmp = nStt; nTmp <= nEnd; ++nTmp)
+ {
+ SwTxtNode* pTNd = GetNodes()[ nTmp ]->GetTxtNode();
+
+ // --> OD 2006-10-19 #134160# - make code robust:
+ // consider case that the node doesn't denote a text node.
+ if ( pTNd )
+ {
+ SwNumRule * pRule = pTNd->GetNumRule();
+
+ if (pRule)
+ {
+ BYTE nLevel = static_cast<BYTE>(pTNd->GetActualListLevel());
+ if( (-1 == nDiff && 0 >= nLevel) ||
+ (1 == nDiff && MAXLEVEL - 1 <= nLevel))
+ bRet = FALSE;
+ }
+ }
+ // <--
+ }
+
+ if( bRet )
+ {
+ /* <-- #i24560# */
+ if( DoesUndo() )
+ {
+ ClearRedo();
+ AppendUndo( new SwUndoNumUpDown( rPam, nDiff ) );
+ }
+
+ String sNumRule;
+
+ for(ULONG nTmp = nStt; nTmp <= nEnd; ++nTmp )
+ {
+ SwTxtNode* pTNd = GetNodes()[ nTmp ]->GetTxtNode();
+
+ if( pTNd)
+ {
+ SwNumRule * pRule = pTNd->GetNumRule();
+
+ if (pRule)
+ {
+ BYTE nLevel = static_cast<BYTE>(pTNd->GetActualListLevel());
+ nLevel = nLevel + nDiff;
+
+ pTNd->SetAttrListLevel(nLevel);
+ }
+ }
+ }
+
+ ChkCondColls();
+ SetModified();
+ }
+ }
+
+ return bRet;
+}
+
+BOOL SwDoc::MoveParagraph( const SwPaM& rPam, long nOffset, BOOL bIsOutlMv )
+{
+ const SwPosition *pStt = rPam.Start(), *pEnd = rPam.End();
+
+ ULONG nStIdx = pStt->nNode.GetIndex();
+ ULONG nEndIdx = pEnd->nNode.GetIndex();
+
+ // Here are some sophisticated checks whether the wished PaM will be moved or not.
+ // For moving outlines (bIsOutlMv) I've already done some checks, so here are two different
+ // checks...
+ SwNode *pTmp1;
+ SwNode *pTmp2;
+ if( bIsOutlMv )
+ {
+ // For moving chapters (outline) the following reason will deny the move:
+ // if a start node is inside the moved area and its end node outside or vice versa.
+ // If a start node is the first moved paragraph, its end node has to be within the moved
+ // area, too (e.g. as last node).
+ // If an end node is the last node of the moved area, its start node has to be a part of
+ // the moved section, too.
+ pTmp1 = GetNodes()[ nStIdx ];
+ if( pTmp1->IsStartNode() )
+ { // First is a start node
+ pTmp2 = pTmp1->EndOfSectionNode();
+ if( pTmp2->GetIndex() > nEndIdx )
+ return FALSE; // Its end node is behind the moved range
+ }
+ pTmp1 = pTmp1->StartOfSectionNode()->EndOfSectionNode();
+ if( pTmp1->GetIndex() <= nEndIdx )
+ return FALSE; // End node inside but start node before moved range => no.
+ pTmp1 = GetNodes()[ nEndIdx ];
+ if( pTmp1->IsEndNode() )
+ { // The last one is an end node
+ pTmp1 = pTmp1->StartOfSectionNode();
+ if( pTmp1->GetIndex() < nStIdx )
+ return FALSE; // Its start node is before the moved range.
+ }
+ pTmp1 = pTmp1->StartOfSectionNode();
+ if( pTmp1->GetIndex() >= nStIdx )
+ return FALSE; // A start node which ends behind the moved area => no.
+ }
+
+ ULONG nInStIdx, nInEndIdx;
+ long nOffs = nOffset;
+ if( nOffset > 0 )
+ {
+ nInEndIdx = nEndIdx;
+ nEndIdx += nOffset;
+ ++nOffs;
+ }
+ else
+ {
+ //Impossible to move to negative index
+ if( ULONG(abs( nOffset )) > nStIdx)
+ return FALSE;
+
+ nInEndIdx = nStIdx - 1;
+ nStIdx += nOffset;
+ }
+ nInStIdx = nInEndIdx + 1;
+ // Folgende Absatzbloecke sollen vertauscht werden:
+ // [ nStIdx, nInEndIdx ] mit [ nInStIdx, nEndIdx ]
+
+ if( nEndIdx >= GetNodes().GetEndOfContent().GetIndex() )
+ return FALSE;
+
+ if( !bIsOutlMv )
+ { // And here the restrictions for moving paragraphs other than chapters (outlines)
+ // The plan is to exchange [nStIdx,nInEndIdx] and [nStartIdx,nEndIdx]
+ // It will checked if the both "start" nodes as well as the both "end" notes belongs to
+ // the same start-end-section. This is more restrictive than the conditions checked above.
+ // E.g. a paragraph will not escape from a section or be inserted to another section.
+ pTmp1 = GetNodes()[ nStIdx ]->StartOfSectionNode();
+ pTmp2 = GetNodes()[ nInStIdx ]->StartOfSectionNode();
+ if( pTmp1 != pTmp2 )
+ return FALSE; // "start" nodes in different sections
+ pTmp1 = GetNodes()[ nEndIdx ];
+ bool bIsEndNode = pTmp1->IsEndNode();
+ if( !pTmp1->IsStartNode() )
+ {
+ pTmp1 = pTmp1->StartOfSectionNode();
+ if( bIsEndNode ) // For end nodes the first start node is of course inside the range,
+ pTmp1 = pTmp1->StartOfSectionNode(); // I've to check the start node of the start node.
+ }
+ pTmp1 = pTmp1->EndOfSectionNode();
+ pTmp2 = GetNodes()[ nInEndIdx ];
+ if( !pTmp2->IsStartNode() )
+ {
+ bIsEndNode = pTmp2->IsEndNode();
+ pTmp2 = pTmp2->StartOfSectionNode();
+ if( bIsEndNode )
+ pTmp2 = pTmp2->StartOfSectionNode();
+ }
+ pTmp2 = pTmp2->EndOfSectionNode();
+ if( pTmp1 != pTmp2 )
+ return FALSE; // The "end" notes are in different sections
+ }
+
+ // auf Redlining testen - darf die Selektion ueberhaupt verschoben
+ // werden?
+ if( !IsIgnoreRedline() )
+ {
+ USHORT nRedlPos = GetRedlinePos( pStt->nNode.GetNode(), nsRedlineType_t::REDLINE_DELETE );
+ if( USHRT_MAX != nRedlPos )
+ {
+ SwPosition aStPos( *pStt ), aEndPos( *pEnd );
+ aStPos.nContent = 0;
+ SwCntntNode* pCNd = pEnd->nNode.GetNode().GetCntntNode();
+ aEndPos.nContent = pCNd ? pCNd->Len() : 1;
+ BOOL bCheckDel = TRUE;
+
+ // es existiert fuer den Bereich irgendein Redline-Delete-Object
+ for( ; nRedlPos < GetRedlineTbl().Count(); ++nRedlPos )
+ {
+ const SwRedline* pTmp = GetRedlineTbl()[ nRedlPos ];
+ if( !bCheckDel || nsRedlineType_t::REDLINE_DELETE == pTmp->GetType() )
+ {
+ const SwPosition *pRStt = pTmp->Start(), *pREnd = pTmp->End();
+ switch( ComparePosition( *pRStt, *pREnd, aStPos, aEndPos ))
+ {
+ case POS_COLLIDE_START:
+ case POS_BEHIND: // Pos1 liegt hinter Pos2
+ nRedlPos = GetRedlineTbl().Count();
+ break;
+
+ case POS_COLLIDE_END:
+ case POS_BEFORE: // Pos1 liegt vor Pos2
+ break;
+ case POS_INSIDE: // Pos1 liegt vollstaendig in Pos2
+ // ist erlaubt, aber checke dann alle nachfolgenden
+ // auf Ueberlappungen
+ bCheckDel = FALSE;
+ break;
+
+ case POS_OUTSIDE: // Pos2 liegt vollstaendig in Pos1
+ case POS_EQUAL: // Pos1 ist genauso gross wie Pos2
+ case POS_OVERLAP_BEFORE: // Pos1 ueberlappt Pos2 am Anfang
+ case POS_OVERLAP_BEHIND: // Pos1 ueberlappt Pos2 am Ende
+ return FALSE;
+ }
+ }
+ }
+ }
+ }
+
+ {
+ // DataChanged vorm verschieben verschicken, dann bekommt
+ // man noch mit, welche Objecte sich im Bereich befinden.
+ // Danach koennen sie vor/hinter der Position befinden.
+ SwDataChanged aTmp( rPam, 0 );
+ }
+
+ SwNodeIndex aIdx( nOffset > 0 ? pEnd->nNode : pStt->nNode, nOffs );
+ SwNodeRange aMvRg( pStt->nNode, 0, pEnd->nNode, +1 );
+
+ SwRedline* pOwnRedl = 0;
+ if( IsRedlineOn() )
+ {
+ // wenn der Bereich komplett im eigenen Redline liegt, kann es
+ // verschoben werden!
+ USHORT nRedlPos = GetRedlinePos( pStt->nNode.GetNode(), nsRedlineType_t::REDLINE_INSERT );
+ if( USHRT_MAX != nRedlPos )
+ {
+ SwRedline* pTmp = GetRedlineTbl()[ nRedlPos ];
+ const SwPosition *pRStt = pTmp->Start(), *pREnd = pTmp->End();
+ SwRedline aTmpRedl( nsRedlineType_t::REDLINE_INSERT, rPam );
+ const SwCntntNode* pCEndNd = pEnd->nNode.GetNode().GetCntntNode();
+ // liegt komplett im Bereich, und ist auch der eigene Redline?
+ if( aTmpRedl.IsOwnRedline( *pTmp ) &&
+ (pRStt->nNode < pStt->nNode ||
+ (pRStt->nNode == pStt->nNode && !pRStt->nContent.GetIndex()) ) &&
+ (pEnd->nNode < pREnd->nNode ||
+ (pEnd->nNode == pREnd->nNode &&
+ pCEndNd ? pREnd->nContent.GetIndex() == pCEndNd->Len()
+ : !pREnd->nContent.GetIndex() )) )
+ {
+ pOwnRedl = pTmp;
+ if( nRedlPos + 1 < GetRedlineTbl().Count() )
+ {
+ pTmp = GetRedlineTbl()[ nRedlPos+1 ];
+ if( *pTmp->Start() == *pREnd )
+ // dann doch nicht!
+ pOwnRedl = 0;
+ }
+
+ if( pOwnRedl &&
+ !( pRStt->nNode <= aIdx && aIdx <= pREnd->nNode ))
+ {
+ // nicht in sich selbst, dann auch nicht moven
+ pOwnRedl = 0;
+ }
+ }
+ }
+
+ if( !pOwnRedl )
+ {
+ StartUndo( UNDO_START, NULL );
+
+ // zuerst das Insert, dann das Loeschen
+ SwPosition aInsPos( aIdx );
+ aInsPos.nContent.Assign( aIdx.GetNode().GetCntntNode(), 0 );
+
+ SwPaM aPam( pStt->nNode, aMvRg.aEnd );
+
+ SwPaM& rOrigPam = (SwPaM&)rPam;
+ rOrigPam.DeleteMark();
+ rOrigPam.GetPoint()->nNode = aIdx.GetIndex() - 1;
+
+ BOOL bDelLastPara = !aInsPos.nNode.GetNode().IsCntntNode();
+
+ /* #101076# When copying to a non-content node Copy will
+ insert a paragraph before that node and insert before
+ that inserted node. Copy creates an SwUndoInserts that
+ does not cover the extra paragraph. Thus we insert the
+ extra paragraph ourselves, _with_ correct undo
+ information. */
+ if (bDelLastPara)
+ {
+ /* aInsPos points to the non-content node. Move it to
+ the previous content node. */
+ SwPaM aInsPam(aInsPos);
+ BOOL bMoved = aInsPam.Move(fnMoveBackward);
+ ASSERT(bMoved, "No content node found!");
+
+ if (bMoved)
+ {
+ /* Append the new node after the content node
+ found. The new position to insert the moved
+ paragraph at is before the inserted
+ paragraph. */
+ AppendTxtNode(*aInsPam.GetPoint());
+ aInsPos = *aInsPam.GetPoint();
+ }
+ }
+
+ CopyRange( aPam, aInsPos, false );
+ if( bDelLastPara )
+ {
+ // dann muss der letzte leere Node wieder entfernt werden
+ aIdx = aInsPos.nNode;
+ SwCntntNode* pCNd = GetNodes().GoPrevious( &aInsPos.nNode );
+ xub_StrLen nCLen = 0; if( pCNd ) nCLen = pCNd->Len();
+ aInsPos.nContent.Assign( pCNd, nCLen );
+
+ // alle die im zu loeschenden Node stehen, mussen auf den
+ // naechsten umgestezt werden
+ SwPosition* pPos;
+ for( USHORT n = 0; n < GetRedlineTbl().Count(); ++n )
+ {
+ SwRedline* pTmp = GetRedlineTbl()[ n ];
+ if( ( pPos = &pTmp->GetBound(TRUE))->nNode == aIdx )
+ {
+ pPos->nNode++;
+ pPos->nContent.Assign( pPos->nNode.GetNode().GetCntntNode(),0);
+ }
+ if( ( pPos = &pTmp->GetBound(FALSE))->nNode == aIdx )
+ {
+ pPos->nNode++;
+ pPos->nContent.Assign( pPos->nNode.GetNode().GetCntntNode(),0);
+ }
+ }
+ CorrRel( aIdx, aInsPos, 0, FALSE );
+
+ pCNd->JoinNext();
+ }
+
+ rOrigPam.GetPoint()->nNode++;
+ rOrigPam.GetPoint()->nContent.Assign( rOrigPam.GetCntntNode(), 0 );
+
+ RedlineMode_t eOld = GetRedlineMode();
+ checkRedlining(eOld);
+ if( DoesUndo() )
+ {
+ //JP 06.01.98: MUSS noch optimiert werden!!!
+ SetRedlineMode(
+ (RedlineMode_t)(nsRedlineMode_t::REDLINE_ON | nsRedlineMode_t::REDLINE_SHOW_INSERT | nsRedlineMode_t::REDLINE_SHOW_DELETE));
+ AppendUndo( new SwUndoRedlineDelete( aPam, UNDO_DELETE ));
+ }
+
+ SwRedline* pNewRedline = new SwRedline( nsRedlineType_t::REDLINE_DELETE, aPam );
+
+ // #101654# prevent assertion from aPam's target being deleted
+ // (Alternatively, one could just let aPam go out of scope, but
+ // that requires touching a lot of code.)
+ aPam.GetBound(TRUE).nContent.Assign( NULL, 0 );
+ aPam.GetBound(FALSE).nContent.Assign( NULL, 0 );
+
+ AppendRedline( pNewRedline, true );
+
+//JP 06.01.98: MUSS noch optimiert werden!!!
+SetRedlineMode( eOld );
+ EndUndo( UNDO_END, NULL );
+ SetModified();
+
+ return TRUE;
+ }
+ }
+
+ if( !pOwnRedl && !IsIgnoreRedline() && GetRedlineTbl().Count() )
+ {
+ SwPaM aTemp(aIdx);
+ SplitRedline(aTemp);
+ }
+
+ ULONG nRedlSttNd(0), nRedlEndNd(0);
+ if( pOwnRedl )
+ {
+ const SwPosition *pRStt = pOwnRedl->Start(), *pREnd = pOwnRedl->End();
+ nRedlSttNd = pRStt->nNode.GetIndex();
+ nRedlEndNd = pREnd->nNode.GetIndex();
+ }
+
+ SwUndoMoveNum* pUndo = 0;
+ ULONG nMoved = 0;
+ if( DoesUndo() )
+ {
+ pUndo = new SwUndoMoveNum( rPam, nOffset, bIsOutlMv );
+ nMoved = rPam.End()->nNode.GetIndex() - rPam.Start()->nNode.GetIndex() + 1;
+ }
+
+
+ MoveNodeRange( aMvRg, aIdx, DOC_MOVEREDLINES );
+
+ if( pUndo )
+ {
+ ClearRedo();
+ // i57907: Under circumstances (sections at the end of a chapter)
+ // the rPam.Start() is not moved to the new position.
+ // But aIdx should be at the new end position and as long as the number of moved paragraphs
+ // is nMoved, I know, where the new position is.
+ pUndo->SetStartNode( aIdx.GetIndex() - nMoved );
+ AppendUndo( pUndo );
+ }
+
+ if( pOwnRedl )
+ {
+ SwPosition *pRStt = pOwnRedl->Start(), *pREnd = pOwnRedl->End();
+ if( pRStt->nNode.GetIndex() != nRedlSttNd )
+ {
+ pRStt->nNode = nRedlSttNd;
+ pRStt->nContent.Assign( pRStt->nNode.GetNode().GetCntntNode(),0);
+ }
+ if( pREnd->nNode.GetIndex() != nRedlEndNd )
+ {
+ pREnd->nNode = nRedlEndNd;
+ SwCntntNode* pCNd = pREnd->nNode.GetNode().GetCntntNode();
+ xub_StrLen nL = 0; if( pCNd ) nL = pCNd->Len();
+ pREnd->nContent.Assign( pCNd, nL );
+ }
+ }
+
+ SetModified();
+ return TRUE;
+}
+
+BOOL SwDoc::NumOrNoNum( const SwNodeIndex& rIdx, BOOL bDel )
+{
+ BOOL bResult = FALSE;
+ SwTxtNode * pTxtNd = rIdx.GetNode().GetTxtNode();
+
+ if (pTxtNd && pTxtNd->GetNumRule() != NULL &&
+ (pTxtNd->HasNumber() || pTxtNd->HasBullet()))
+ {
+ if ( !pTxtNd->IsCountedInList() == !bDel)
+ {
+ BOOL bOldNum = bDel; // == pTxtNd->IsCounted();
+ BOOL bNewNum = bDel ? FALSE : TRUE;
+ pTxtNd->SetCountedInList(bNewNum ? true : false);
+
+ SetModified();
+
+ bResult = TRUE;
+
+ if (DoesUndo())
+ {
+ SwUndoNumOrNoNum * pUndo =
+ new SwUndoNumOrNoNum(rIdx, bOldNum, bNewNum);
+
+ AppendUndo(pUndo);
+ }
+ }
+ else if (bDel && pTxtNd->GetNumRule(FALSE) &&
+ pTxtNd->GetActualListLevel() >= 0 &&
+ pTxtNd->GetActualListLevel() < MAXLEVEL)
+ {
+ SwPaM aPam(*pTxtNd);
+
+ DelNumRules(aPam);
+
+ bResult = TRUE;
+ }
+ }
+
+ return bResult;
+}
+
+SwNumRule* SwDoc::GetCurrNumRule( const SwPosition& rPos ) const
+{
+ SwNumRule* pRet = 0;
+ SwTxtNode* pTNd = rPos.nNode.GetNode().GetTxtNode();
+
+ if( pTNd )
+ {
+ // --> OD 2008-02-20 #refactorlists#
+// pTNd->SyncNumberAndNumRule();
+ // <--
+ pRet = pTNd->GetNumRule();
+ }
+
+ return pRet;
+}
+
+USHORT SwDoc::FindNumRule( const String& rName ) const
+{
+ for( USHORT n = pNumRuleTbl->Count(); n; )
+ if( (*pNumRuleTbl)[ --n ]->GetName() == rName )
+ return n;
+
+ return USHRT_MAX;
+}
+
+SwNumRule* SwDoc::FindNumRulePtr( const String& rName ) const
+{
+ SwNumRule * pResult = 0;
+
+ pResult = maNumRuleMap[rName];
+
+ if ( !pResult )
+ {
+ for (USHORT n = 0; n < pNumRuleTbl->Count(); ++n)
+ {
+ if ((*pNumRuleTbl)[n]->GetName() == rName)
+ {
+ pResult = (*pNumRuleTbl)[n];
+
+ break;
+ }
+ }
+ }
+
+ return pResult;
+}
+
+// #i36749#
+void SwDoc::AddNumRule(SwNumRule * pRule)
+{
+ pNumRuleTbl->Insert(pRule, pNumRuleTbl->Count());
+ maNumRuleMap[pRule->GetName()] = pRule;
+ pRule->SetNumRuleMap(&maNumRuleMap);
+
+ // --> OD 2008-03-26 #refactorlists#
+ createListForListStyle( pRule->GetName() );
+ // <--
+}
+
+// --> OD 2008-02-11 #newlistlevelattrs#
+USHORT SwDoc::MakeNumRule( const String &rName,
+ const SwNumRule* pCpy,
+ BOOL bBroadcast,
+ const SvxNumberFormat::SvxNumPositionAndSpaceMode eDefaultNumberFormatPositionAndSpaceMode )
+{
+ SwNumRule* pNew;
+ if( pCpy )
+ {
+ pNew = new SwNumRule( *pCpy );
+
+ // --> OD 2008-07-08 #i91400#
+ pNew->SetName( GetUniqueNumRuleName( &rName ), *this );
+ // <--
+ if( pNew->GetName() != rName )
+ {
+ pNew->SetPoolFmtId( USHRT_MAX );
+ pNew->SetPoolHelpId( USHRT_MAX );
+ pNew->SetPoolHlpFileId( UCHAR_MAX );
+ // --> OD 2008-04-03 #refactorlists#
+ pNew->SetDefaultListId( String() );
+ // <--
+ }
+ pNew->CheckCharFmts( this );
+ }
+ else
+ {
+ // --> OD 2008-02-11 #newlistlevelattrs#
+ pNew = new SwNumRule( GetUniqueNumRuleName( &rName ),
+ eDefaultNumberFormatPositionAndSpaceMode );
+ // <--
+ }
+
+ USHORT nRet = pNumRuleTbl->Count();
+
+ AddNumRule(pNew); // #i36749#
+
+ if (DoesUndo())
+ {
+ SwUndo * pUndo = new SwUndoNumruleCreate(pNew, this);
+
+ AppendUndo(pUndo);
+ }
+
+ if (bBroadcast)
+ BroadcastStyleOperation(pNew->GetName(), SFX_STYLE_FAMILY_PSEUDO,
+ SFX_STYLESHEET_CREATED);
+
+ return nRet;
+}
+
+String SwDoc::GetUniqueNumRuleName( const String* pChkStr, BOOL bAutoNum ) const
+{
+ String aName;
+ if( bAutoNum )
+ {
+ long n = Time().GetTime();
+ n += Date().GetDate();
+ aName = String::CreateFromInt32( n );
+ if( pChkStr && !pChkStr->Len() )
+ pChkStr = 0;
+ }
+ else if( pChkStr && pChkStr->Len() )
+ aName = *pChkStr;
+ else
+ {
+ pChkStr = 0;
+ aName = SW_RESSTR( STR_NUMRULE_DEFNAME );
+ }
+
+ USHORT nNum(0), nTmp, nFlagSize = ( pNumRuleTbl->Count() / 8 ) +2;
+ BYTE* pSetFlags = new BYTE[ nFlagSize ];
+ memset( pSetFlags, 0, nFlagSize );
+
+ xub_StrLen nNmLen = aName.Len();
+ if( !bAutoNum && pChkStr )
+ {
+ while( nNmLen-- && '0' <= aName.GetChar( nNmLen ) &&
+ '9' >= aName.GetChar( nNmLen ) )
+ ; //nop
+
+ if( ++nNmLen < aName.Len() )
+ {
+ aName.Erase( nNmLen );
+ pChkStr = 0;
+ }
+ }
+
+ const SwNumRule* pNumRule;
+ USHORT n;
+
+ for( n = 0; n < pNumRuleTbl->Count(); ++n )
+ if( 0 != ( pNumRule = (*pNumRuleTbl)[ n ] ) )
+ {
+ const String& rNm = pNumRule->GetName();
+ if( rNm.Match( aName ) == nNmLen )
+ {
+ // Nummer bestimmen und das Flag setzen
+ nNum = (USHORT)rNm.Copy( nNmLen ).ToInt32();
+ if( nNum-- && nNum < pNumRuleTbl->Count() )
+ pSetFlags[ nNum / 8 ] |= (0x01 << ( nNum & 0x07 ));
+ }
+ if( pChkStr && pChkStr->Equals( rNm ) )
+ pChkStr = 0;
+ }
+
+ if( !pChkStr )
+ {
+ // alle Nummern entsprechend geflag, also bestimme die richtige Nummer
+ nNum = pNumRuleTbl->Count();
+ for( n = 0; n < nFlagSize; ++n )
+ if( 0xff != ( nTmp = pSetFlags[ n ] ))
+ {
+ // also die Nummer bestimmen
+ nNum = n * 8;
+ while( nTmp & 1 )
+ ++nNum, nTmp >>= 1;
+ break;
+ }
+
+ }
+ delete [] pSetFlags;
+ if( pChkStr && pChkStr->Len() )
+ return *pChkStr;
+ return aName += String::CreateFromInt32( ++nNum );
+}
+
+void SwDoc::UpdateNumRule()
+{
+ const SwNumRuleTbl& rNmTbl = GetNumRuleTbl();
+ for( USHORT n = 0; n < rNmTbl.Count(); ++n )
+ if( rNmTbl[ n ]->IsInvalidRule() )
+ rNmTbl[ n ]->Validate();
+}
+
+// --> OD 2008-04-02 #refactorlists#
+void SwDoc::MarkListLevel( const String& sListId,
+ const int nListLevel,
+ const BOOL bValue )
+{
+ SwList* pList = getListByName( sListId );
+
+ if ( pList )
+ {
+ MarkListLevel( *pList, nListLevel, bValue );
+ }
+}
+
+void SwDoc::MarkListLevel( SwList& rList,
+ const int nListLevel,
+ const BOOL bValue )
+{
+ // Set new marked list level and notify all affected nodes of the changed mark.
+ rList.MarkListLevel( nListLevel, bValue );
+}
+// <- #i27615#
+// <--
+
+// #i23726#
+BOOL SwDoc::IsFirstOfNumRule(SwPosition & rPos)
+{
+ BOOL bResult = FALSE;
+ SwTxtNode * pTxtNode = rPos.nNode.GetNode().GetTxtNode();
+
+ if (pTxtNode)
+ {
+ SwNumRule * pNumRule = pTxtNode->GetNumRule();
+
+ if (pNumRule)
+ bResult = pTxtNode->IsFirstOfNumRule();
+ }
+
+ return bResult;
+}
+
+// --> OD 2007-10-26 #i83479#
+// implementation for interface <IDocumentListItems>
+bool SwDoc::lessThanNodeNum::operator()( const SwNodeNum* pNodeNumOne,
+ const SwNodeNum* pNodeNumTwo ) const
+{
+ return pNodeNumOne->LessThan( *pNodeNumTwo );
+}
+
+void SwDoc::addListItem( const SwNodeNum& rNodeNum )
+{
+ if ( mpListItemsList == 0 )
+ {
+ return;
+ }
+
+ const bool bAlreadyInserted(
+ mpListItemsList->find( &rNodeNum ) != mpListItemsList->end() );
+ ASSERT( !bAlreadyInserted,
+ "<SwDoc::InsertListItem(..)> - <SwNodeNum> instance already registered as numbered item!" );
+ if ( !bAlreadyInserted )
+ {
+ mpListItemsList->insert( &rNodeNum );
+ }
+}
+
+void SwDoc::removeListItem( const SwNodeNum& rNodeNum )
+{
+ if ( mpListItemsList == 0 )
+ {
+ return;
+ }
+
+ const tImplSortedNodeNumList::size_type nDeleted = mpListItemsList->erase( &rNodeNum );
+ if ( nDeleted > 1 )
+ {
+ ASSERT( false,
+ "<SwDoc::RemoveListItem(..)> - <SwNodeNum> was registered more than once as numbered item!" );
+ }
+}
+
+String SwDoc::getListItemText( const SwNodeNum& rNodeNum,
+ const bool bWithNumber,
+ const bool bWithSpacesForLevel ) const
+{
+ return rNodeNum.GetTxtNode()
+ ? rNodeNum.GetTxtNode()->GetExpandTxt( 0, STRING_LEN, bWithNumber,
+ bWithNumber, bWithSpacesForLevel )
+ : String();
+}
+
+void SwDoc::getListItems( tSortedNodeNumList& orNodeNumList ) const
+{
+ orNodeNumList.clear();
+ orNodeNumList.reserve( mpListItemsList->size() );
+
+ tImplSortedNodeNumList::iterator aIter;
+ tImplSortedNodeNumList::iterator aEndIter = mpListItemsList->end();
+ for ( aIter = mpListItemsList->begin(); aIter != aEndIter; ++aIter )
+ {
+ orNodeNumList.push_back( (*aIter) );
+ }
+}
+
+void SwDoc::getNumItems( tSortedNodeNumList& orNodeNumList ) const
+{
+ orNodeNumList.clear();
+ orNodeNumList.reserve( mpListItemsList->size() );
+
+ tImplSortedNodeNumList::iterator aIter;
+ tImplSortedNodeNumList::iterator aEndIter = mpListItemsList->end();
+ for ( aIter = mpListItemsList->begin(); aIter != aEndIter; ++aIter )
+ {
+ const SwNodeNum* pNodeNum = (*aIter);
+ if ( pNodeNum->IsCounted() &&
+ pNodeNum->GetTxtNode() && pNodeNum->GetTxtNode()->HasNumber() )
+ {
+ orNodeNumList.push_back( pNodeNum );
+ }
+ }
+}
+// <--
+
+// --> OD 2007-11-15 #i83479#
+// implementation for interface <IDocumentOutlineNodes>
+sal_Int32 SwDoc::getOutlineNodesCount() const
+{
+ return GetNodes().GetOutLineNds().Count();
+}
+
+int SwDoc::getOutlineLevel( const sal_Int32 nIdx ) const
+{
+ return GetNodes().GetOutLineNds()[ static_cast<USHORT>(nIdx) ]->
+ // GetTxtNode()->GetOutlineLevel(); //#outline level,zhaojianwei
+ GetTxtNode()->GetAttrOutlineLevel()-1; //<-end,zhaojianwei
+}
+
+String SwDoc::getOutlineText( const sal_Int32 nIdx,
+ const bool bWithNumber,
+ const bool bWithSpacesForLevel ) const
+{
+ return GetNodes().GetOutLineNds()[ static_cast<USHORT>(nIdx) ]->
+ GetTxtNode()->GetExpandTxt( 0, STRING_LEN, bWithNumber,
+ bWithNumber, bWithSpacesForLevel );
+}
+
+SwTxtNode* SwDoc::getOutlineNode( const sal_Int32 nIdx ) const
+{
+ return GetNodes().GetOutLineNds()[ static_cast<USHORT>(nIdx) ]->GetTxtNode();
+}
+
+void SwDoc::getOutlineNodes( IDocumentOutlineNodes::tSortedOutlineNodeList& orOutlineNodeList ) const
+{
+ orOutlineNodeList.clear();
+ orOutlineNodeList.reserve( getOutlineNodesCount() );
+
+ const USHORT nOutlCount( static_cast<USHORT>(getOutlineNodesCount()) );
+ for ( USHORT i = 0; i < nOutlCount; ++i )
+ {
+ orOutlineNodeList.push_back(
+ GetNodes().GetOutLineNds()[i]->GetTxtNode() );
+ }
+}
+// <--
+
+// --> OD 2008-03-26 #refactorlists#
+// implementation of interface IDocumentListsAccess
+SwList* SwDoc::createList( String sListId,
+ const String sDefaultListStyleName )
+{
+ if ( sListId.Len() == 0 )
+ {
+ sListId = listfunc::CreateUniqueListId( *this );
+ }
+
+ if ( getListByName( sListId ) )
+ {
+ ASSERT( false,
+ "<SwDoc::createList(..)> - provided list id already used. Serious defect -> please inform OD." );
+ return 0;
+ }
+
+ SwNumRule* pDefaultNumRuleForNewList = FindNumRulePtr( sDefaultListStyleName );
+ if ( !pDefaultNumRuleForNewList )
+ {
+ ASSERT( false,
+ "<SwDoc::createList(..)> - for provided default list style name no list style is found. Serious defect -> please inform OD." );
+ return 0;
+ }
+
+ SwList* pNewList = new SwList( sListId, *pDefaultNumRuleForNewList, GetNodes() );
+ maLists[sListId] = pNewList;
+
+ return pNewList;
+}
+
+void SwDoc::deleteList( const String sListId )
+{
+ SwList* pList = getListByName( sListId );
+ if ( pList )
+ {
+ maLists.erase( sListId );
+ delete pList;
+ }
+}
+
+SwList* SwDoc::getListByName( const String sListId ) const
+{
+ SwList* pList = 0;
+
+ std::hash_map< String, SwList*, StringHash >::const_iterator
+ aListIter = maLists.find( sListId );
+ if ( aListIter != maLists.end() )
+ {
+ pList = (*aListIter).second;
+ }
+
+ return pList;
+}
+
+SwList* SwDoc::createListForListStyle( const String sListStyleName )
+{
+ if ( sListStyleName.Len() == 0 )
+ {
+ ASSERT( false,
+ "<SwDoc::createListForListStyle(..)> - no list style name provided. Serious defect -> please inform OD." );
+ return 0;
+ }
+
+ if ( getListForListStyle( sListStyleName ) )
+ {
+ ASSERT( false,
+ "<SwDoc::createListForListStyle(..)> - a list for the provided list style name already exists. Serious defect -> please inform OD." );
+ return 0;
+ }
+
+ SwNumRule* pNumRule = FindNumRulePtr( sListStyleName );
+ if ( !pNumRule )
+ {
+ ASSERT( false,
+ "<SwDoc::createListForListStyle(..)> - for provided list style name no list style is found. Serious defect -> please inform OD." );
+ return 0;
+ }
+
+ String sListId( pNumRule->GetDefaultListId() ); // can be empty String
+ if ( getListByName( sListId ) )
+ {
+ sListId = String();
+ }
+ SwList* pNewList = createList( sListId, sListStyleName );
+ maListStyleLists[sListStyleName] = pNewList;
+ pNumRule->SetDefaultListId( pNewList->GetListId() );
+
+ return pNewList;
+}
+
+SwList* SwDoc::getListForListStyle( const String sListStyleName ) const
+{
+ SwList* pList = 0;
+
+ std::hash_map< String, SwList*, StringHash >::const_iterator
+ aListIter = maListStyleLists.find( sListStyleName );
+ if ( aListIter != maListStyleLists.end() )
+ {
+ pList = (*aListIter).second;
+ }
+
+ return pList;
+}
+
+void SwDoc::deleteListForListStyle( const String sListStyleName )
+{
+ String sListId;
+ {
+ SwList* pList = getListForListStyle( sListStyleName );
+ ASSERT( pList,
+ "<SwDoc::deleteListForListStyle(..)> - misusage of method: no list found for given list style name" );
+ if ( pList )
+ {
+ sListId = pList->GetListId();
+ }
+ }
+ if ( sListId.Len() > 0 )
+ {
+ maListStyleLists.erase( sListStyleName );
+ deleteList( sListId );
+ }
+}
+// <--
+// --> OD 2008-07-08 #i91400#
+void SwDoc::trackChangeOfListStyleName( const String sListStyleName,
+ const String sNewListStyleName )
+{
+ SwList* pList = getListForListStyle( sListStyleName );
+ ASSERT( pList,
+ "<SwDoc::changeOfListStyleName(..)> - misusage of method: no list found for given list style name" );
+
+ if ( pList != 0 )
+ {
+ maListStyleLists.erase( sListStyleName );
+ maListStyleLists[sNewListStyleName] = pList;
+ }
+}
+// <--
+
+// --> OD 2008-03-13 #refactorlists#
+namespace listfunc
+{
+ const String MakeListIdUnique( const SwDoc& rDoc,
+ const String aSuggestedUniqueListId )
+ {
+ long nHitCount = 0;
+ String aTmpStr = aSuggestedUniqueListId;
+ while ( rDoc.getListByName( aTmpStr ) )
+ {
+ ++nHitCount;
+ aTmpStr = aSuggestedUniqueListId;
+ aTmpStr += String::CreateFromInt32( nHitCount );
+ }
+
+ return aTmpStr;
+ }
+ const String CreateUniqueListId( const SwDoc& rDoc )
+ {
+ // --> OD 2008-08-06 #i92478#
+ String aNewListId = String::CreateFromAscii( "list" );
+ // <--
+ sal_Int64 n = Time().GetTime();
+ n += Date().GetDate();
+ n += rand();
+ // --> OD 2008-08-06 #i92478#
+ aNewListId += String::CreateFromInt64( n );
+ // <--
+
+ return MakeListIdUnique( rDoc, aNewListId );
+ }
+}
+// <--