(cherry picked from commit c2afeb1c3f11e8f420b59f3786eb8626c99ff595) Conflicts: sw/inc/IDocumentContentOperations.hxx sw/inc/crsrsh.hxx sw/inc/doc.hxx sw/inc/editsh.hxx sw/inc/expfld.hxx sw/inc/fldbas.hxx sw/inc/fmtfld.hxx sw/inc/hintids.hxx sw/inc/ndtxt.hxx sw/inc/txatbase.hxx sw/inc/txtfld.hxx sw/inc/txtrfmrk.hxx sw/inc/txttxmrk.hxx sw/inc/viscrs.hxx sw/source/core/bastyp/init.cxx sw/source/core/crsr/crsrsh.cxx sw/source/core/crsr/crstrvl.cxx sw/source/core/crsr/findattr.cxx sw/source/core/crsr/findtxt.cxx sw/source/core/crsr/swcrsr.cxx sw/source/core/crsr/viscrs.cxx sw/source/core/doc/dbgoutsw.cxx sw/source/core/doc/doc.cxx sw/source/core/doc/docfld.cxx sw/source/core/doc/docfmt.cxx sw/source/core/doc/docnum.cxx sw/source/core/doc/docredln.cxx sw/source/core/doc/docruby.cxx sw/source/core/doc/doctxm.cxx sw/source/core/doc/visiturl.cxx sw/source/core/docnode/nodes.cxx sw/source/core/edit/edatmisc.cxx sw/source/core/edit/edattr.cxx sw/source/core/edit/edfld.cxx sw/source/core/edit/edfldexp.cxx sw/source/core/edit/editsh.cxx sw/source/core/edit/edlingu.cxx sw/source/core/edit/ednumber.cxx sw/source/core/fields/expfld.cxx sw/source/core/fields/fldbas.cxx sw/source/core/fields/reffld.cxx sw/source/core/frmedt/fefly1.cxx sw/source/core/inc/docfld.hxx sw/source/core/inc/rolbck.hxx sw/source/core/inc/swfont.hxx sw/source/core/layout/flycnt.cxx sw/source/core/layout/laycache.cxx sw/source/core/layout/trvlfrm.cxx sw/source/core/text/atrhndl.hxx sw/source/core/text/atrstck.cxx sw/source/core/text/inftxt.cxx sw/source/core/text/porfld.cxx sw/source/core/text/porfld.hxx sw/source/core/text/txtfld.cxx sw/source/core/tox/tox.cxx sw/source/core/tox/txmsrt.cxx sw/source/core/txtnode/atrfld.cxx sw/source/core/txtnode/ndhints.cxx sw/source/core/txtnode/ndtxt.cxx sw/source/core/txtnode/thints.cxx sw/source/core/txtnode/txatbase.cxx sw/source/core/txtnode/txatritr.cxx sw/source/core/txtnode/txtedt.cxx sw/source/core/undo/SwUndoField.cxx sw/source/core/undo/rolbck.cxx sw/source/core/undo/unsect.cxx sw/source/core/undo/untbl.cxx sw/source/core/unocore/unocrsrhelper.cxx sw/source/core/unocore/unofield.cxx sw/source/core/unocore/unoframe.cxx sw/source/core/unocore/unorefmk.cxx sw/source/core/view/vprint.cxx sw/source/filter/ascii/ascatr.cxx sw/source/filter/html/htmlatr.cxx sw/source/filter/rtf/rtffly.cxx sw/source/filter/ww1/fltshell.cxx sw/source/filter/ww1/w1filter.cxx sw/source/filter/ww8/wrtw8esh.cxx sw/source/filter/ww8/wrtw8nds.cxx sw/source/filter/ww8/ww8atr.cxx sw/source/filter/ww8/ww8par.cxx sw/source/filter/ww8/ww8par3.cxx sw/source/filter/ww8/ww8par5.cxx sw/source/ui/app/applab.cxx sw/source/ui/docvw/edtwin.cxx sw/source/ui/docvw/edtwin2.cxx sw/source/ui/envelp/envfmt.cxx sw/source/ui/fldui/fldedt.cxx sw/source/ui/fldui/fldmgr.cxx sw/source/ui/inc/wrtsh.hxx sw/source/ui/lingu/hhcwrp.cxx sw/source/ui/shells/basesh.cxx sw/source/ui/shells/drwtxtex.cxx sw/source/ui/shells/grfsh.cxx sw/source/ui/shells/tabsh.cxx sw/source/ui/shells/textfld.cxx sw/source/ui/shells/textidx.cxx sw/source/ui/shells/textsh.cxx sw/source/ui/shells/textsh1.cxx sw/source/ui/shells/txtattr.cxx sw/source/ui/uiview/formatclipboard.cxx sw/source/ui/uiview/view.cxx sw/source/ui/uiview/view2.cxx sw/source/ui/uiview/viewsrch.cxx sw/source/ui/uiview/viewstat.cxx sw/source/ui/uiview/viewtab.cxx sw/source/ui/wrtsh/wrtsh1.cxx sw/source/ui/wrtsh/wrtsh2.cxx Change-Id: I6758d6719e1fd523797cc6ee2648ffcb47d52595
841 lines
25 KiB
C++
841 lines
25 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*
|
|
* This file is part of the LibreOffice project.
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*
|
|
* This file incorporates work covered by the following license notice:
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed
|
|
* with this work for additional information regarding copyright
|
|
* ownership. The ASF licenses this file to you under the Apache
|
|
* License, Version 2.0 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
|
|
*/
|
|
|
|
#include <hintids.hxx>
|
|
#include <editsh.hxx>
|
|
#include <edimp.hxx>
|
|
#include <doc.hxx>
|
|
#include <IDocumentUndoRedo.hxx>
|
|
#include <ndtxt.hxx>
|
|
#include <paratr.hxx>
|
|
#include <swundo.hxx>
|
|
#include <numrule.hxx>
|
|
|
|
SwPamRanges::SwPamRanges( const SwPaM& rRing )
|
|
{
|
|
const SwPaM* pTmp = &rRing;
|
|
do {
|
|
Insert( pTmp->GetMark()->nNode, pTmp->GetPoint()->nNode );
|
|
} while( &rRing != ( pTmp = (const SwPaM*)pTmp->GetNext() ));
|
|
}
|
|
|
|
void SwPamRanges::Insert( const SwNodeIndex& rIdx1, const SwNodeIndex& rIdx2 )
|
|
{
|
|
SwPamRange aRg( rIdx1.GetIndex(), rIdx2.GetIndex() );
|
|
if( aRg.nEnd < aRg.nStart )
|
|
{ aRg.nStart = aRg.nEnd; aRg.nEnd = rIdx1.GetIndex(); }
|
|
|
|
_SwPamRanges::const_iterator it = lower_bound(aRg); //search Insert Position
|
|
sal_uInt16 nPos = it - begin();
|
|
if (!empty() && (it != end()) && (*it) == aRg)
|
|
{
|
|
// is the one in the Array smaller?
|
|
SwPamRange const& rTmp = _SwPamRanges::operator[](nPos);
|
|
if( rTmp.nEnd < aRg.nEnd )
|
|
{
|
|
aRg.nEnd = rTmp.nEnd;
|
|
erase(begin() + nPos); // combine
|
|
}
|
|
else
|
|
return; // done, because by precondition everything is combined
|
|
}
|
|
|
|
bool bEnde;
|
|
do {
|
|
bEnde = true;
|
|
|
|
// combine with predecessor?
|
|
if( nPos > 0 )
|
|
{
|
|
SwPamRange const& rTmp = _SwPamRanges::operator[](nPos-1);
|
|
if( rTmp.nEnd == aRg.nStart
|
|
|| rTmp.nEnd+1 == aRg.nStart )
|
|
{
|
|
aRg.nStart = rTmp.nStart;
|
|
bEnde = false;
|
|
erase( begin() + --nPos ); // combine
|
|
}
|
|
// range contained in rTmp?
|
|
else if( rTmp.nStart <= aRg.nStart && aRg.nEnd <= rTmp.nEnd )
|
|
return;
|
|
}
|
|
// combine with successor?
|
|
if( nPos < size() )
|
|
{
|
|
SwPamRange const& rTmp = _SwPamRanges::operator[](nPos);
|
|
if( rTmp.nStart == aRg.nEnd ||
|
|
rTmp.nStart == aRg.nEnd+1 )
|
|
{
|
|
aRg.nEnd = rTmp.nEnd;
|
|
bEnde = false;
|
|
erase( begin() + nPos ); // combine
|
|
}
|
|
|
|
// range contained in rTmp?
|
|
else if( rTmp.nStart <= aRg.nStart && aRg.nEnd <= rTmp.nEnd )
|
|
return;
|
|
}
|
|
} while( !bEnde );
|
|
|
|
_SwPamRanges::insert( aRg );
|
|
}
|
|
|
|
SwPaM& SwPamRanges::SetPam( sal_uInt16 nArrPos, SwPaM& rPam )
|
|
{
|
|
OSL_ASSERT( nArrPos < Count() );
|
|
const SwPamRange& rTmp = (*this)[ nArrPos ];
|
|
rPam.GetPoint()->nNode = rTmp.nStart;
|
|
rPam.GetPoint()->nContent.Assign( rPam.GetCntntNode(), 0 );
|
|
rPam.SetMark();
|
|
rPam.GetPoint()->nNode = rTmp.nEnd;
|
|
rPam.GetPoint()->nContent.Assign( rPam.GetCntntNode(), 0 );
|
|
return rPam;
|
|
}
|
|
|
|
// Rule book for outline numbering
|
|
|
|
void SwEditShell::SetOutlineNumRule(const SwNumRule& rRule)
|
|
{
|
|
StartAllAction(); // bracketing for updating!
|
|
GetDoc()->SetOutlineNumRule(rRule);
|
|
EndAllAction();
|
|
}
|
|
|
|
const SwNumRule* SwEditShell::GetOutlineNumRule() const
|
|
{
|
|
return GetDoc()->GetOutlineNumRule();
|
|
}
|
|
|
|
// Set if there is no numbering yet, else update.
|
|
// Works with old and new rules. Update only differences.
|
|
|
|
// paragraphs without numbering, with indentations
|
|
bool SwEditShell::NoNum()
|
|
{
|
|
bool bRet = true;
|
|
StartAllAction();
|
|
|
|
SwPaM* pCrsr = GetCrsr();
|
|
if( pCrsr->GetNext() != pCrsr ) // Multiple selection?
|
|
{
|
|
GetDoc()->GetIDocumentUndoRedo().StartUndo( UNDO_START, NULL );
|
|
SwPamRanges aRangeArr( *pCrsr );
|
|
SwPaM aPam( *pCrsr->GetPoint() );
|
|
for( sal_uInt16 n = 0; n < aRangeArr.Count(); ++n )
|
|
bRet = bRet && GetDoc()->NoNum( aRangeArr.SetPam( n, aPam ));
|
|
GetDoc()->GetIDocumentUndoRedo().EndUndo( UNDO_END, NULL );
|
|
}
|
|
else
|
|
bRet = GetDoc()->NoNum( *pCrsr );
|
|
|
|
EndAllAction();
|
|
return bRet;
|
|
}
|
|
|
|
sal_Bool SwEditShell::SelectionHasNumber() const
|
|
{
|
|
sal_Bool bResult = HasNumber();
|
|
const SwTxtNode * pTxtNd =
|
|
GetCrsr()->GetPoint()->nNode.GetNode().GetTxtNode();
|
|
if (!bResult && pTxtNd && pTxtNd->Len()==0 && !pTxtNd->GetNumRule()) {
|
|
SwPamRanges aRangeArr( *GetCrsr() );
|
|
SwPaM aPam( *GetCrsr()->GetPoint() );
|
|
for( sal_uInt16 n = 0; n < aRangeArr.Count(); ++n )
|
|
{
|
|
aRangeArr.SetPam( n, aPam );
|
|
{
|
|
sal_uInt32 nStt = aPam.GetPoint()->nNode.GetIndex(),
|
|
nEnd = aPam.GetMark()->nNode.GetIndex();
|
|
if( nStt > nEnd )
|
|
{
|
|
sal_uInt32 nTmp = nStt; nStt = nEnd; nEnd = nTmp;
|
|
}
|
|
for (sal_uInt32 nPos = nStt; nPos<=nEnd; nPos++)
|
|
{
|
|
pTxtNd = mpDoc->GetNodes()[nPos]->GetTxtNode();
|
|
if (pTxtNd && pTxtNd->Len()!=0)
|
|
{
|
|
bResult = pTxtNd->HasNumber();
|
|
|
|
// #b6340308# special case: outline numbered, not counted paragraph
|
|
if ( bResult &&
|
|
pTxtNd->GetNumRule() == GetDoc()->GetOutlineNumRule() &&
|
|
!pTxtNd->IsCountedInList() )
|
|
{
|
|
bResult = sal_False;
|
|
}
|
|
if (bResult==sal_False) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
// add a new function to determine number on/off status
|
|
sal_Bool SwEditShell::SelectionHasBullet() const
|
|
{
|
|
sal_Bool bResult = HasBullet();
|
|
const SwTxtNode * pTxtNd =
|
|
GetCrsr()->GetPoint()->nNode.GetNode().GetTxtNode();
|
|
if (!bResult && pTxtNd && pTxtNd->Len()==0 && !pTxtNd->GetNumRule()) {
|
|
SwPamRanges aRangeArr( *GetCrsr() );
|
|
SwPaM aPam( *GetCrsr()->GetPoint() );
|
|
for( sal_uInt16 n = 0; n < aRangeArr.Count(); ++n )
|
|
{
|
|
aRangeArr.SetPam( n, aPam );
|
|
{
|
|
sal_uInt32 nStt = aPam.GetPoint()->nNode.GetIndex(),
|
|
nEnd = aPam.GetMark()->nNode.GetIndex();
|
|
if( nStt > nEnd )
|
|
{
|
|
sal_uInt32 nTmp = nStt; nStt = nEnd; nEnd = nTmp;
|
|
}
|
|
for (sal_uInt32 nPos = nStt; nPos<=nEnd; nPos++)
|
|
{
|
|
pTxtNd = mpDoc->GetNodes()[nPos]->GetTxtNode();
|
|
if (pTxtNd && pTxtNd->Len()!=0)
|
|
{
|
|
bResult = pTxtNd->HasBullet();
|
|
|
|
if (bResult==sal_False) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
// -> #i29560#
|
|
sal_Bool SwEditShell::HasNumber() const
|
|
{
|
|
sal_Bool bResult = sal_False;
|
|
|
|
const SwTxtNode * pTxtNd =
|
|
GetCrsr()->GetPoint()->nNode.GetNode().GetTxtNode();
|
|
|
|
if (pTxtNd)
|
|
{
|
|
bResult = pTxtNd->HasNumber();
|
|
|
|
// special case: outline numbered, not counted paragraph
|
|
if ( bResult &&
|
|
pTxtNd->GetNumRule() == GetDoc()->GetOutlineNumRule() &&
|
|
!pTxtNd->IsCountedInList() )
|
|
{
|
|
bResult = sal_False;
|
|
}
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
sal_Bool SwEditShell::HasBullet() const
|
|
{
|
|
sal_Bool bResult = sal_False;
|
|
|
|
const SwTxtNode * pTxtNd =
|
|
GetCrsr()->GetPoint()->nNode.GetNode().GetTxtNode();
|
|
|
|
if (pTxtNd)
|
|
{
|
|
bResult = pTxtNd->HasBullet();
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
// <- #i29560#
|
|
|
|
/// delete, split list
|
|
void SwEditShell::DelNumRules()
|
|
{
|
|
StartAllAction();
|
|
|
|
SwPaM* pCrsr = GetCrsr();
|
|
if( pCrsr->GetNext() != pCrsr ) // multi-selection?
|
|
{
|
|
GetDoc()->GetIDocumentUndoRedo().StartUndo( UNDO_START, NULL );
|
|
SwPamRanges aRangeArr( *pCrsr );
|
|
SwPaM aPam( *pCrsr->GetPoint() );
|
|
for( sal_uInt16 n = 0; n < aRangeArr.Count(); ++n )
|
|
{
|
|
GetDoc()->DelNumRules( aRangeArr.SetPam( n, aPam ) );
|
|
}
|
|
GetDoc()->GetIDocumentUndoRedo().EndUndo( UNDO_END, NULL );
|
|
}
|
|
else
|
|
GetDoc()->DelNumRules( *pCrsr );
|
|
|
|
// Call AttrChangeNotify on the UI-side. Should actually be redundant but there was a bug once.
|
|
CallChgLnk();
|
|
|
|
// Cursor cannot be in front of a label anymore, because numbering/bullet is deleted.
|
|
SetInFrontOfLabel( false );
|
|
|
|
GetDoc()->SetModified();
|
|
EndAllAction();
|
|
}
|
|
|
|
// up- & downgrading
|
|
bool SwEditShell::NumUpDown( bool bDown )
|
|
{
|
|
StartAllAction();
|
|
|
|
bool bRet = true;
|
|
SwPaM* pCrsr = GetCrsr();
|
|
if( pCrsr->GetNext() == pCrsr ) // no multiple selection ?
|
|
bRet = GetDoc()->NumUpDown( *pCrsr, bDown );
|
|
else
|
|
{
|
|
GetDoc()->GetIDocumentUndoRedo().StartUndo( UNDO_START, NULL );
|
|
SwPamRanges aRangeArr( *pCrsr );
|
|
SwPaM aPam( *pCrsr->GetPoint() );
|
|
for( sal_uInt16 n = 0; n < aRangeArr.Count(); ++n )
|
|
bRet = bRet && GetDoc()->NumUpDown( aRangeArr.SetPam( n, aPam ), bDown );
|
|
GetDoc()->GetIDocumentUndoRedo().EndUndo( UNDO_END, NULL );
|
|
}
|
|
GetDoc()->SetModified();
|
|
|
|
// #i54693# Update marked numbering levels
|
|
if ( IsInFrontOfLabel() )
|
|
UpdateMarkedListLevel();
|
|
|
|
CallChgLnk();
|
|
|
|
EndAllAction();
|
|
return bRet;
|
|
}
|
|
|
|
// -> #i23726#
|
|
bool SwEditShell::IsFirstOfNumRule() const
|
|
{
|
|
bool bResult = false;
|
|
|
|
SwPaM * pCrsr = GetCrsr();
|
|
if (pCrsr->GetNext() == pCrsr)
|
|
{
|
|
bResult = IsFirstOfNumRule(*pCrsr);
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
bool SwEditShell::IsFirstOfNumRule(const SwPaM & rPaM) const
|
|
{
|
|
SwPosition aPos(*rPaM.GetPoint());
|
|
return (GetDoc()->IsFirstOfNumRule(aPos));
|
|
}
|
|
// <- #i23726#
|
|
|
|
// -> #i23725#, #i90078#
|
|
void SwEditShell::ChangeIndentOfAllListLevels( short nDiff )
|
|
{
|
|
StartAllAction();
|
|
|
|
const SwNumRule *pCurNumRule = GetCurNumRule();
|
|
//#120911# check if numbering rule really exists
|
|
if (pCurNumRule)
|
|
{
|
|
SwNumRule aRule(*pCurNumRule);
|
|
// #i90078#
|
|
aRule.ChangeIndent( nDiff );
|
|
|
|
// no start of new list
|
|
SetCurNumRule( aRule, false );
|
|
}
|
|
|
|
EndAllAction();
|
|
}
|
|
|
|
// #i90078#
|
|
void SwEditShell::SetIndent(short nIndent, const SwPosition & rPos)
|
|
{
|
|
StartAllAction();
|
|
|
|
SwNumRule *pCurNumRule = GetDoc()->GetCurrNumRule(rPos);
|
|
|
|
if (pCurNumRule)
|
|
{
|
|
SwPaM aPaM(rPos);
|
|
SwTxtNode * pTxtNode = aPaM.GetNode()->GetTxtNode();
|
|
|
|
SwNumRule aRule(*pCurNumRule);
|
|
|
|
if ( IsFirstOfNumRule() )
|
|
{
|
|
aRule.SetIndentOfFirstListLevelAndChangeOthers( nIndent );
|
|
}
|
|
else if ( pTxtNode->GetActualListLevel() >= 0 )
|
|
{
|
|
aRule.SetIndent( nIndent,
|
|
static_cast<sal_uInt16>(pTxtNode->GetActualListLevel()) );
|
|
}
|
|
|
|
// #i42921# - 3rd parameter = false in order to
|
|
// suppress setting of num rule at <aPaM>.
|
|
// do not apply any list
|
|
GetDoc()->SetNumRule( aPaM, aRule, false, OUString(), sal_False );
|
|
}
|
|
|
|
EndAllAction();
|
|
}
|
|
// <- #i23725#
|
|
|
|
bool SwEditShell::MoveParagraph( long nOffset )
|
|
{
|
|
StartAllAction();
|
|
|
|
SwPaM *pCrsr = GetCrsr();
|
|
if( !pCrsr->HasMark() )
|
|
{
|
|
// Ensures that Bound1 and Bound2 are in the same Node
|
|
pCrsr->SetMark();
|
|
pCrsr->DeleteMark();
|
|
}
|
|
|
|
bool bRet = GetDoc()->MoveParagraph( *pCrsr, nOffset );
|
|
|
|
GetDoc()->SetModified();
|
|
EndAllAction();
|
|
return bRet;
|
|
}
|
|
|
|
int SwEditShell::GetCurrentParaOutlineLevel( ) const
|
|
{
|
|
int nLevel = 0;
|
|
|
|
SwPaM* pCrsr = GetCrsr();
|
|
const SwTxtNode* pTxtNd = pCrsr->GetNode()->GetTxtNode();
|
|
if( pTxtNd )
|
|
nLevel = pTxtNd->GetAttrOutlineLevel();
|
|
return nLevel;
|
|
}
|
|
|
|
void SwEditShell::GetCurrentOutlineLevels( sal_uInt8& rUpper, sal_uInt8& rLower )
|
|
{
|
|
SwPaM* pCrsr = GetCrsr();
|
|
SwPaM aCrsr( *pCrsr->Start() );
|
|
aCrsr.SetMark();
|
|
if( pCrsr->HasMark() )
|
|
*aCrsr.GetPoint() = *pCrsr->End();
|
|
GetDoc()->GotoNextNum( *aCrsr.GetPoint(), false, &rUpper, &rLower );
|
|
}
|
|
|
|
bool SwEditShell::MoveNumParas( bool bUpperLower, bool bUpperLeft )
|
|
{
|
|
StartAllAction();
|
|
|
|
// On all selections?
|
|
SwPaM* pCrsr = GetCrsr();
|
|
SwPaM aCrsr( *pCrsr->Start() );
|
|
aCrsr.SetMark();
|
|
|
|
if( pCrsr->HasMark() )
|
|
*aCrsr.GetPoint() = *pCrsr->End();
|
|
|
|
sal_Bool bRet = sal_False;
|
|
sal_uInt8 nUpperLevel, nLowerLevel;
|
|
if( GetDoc()->GotoNextNum( *aCrsr.GetPoint(), false,
|
|
&nUpperLevel, &nLowerLevel ))
|
|
{
|
|
if( bUpperLower )
|
|
{
|
|
// on top of the next numbering
|
|
long nOffset = 0;
|
|
const SwNode* pNd;
|
|
|
|
if( bUpperLeft ) // move up
|
|
{
|
|
SwPosition aPos( *aCrsr.GetMark() );
|
|
if( GetDoc()->GotoPrevNum( aPos, false ) )
|
|
nOffset = aPos.nNode.GetIndex() -
|
|
aCrsr.GetMark()->nNode.GetIndex();
|
|
else
|
|
{
|
|
sal_uLong nStt = aPos.nNode.GetIndex(), nIdx = nStt - 1;
|
|
while( nIdx && (
|
|
( pNd = GetDoc()->GetNodes()[ nIdx ])->IsSectionNode() ||
|
|
( pNd->IsEndNode() && pNd->StartOfSectionNode()->IsSectionNode())))
|
|
--nIdx;
|
|
if( GetDoc()->GetNodes()[ nIdx ]->IsTxtNode() )
|
|
nOffset = nIdx - nStt;
|
|
}
|
|
}
|
|
else // move down
|
|
{
|
|
const SwNumRule* pOrig = aCrsr.GetNode(sal_False)->GetTxtNode()->GetNumRule();
|
|
if( aCrsr.GetNode()->IsTxtNode() &&
|
|
pOrig == aCrsr.GetNode()->GetTxtNode()->GetNumRule() )
|
|
{
|
|
sal_uLong nStt = aCrsr.GetPoint()->nNode.GetIndex(), nIdx = nStt+1;
|
|
|
|
while (nIdx < GetDoc()->GetNodes().Count()-1)
|
|
{
|
|
pNd = GetDoc()->GetNodes()[ nIdx ];
|
|
|
|
if (pNd->IsSectionNode() ||
|
|
( pNd->IsEndNode() && pNd->StartOfSectionNode()->IsSectionNode()) ||
|
|
( pNd->IsTxtNode() && pOrig == ((SwTxtNode*)pNd)->GetNumRule() &&
|
|
((SwTxtNode*)pNd)->GetActualListLevel() > nUpperLevel ))
|
|
{
|
|
++nIdx;
|
|
}
|
|
// #i57856#
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if( nStt == nIdx || !GetDoc()->GetNodes()[ nIdx ]->IsTxtNode() )
|
|
nOffset = 1;
|
|
else
|
|
nOffset = nIdx - nStt;
|
|
}
|
|
else
|
|
nOffset = 1;
|
|
}
|
|
|
|
if( nOffset )
|
|
{
|
|
aCrsr.Move( fnMoveBackward, fnGoNode );
|
|
bRet = GetDoc()->MoveParagraph( aCrsr, nOffset );
|
|
}
|
|
}
|
|
else if( bUpperLeft ? nUpperLevel : nLowerLevel+1 < MAXLEVEL )
|
|
{
|
|
aCrsr.Move( fnMoveBackward, fnGoNode );
|
|
bRet = GetDoc()->NumUpDown( aCrsr, !bUpperLeft );
|
|
}
|
|
}
|
|
|
|
GetDoc()->SetModified();
|
|
EndAllAction();
|
|
return bRet;
|
|
}
|
|
|
|
bool SwEditShell::OutlineUpDown( short nOffset )
|
|
{
|
|
StartAllAction();
|
|
|
|
bool bRet = true;
|
|
SwPaM* pCrsr = GetCrsr();
|
|
if( pCrsr->GetNext() == pCrsr ) // no multi selection?
|
|
bRet = GetDoc()->OutlineUpDown( *pCrsr, nOffset );
|
|
else
|
|
{
|
|
GetDoc()->GetIDocumentUndoRedo().StartUndo( UNDO_START, NULL );
|
|
SwPamRanges aRangeArr( *pCrsr );
|
|
SwPaM aPam( *pCrsr->GetPoint() );
|
|
for( sal_uInt16 n = 0; n < aRangeArr.Count(); ++n )
|
|
bRet = bRet && GetDoc()->OutlineUpDown(
|
|
aRangeArr.SetPam( n, aPam ), nOffset );
|
|
GetDoc()->GetIDocumentUndoRedo().EndUndo( UNDO_END, NULL );
|
|
}
|
|
GetDoc()->SetModified();
|
|
EndAllAction();
|
|
return bRet;
|
|
}
|
|
|
|
bool SwEditShell::MoveOutlinePara( short nOffset )
|
|
{
|
|
StartAllAction();
|
|
bool bRet = GetDoc()->MoveOutlinePara( *GetCrsr(), nOffset );
|
|
EndAllAction();
|
|
return bRet;
|
|
}
|
|
|
|
// Outlines and SubOutline are ReadOnly?
|
|
sal_Bool SwEditShell::IsProtectedOutlinePara() const
|
|
{
|
|
sal_Bool bRet = sal_False;
|
|
const SwNode& rNd = GetCrsr()->Start()->nNode.GetNode();
|
|
if( rNd.IsTxtNode() )
|
|
{
|
|
const SwOutlineNodes& rOutlNd = GetDoc()->GetNodes().GetOutLineNds();
|
|
SwNodePtr pNd = (SwNodePtr)&rNd;
|
|
bool bFirst = true;
|
|
sal_uInt16 nPos;
|
|
int nLvl(0);
|
|
if( !rOutlNd.Seek_Entry( pNd, &nPos ) && nPos )
|
|
--nPos;
|
|
|
|
for( ; nPos < rOutlNd.size(); ++nPos )
|
|
{
|
|
SwNodePtr pTmpNd = rOutlNd[ nPos ];
|
|
|
|
int nTmpLvl = pTmpNd->GetTxtNode()->GetAttrOutlineLevel();
|
|
|
|
OSL_ENSURE( nTmpLvl >= 0 && nTmpLvl <= MAXLEVEL,
|
|
"<SwEditShell::IsProtectedOutlinePara()>" );
|
|
|
|
if( bFirst )
|
|
{
|
|
nLvl = nTmpLvl;
|
|
bFirst = false;
|
|
}
|
|
else if( nLvl >= nTmpLvl )
|
|
break;
|
|
|
|
if( pTmpNd->IsProtect() )
|
|
{
|
|
bRet = sal_True;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
#if OSL_DEBUG_LEVEL > 0
|
|
else
|
|
{
|
|
OSL_FAIL("Cursor not on an outline node");
|
|
}
|
|
#endif
|
|
return bRet;
|
|
}
|
|
|
|
/** Test whether outline may be moved (bCopy == false)
|
|
* or copied (bCopy == true)
|
|
* Verify these conditions:
|
|
* 1) outline must be within main body (and not in redline)
|
|
* 2) outline must not be within table
|
|
* 3) if bCopy is set, outline must not be write protected
|
|
*/
|
|
static sal_Bool lcl_IsOutlineMoveAndCopyable( const SwDoc* pDoc, sal_uInt16 nIdx, bool bCopy )
|
|
{
|
|
const SwNodes& rNds = pDoc->GetNodes();
|
|
const SwNode* pNd = rNds.GetOutLineNds()[ nIdx ];
|
|
return pNd->GetIndex() >= rNds.GetEndOfExtras().GetIndex() && // 1) body
|
|
!pNd->FindTableNode() && // 2) table
|
|
( bCopy || !pNd->IsProtect() ); // 3) write
|
|
}
|
|
|
|
sal_Bool SwEditShell::IsOutlineMovable( sal_uInt16 nIdx ) const
|
|
{
|
|
return lcl_IsOutlineMoveAndCopyable( GetDoc(), nIdx, false );
|
|
}
|
|
|
|
sal_Bool SwEditShell::IsOutlineCopyable( sal_uInt16 nIdx ) const
|
|
{
|
|
return lcl_IsOutlineMoveAndCopyable( GetDoc(), nIdx, true );
|
|
}
|
|
|
|
bool SwEditShell::NumOrNoNum( sal_Bool bNumOn, bool bChkStart )
|
|
{
|
|
bool bRet = false;
|
|
SwPaM* pCrsr = GetCrsr();
|
|
if( pCrsr->GetNext() == pCrsr && !pCrsr->HasMark() &&
|
|
( !bChkStart || !pCrsr->GetPoint()->nContent.GetIndex()) )
|
|
{
|
|
StartAllAction(); // Klammern fuers Updaten !!
|
|
bRet = GetDoc()->NumOrNoNum( pCrsr->GetPoint()->nNode, !bNumOn ); // #i29560#
|
|
EndAllAction();
|
|
}
|
|
return bRet;
|
|
}
|
|
|
|
sal_Bool SwEditShell::IsNoNum( sal_Bool bChkStart ) const
|
|
{
|
|
// a Backspace in the paragraph without number becomes a Delete
|
|
sal_Bool bResult = sal_False;
|
|
SwPaM* pCrsr = GetCrsr();
|
|
|
|
if (pCrsr->GetNext() == pCrsr && !pCrsr->HasMark() &&
|
|
(!bChkStart || !pCrsr->GetPoint()->nContent.GetIndex()))
|
|
{
|
|
const SwTxtNode* pTxtNd = pCrsr->GetNode()->GetTxtNode();
|
|
|
|
if (pTxtNd)
|
|
{
|
|
bResult = ! pTxtNd->IsCountedInList();
|
|
}
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
sal_uInt8 SwEditShell::GetNumLevel() const
|
|
{
|
|
// return current level where the point of the cursor is
|
|
sal_uInt8 nLevel = MAXLEVEL;
|
|
|
|
SwPaM* pCrsr = GetCrsr();
|
|
const SwTxtNode* pTxtNd = pCrsr->GetNode()->GetTxtNode();
|
|
|
|
OSL_ENSURE( pTxtNd, "GetNumLevel() without text node" );
|
|
if ( !pTxtNd )
|
|
return nLevel;
|
|
|
|
const SwNumRule* pRule = pTxtNd->GetNumRule();
|
|
if(pRule)
|
|
{
|
|
const int nListLevelOfTxtNode( pTxtNd->GetActualListLevel() );
|
|
if ( nListLevelOfTxtNode >= 0 )
|
|
{
|
|
nLevel = static_cast<sal_uInt8>( nListLevelOfTxtNode );
|
|
}
|
|
}
|
|
|
|
return nLevel;
|
|
}
|
|
|
|
const SwNumRule* SwEditShell::GetCurNumRule() const
|
|
{
|
|
return GetDoc()->GetCurrNumRule( *GetCrsr()->GetPoint() );
|
|
}
|
|
|
|
void SwEditShell::SetCurNumRule( const SwNumRule& rRule,
|
|
const bool bCreateNewList,
|
|
const OUString sContinuedListId,
|
|
const bool bResetIndentAttrs )
|
|
{
|
|
StartAllAction();
|
|
|
|
GetDoc()->GetIDocumentUndoRedo().StartUndo( UNDO_START, NULL );
|
|
|
|
SwPaM* pCrsr = GetCrsr();
|
|
if( pCrsr->GetNext() != pCrsr ) // multiple selection ?
|
|
{
|
|
SwPamRanges aRangeArr( *pCrsr );
|
|
SwPaM aPam( *pCrsr->GetPoint() );
|
|
for( sal_uInt16 n = 0; n < aRangeArr.Count(); ++n )
|
|
{
|
|
aRangeArr.SetPam( n, aPam );
|
|
GetDoc()->SetNumRule( aPam, rRule,
|
|
bCreateNewList, sContinuedListId,
|
|
true , bResetIndentAttrs );
|
|
GetDoc()->SetCounted( aPam, true );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
GetDoc()->SetNumRule( *pCrsr, rRule,
|
|
bCreateNewList, sContinuedListId,
|
|
true, bResetIndentAttrs );
|
|
GetDoc()->SetCounted( *pCrsr, true );
|
|
}
|
|
GetDoc()->GetIDocumentUndoRedo().EndUndo( UNDO_END, NULL );
|
|
|
|
EndAllAction();
|
|
}
|
|
|
|
OUString SwEditShell::GetUniqueNumRuleName( const OUString* pChkStr, bool bAutoNum ) const
|
|
{
|
|
return GetDoc()->GetUniqueNumRuleName( pChkStr, bAutoNum );
|
|
}
|
|
|
|
void SwEditShell::ChgNumRuleFmts( const SwNumRule& rRule )
|
|
{
|
|
StartAllAction();
|
|
GetDoc()->ChgNumRuleFmts( rRule );
|
|
EndAllAction();
|
|
}
|
|
|
|
bool SwEditShell::ReplaceNumRule( const OUString& rOldRule, const OUString& rNewRule )
|
|
{
|
|
StartAllAction();
|
|
bool bRet = GetDoc()->ReplaceNumRule( *GetCrsr()->GetPoint(), rOldRule, rNewRule );
|
|
EndAllAction();
|
|
return bRet;
|
|
}
|
|
|
|
void SwEditShell::SetNumRuleStart( sal_Bool bFlag, SwPaM* pPaM )
|
|
{
|
|
StartAllAction();
|
|
SwPaM* pCrsr = pPaM ? pPaM : GetCrsr();
|
|
if( pCrsr->GetNext() != pCrsr ) // multiple selection ?
|
|
{
|
|
GetDoc()->GetIDocumentUndoRedo().StartUndo( UNDO_START, NULL );
|
|
SwPamRanges aRangeArr( *pCrsr );
|
|
SwPaM aPam( *pCrsr->GetPoint() );
|
|
for( sal_uInt16 n = 0; n < aRangeArr.Count(); ++n )
|
|
GetDoc()->SetNumRuleStart( *aRangeArr.SetPam( n, aPam ).GetPoint(), bFlag );
|
|
GetDoc()->GetIDocumentUndoRedo().EndUndo( UNDO_END, NULL );
|
|
}
|
|
else
|
|
GetDoc()->SetNumRuleStart( *pCrsr->GetPoint(), bFlag );
|
|
|
|
EndAllAction();
|
|
}
|
|
|
|
sal_Bool SwEditShell::IsNumRuleStart( SwPaM* pPaM ) const
|
|
{
|
|
sal_Bool bResult = sal_False;
|
|
SwPaM* pCrsr = pPaM ? pPaM : GetCrsr( );
|
|
const SwTxtNode* pTxtNd = pCrsr->GetNode()->GetTxtNode();
|
|
if( pTxtNd )
|
|
bResult = pTxtNd->IsListRestart() ? sal_True : sal_False;
|
|
return bResult;
|
|
}
|
|
|
|
void SwEditShell::SetNodeNumStart( sal_uInt16 nStt, SwPaM* pPaM )
|
|
{
|
|
StartAllAction();
|
|
|
|
SwPaM* pCrsr = pPaM ? pPaM : GetCrsr();
|
|
if( pCrsr->GetNext() != pCrsr ) // multiple selection ?
|
|
{
|
|
GetDoc()->GetIDocumentUndoRedo().StartUndo( UNDO_START, NULL );
|
|
SwPamRanges aRangeArr( *pCrsr );
|
|
SwPaM aPam( *pCrsr->GetPoint() );
|
|
for( sal_uInt16 n = 0; n < aRangeArr.Count(); ++n )
|
|
GetDoc()->SetNodeNumStart( *aRangeArr.SetPam( n, aPam ).GetPoint(), nStt );
|
|
GetDoc()->GetIDocumentUndoRedo().EndUndo( UNDO_END, NULL );
|
|
}
|
|
else
|
|
GetDoc()->SetNodeNumStart( *pCrsr->GetPoint(), nStt );
|
|
|
|
EndAllAction();
|
|
}
|
|
|
|
sal_uInt16 SwEditShell::GetNodeNumStart( SwPaM* pPaM ) const
|
|
{
|
|
SwPaM* pCrsr = pPaM ? pPaM : GetCrsr();
|
|
const SwTxtNode* pTxtNd = pCrsr->GetNode()->GetTxtNode();
|
|
// correction: check, if list restart value is set at text node and
|
|
// use new method <SwTxtNode::GetAttrListRestartValue()>.
|
|
// return USHRT_MAX, if no list restart value is found.
|
|
if ( pTxtNd && pTxtNd->HasAttrListRestartValue() )
|
|
{
|
|
return static_cast<sal_uInt16>(pTxtNd->GetAttrListRestartValue());
|
|
}
|
|
return USHRT_MAX;
|
|
}
|
|
|
|
const SwNumRule * SwEditShell::SearchNumRule( const bool bForward,
|
|
const bool bNum,
|
|
const bool bOutline,
|
|
int nNonEmptyAllowed,
|
|
OUString& sListId )
|
|
{
|
|
return GetDoc()->SearchNumRule( *(bForward ? GetCrsr()->End() : GetCrsr()->Start()),
|
|
bForward, bNum, bOutline, nNonEmptyAllowed,
|
|
sListId );
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|