1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include <hintids.hxx>
21 #include <osl/diagnose.h>
22 #include <vcl/commandevent.hxx>
23 #include <unotools/charclass.hxx>
24 #include <comphelper/processfactory.hxx>
25 #include <comphelper/string.hxx>
26 #include <unotools/transliterationwrapper.hxx>
27 #include <fmtsrnd.hxx>
28 #include <fmtinfmt.hxx>
29 #include <txtinet.hxx>
31 #include <charfmt.hxx>
33 #include <IDocumentUndoRedo.hxx>
34 #include <IDocumentSettingAccess.hxx>
35 #include <IDocumentLinksAdministration.hxx>
36 #include <IDocumentFieldsAccess.hxx>
37 #include <IDocumentStatistics.hxx>
38 #include <IDocumentState.hxx>
50 #include <rootfrm.hxx>
51 #include <extinput.hxx>
52 #include <scriptinfo.hxx>
53 #include <unocrsrhelper.hxx>
54 #include <section.hxx>
55 #include <numrule.hxx>
56 #include <SwNodeNum.hxx>
57 #include <unocrsr.hxx>
60 using namespace com::sun::star
;
62 void SwEditShell::Insert( sal_Unicode c
, bool bOnlyCurrCursor
)
65 for(SwPaM
& rPaM
: GetCursor()->GetRingContainer())
67 const bool bSuccess
= GetDoc()->getIDocumentContentOperations().InsertString(rPaM
, OUString(c
));
68 OSL_ENSURE( bSuccess
, "Doc->Insert() failed." );
70 SaveTableBoxContent( rPaM
.GetPoint() );
79 void SwEditShell::Insert2(const OUString
&rStr
, const bool bForceExpandHints
)
83 const SwInsertFlags nInsertFlags
=
85 ? (SwInsertFlags::FORCEHINTEXPAND
| SwInsertFlags::EMPTYEXPAND
)
86 : SwInsertFlags::EMPTYEXPAND
;
88 for(SwPaM
& rCurrentCursor
: getShellCursor( true )->GetRingContainer())
90 //OPT: GetSystemCharSet
91 GetDoc()->getIDocumentContentOperations().SetIME(false);
93 GetDoc()->getIDocumentContentOperations().InsertString(rCurrentCursor
, rStr
, nInsertFlags
);
94 OSL_ENSURE( bSuccess
, "Doc->Insert() failed." );
98 GetDoc()->UpdateRsid( rCurrentCursor
, rStr
.getLength() );
100 // Set paragraph rsid if beginning of paragraph
101 SwTextNode
*const pTextNode
=
102 rCurrentCursor
.GetPoint()->GetNode().GetTextNode();
103 if( pTextNode
&& pTextNode
->Len() == 1)
104 GetDoc()->UpdateParRsid( pTextNode
);
107 SaveTableBoxContent( rCurrentCursor
.GetPoint() );
112 // calculate cursor bidi level
113 SwCursor
* pTmpCursor
= GetCursor_();
114 const bool bDoNotSetBidiLevel
= ! pTmpCursor
||
115 ( dynamic_cast<SwUnoCursor
*>(pTmpCursor
) != nullptr );
117 if ( ! bDoNotSetBidiLevel
)
119 SwNode
& rNode
= pTmpCursor
->GetPoint()->GetNode();
120 if ( rNode
.IsTextNode() )
122 sal_Int32 nPrevPos
= pTmpCursor
->GetPoint()->GetContentIndex();
126 SwTextFrame
const* pFrame
;
127 SwScriptInfo
*const pSI
= SwScriptInfo::GetScriptInfo(
128 static_cast<SwTextNode
&>(rNode
), &pFrame
, true);
130 sal_uInt8 nLevel
= 0;
133 // seems to be an empty paragraph.
135 std::pair
<Point
, bool> const tmp(aPt
, false);
136 pFrame
= static_cast<SwTextFrame
*>(
137 static_cast<SwTextNode
&>(rNode
).getLayoutFrame(
138 GetLayout(), pTmpCursor
->GetPoint(), &tmp
));
140 SwScriptInfo aScriptInfo
;
141 aScriptInfo
.InitScriptInfo(static_cast<SwTextNode
&>(rNode
),
142 pFrame
->GetMergedPara(), pFrame
->IsRightToLeft());
143 TextFrameIndex
const iPrevPos(pFrame
->MapModelToView(
144 &static_cast<SwTextNode
&>(rNode
), nPrevPos
));
145 nLevel
= aScriptInfo
.DirType( iPrevPos
);
149 if (TextFrameIndex(COMPLETE_STRING
) != pSI
->GetInvalidityA())
151 // mystery why this doesn't use the other overload?
152 pSI
->InitScriptInfo(static_cast<SwTextNode
&>(rNode
), pFrame
->GetMergedPara());
154 TextFrameIndex
const iPrevPos(pFrame
->MapModelToView(
155 &static_cast<SwTextNode
&>(rNode
), nPrevPos
));
156 nLevel
= pSI
->DirType(iPrevPos
);
159 pTmpCursor
->SetCursorBidiLevel( nLevel
);
163 SetInFrontOfLabel( false ); // #i27615#
168 void SwEditShell::Overwrite(const OUString
&rStr
)
171 for(SwPaM
& rPaM
: GetCursor()->GetRingContainer())
173 if( !GetDoc()->getIDocumentContentOperations().Overwrite(rPaM
, rStr
) )
175 OSL_FAIL( "Doc->getIDocumentContentOperations().Overwrite(Str) failed." );
177 SaveTableBoxContent( rPaM
.GetPoint() );
182 void SwEditShell::SplitNode( bool bAutoFormat
, bool bCheckTableStart
)
185 GetDoc()->GetIDocumentUndoRedo().StartUndo(SwUndoId::EMPTY
, nullptr);
187 for(SwPaM
& rPaM
: GetCursor()->GetRingContainer())
189 // Here, a table cell becomes a normal text cell.
190 GetDoc()->ClearBoxNumAttrs( rPaM
.GetPoint()->GetNode() );
191 GetDoc()->getIDocumentContentOperations().SplitNode( *rPaM
.GetPoint(), bCheckTableStart
);
194 GetDoc()->GetIDocumentUndoRedo().EndUndo(SwUndoId::EMPTY
, nullptr);
197 AutoFormatBySplitNode();
199 ClearTableBoxContent();
204 bool SwEditShell::AppendTextNode()
208 GetDoc()->GetIDocumentUndoRedo().StartUndo(SwUndoId::EMPTY
, nullptr);
210 for(SwPaM
& rPaM
: GetCursor()->GetRingContainer())
212 GetDoc()->ClearBoxNumAttrs( rPaM
.GetPoint()->GetNode() );
213 bRet
= GetDoc()->getIDocumentContentOperations().AppendTextNode( *rPaM
.GetPoint()) || bRet
;
216 GetDoc()->GetIDocumentUndoRedo().EndUndo(SwUndoId::EMPTY
, nullptr);
218 ClearTableBoxContent();
224 // the returned SwGrfNode pointer is used in GetGraphic() and GetGraphicSize()
225 SwGrfNode
* SwEditShell::GetGrfNode_() const
227 SwGrfNode
*pGrfNode
= nullptr;
228 SwPaM
* pCursor
= GetCursor();
229 if( !pCursor
->HasMark() ||
230 pCursor
->GetPoint()->GetNode() == pCursor
->GetMark()->GetNode() )
231 pGrfNode
= pCursor
->GetPoint()->GetNode().GetGrfNode();
236 // returns a Graphic pointer if CurrentCursor->GetPoint() points to a SwGrfNode and
237 // GetMark is not set or points to the same Graphic
238 const Graphic
* SwEditShell::GetGraphic( bool bWait
) const
240 SwGrfNode
* pGrfNode
= GetGrfNode_();
241 const Graphic
* pGrf( nullptr );
244 pGrf
= &(pGrfNode
->GetGrf(bWait
&& GraphicType::Default
== pGrfNode
->GetGrf().GetType()));
249 bool SwEditShell::IsLinkedGrfSwapOut() const
251 SwGrfNode
*pGrfNode
= GetGrfNode_();
253 pGrfNode
->IsLinkedFile() &&
254 GraphicType::Default
== pGrfNode
->GetGrfObj().GetType();
257 const GraphicObject
* SwEditShell::GetGraphicObj() const
259 SwGrfNode
* pGrfNode
= GetGrfNode_();
260 return pGrfNode
? &(pGrfNode
->GetGrfObj()) : nullptr;
263 const GraphicAttr
* SwEditShell::GetGraphicAttr( GraphicAttr
& rGA
) const
265 SwGrfNode
* pGrfNode
= GetGrfNode_();
266 const SwFrame
* pFrame
= GetCurrFrame(false);
267 return pGrfNode
? &(pGrfNode
->GetGraphicAttr( rGA
, pFrame
)) : nullptr;
270 GraphicType
SwEditShell::GetGraphicType() const
272 SwGrfNode
*pGrfNode
= GetGrfNode_();
273 return pGrfNode
? pGrfNode
->GetGrfObj().GetType() : GraphicType::NONE
;
276 // returns the size of a graphic in <rSz> if CurrentCursor->GetPoint() points to a SwGrfNode and
277 // GetMark is not set or points to the same graphic
278 bool SwEditShell::GetGrfSize(Size
& rSz
) const
280 SwNoTextNode
* pNoTextNd
;
281 SwPaM
* pCurrentCursor
= GetCursor();
282 if( ( !pCurrentCursor
->HasMark()
283 || pCurrentCursor
->GetPoint()->GetNode() == pCurrentCursor
->GetMark()->GetNode() )
284 && nullptr != ( pNoTextNd
= pCurrentCursor
->GetPointNode().GetNoTextNode() ) )
286 rSz
= pNoTextNd
->GetTwipSize();
293 /// Read again if graphic is not OK and replace old one
294 void SwEditShell::ReRead( const OUString
& rGrfName
, const OUString
& rFltName
,
295 const Graphic
* pGraphic
)
298 mxDoc
->getIDocumentContentOperations().ReRead( *GetCursor(), rGrfName
, rFltName
, pGraphic
);
302 /// Returns the name and the filter name of a graphic if the pointer is on a graphic.
303 /// If a String-pointer is != 0 then return corresponding name.
304 void SwEditShell::GetGrfNms( OUString
* pGrfName
, OUString
* pFltName
,
305 const SwFlyFrameFormat
* pFormat
) const
307 OSL_ENSURE( pGrfName
|| pFltName
, "No parameters" );
309 SwDoc::GetGrfNms( *pFormat
, pGrfName
, pFltName
);
312 SwGrfNode
*pGrfNode
= GetGrfNode_();
313 if( pGrfNode
&& pGrfNode
->IsLinkedFile() )
314 pGrfNode
->GetFileFilterNms( pGrfName
, pFltName
);
318 const tools::PolyPolygon
*SwEditShell::GetGraphicPolygon() const
320 SwNoTextNode
*pNd
= GetCursor()->GetPointNode().GetNoTextNode();
321 return pNd
->HasContour();
324 void SwEditShell::SetGraphicPolygon( const tools::PolyPolygon
*pPoly
)
326 SwNoTextNode
*pNd
= GetCursor()->GetPointNode().GetNoTextNode();
328 pNd
->SetContour( pPoly
);
329 SwFlyFrame
*pFly
= static_cast<SwFlyFrame
*>(pNd
->getLayoutFrame(GetLayout())->GetUpper());
330 const SwFormatSurround
&rSur
= pFly
->GetFormat()->GetSurround();
331 pFly
->GetFormat()->CallSwClientNotify(sw::LegacyModifyHint(&rSur
, &rSur
));
332 GetDoc()->getIDocumentState().SetModified();
336 void SwEditShell::ClearAutomaticContour()
338 SwNoTextNode
*pNd
= GetCursor()->GetPointNode().GetNoTextNode();
339 OSL_ENSURE( pNd
, "is no NoTextNode!" );
340 if( pNd
->HasAutomaticContour() )
343 pNd
->SetContour( nullptr );
344 SwFlyFrame
*pFly
= static_cast<SwFlyFrame
*>(pNd
->getLayoutFrame(GetLayout())->GetUpper());
345 const SwFormatSurround
&rSur
= pFly
->GetFormat()->GetSurround();
346 pFly
->GetFormat()->CallSwClientNotify(sw::LegacyModifyHint(&rSur
, &rSur
));
347 GetDoc()->getIDocumentState().SetModified();
352 /** Get OLE object at pointer.
354 * Returns a pointer to a SvInPlaceObjectRef if CurrentCursor->GetPoint() points to a SwOLENode and
355 * GetMark is not set or points to the same object reference. Gets this pointer from the Doc
356 * if the object should be searched by name.
358 svt::EmbeddedObjectRef
& SwEditShell::GetOLEObject() const
360 OSL_ENSURE( CNT_OLE
== GetCntType(), "GetOLEObj: no OLENode." );
361 OSL_ENSURE( !GetCursor()->HasMark() ||
362 (GetCursor()->HasMark() &&
363 GetCursor()->GetPoint()->GetNode() == GetCursor()->GetMark()->GetNode()),
364 "GetOLEObj: no OLENode." );
366 SwOLENode
*pOLENode
= GetCursor()->GetPointNode().GetOLENode();
367 OSL_ENSURE( pOLENode
, "GetOLEObj: no OLENode." );
368 SwOLEObj
& rOObj
= pOLENode
->GetOLEObj();
369 return rOObj
.GetObject();
372 bool SwEditShell::HasOLEObj( std::u16string_view rName
) const
375 SwNodeIndex
aIdx( *GetNodes().GetEndOfAutotext().StartOfSectionNode(), 1 );
376 while ( nullptr != (pStNd
= aIdx
.GetNode().GetStartNode()) )
379 SwNode
& rNd
= aIdx
.GetNode();
380 if( rNd
.IsOLENode() &&
381 rName
== static_cast<SwOLENode
&>(rNd
).GetChartTableName() &&
382 static_cast<SwOLENode
&>(rNd
).getLayoutFrame( GetLayout() ) )
385 aIdx
.Assign( *pStNd
->EndOfSectionNode(), + 1 );
390 void SwEditShell::SetChartName( const OUString
&rName
)
392 SwOLENode
*pONd
= GetCursor()->GetPointNode().GetOLENode();
393 OSL_ENSURE( pONd
, "ChartNode not found" );
394 pONd
->SetChartTableName( rName
);
397 void SwEditShell::UpdateCharts( const OUString
& rName
)
399 GetDoc()->UpdateCharts( rName
);
402 /// change table name
403 void SwEditShell::SetTableName( SwFrameFormat
& rTableFormat
, const OUString
&rNewName
)
405 GetDoc()->SetTableName( rTableFormat
, rNewName
);
408 /// request current word
409 OUString
SwEditShell::GetCurWord() const
411 const SwPaM
& rPaM
= *GetCursor();
412 const SwTextNode
* pNd
= rPaM
.GetPointNode().GetTextNode();
417 SwTextFrame
const*const pFrame(static_cast<SwTextFrame
*>(pNd
->getLayoutFrame(GetLayout())));
420 return pFrame
->GetCurWord(*rPaM
.GetPoint());
425 void SwEditShell::UpdateDocStat( )
428 GetDoc()->getIDocumentStatistics().UpdateDocStat( false, true );
432 const SwDocStat
& SwEditShell::GetUpdatedDocStat()
435 const SwDocStat
&rRet
= GetDoc()->getIDocumentStatistics().GetUpdatedDocStat( false, true );
440 /// get the reference of a given name in the Doc
441 const SwFormatRefMark
* SwEditShell::GetRefMark( std::u16string_view rName
) const
443 return GetDoc()->GetRefMark( rName
);
446 /// get the names of all references in a Doc
447 sal_uInt16
SwEditShell::GetRefMarks( std::vector
<OUString
>* pStrings
) const
449 return GetDoc()->GetRefMarks( pStrings
);
452 OUString
SwEditShell::GetDropText( const sal_Int32 nChars
) const
455 * pb: made changes for #i74939#
457 * always return a string even though there is a selection
461 SwPaM
* pCursor
= GetCursor();
462 if ( IsMultiSelection() )
464 // if a multi selection exists, search for the first line
465 // -> it is the cursor with the lowest index
466 SwNodeOffset nIndex
= pCursor
->GetMark()->GetNodeIndex();
468 SwPaM
* pLast
= pCursor
;
469 SwPaM
* pTemp
= pCursor
;
472 SwPaM
* pPrev2
= pTemp
->GetPrev();
473 bPrev
= ( pPrev2
&& pPrev2
!= pLast
);
477 SwNodeOffset nTemp
= pPrev2
->GetMark()->GetNodeIndex();
478 if ( nTemp
< nIndex
)
487 SwTextNode
const*const pTextNd
= pCursor
->GetMarkNode().GetTextNode();
490 SwTextFrame
const*const pTextFrame(static_cast<SwTextFrame
const*>(
491 pTextNd
->getLayoutFrame(GetLayout())));
492 SAL_WARN_IF(!pTextFrame
, "sw.core", "GetDropText cursor has no frame?");
495 TextFrameIndex
const nDropLen(pTextFrame
->GetDropLen(TextFrameIndex(nChars
)));
496 aText
= pTextFrame
->GetText().copy(0, sal_Int32(nDropLen
));
503 void SwEditShell::ReplaceDropText( const OUString
&rStr
, SwPaM
* pPaM
)
505 SwPaM
* pCursor
= pPaM
? pPaM
: GetCursor();
506 if( !(pCursor
->GetPoint()->GetNode() == pCursor
->GetMark()->GetNode() &&
507 pCursor
->GetPointNode().GetTextNode()->IsTextNode()) )
512 const SwNode
& rNd
= pCursor
->GetPoint()->GetNode();
513 SwPaM
aPam( rNd
, rStr
.getLength(), rNd
, 0 );
514 SwTextFrame
const*const pTextFrame(static_cast<SwTextFrame
const*>(
515 rNd
.GetTextNode()->getLayoutFrame(GetLayout())));
518 *aPam
.GetPoint() = pTextFrame
->MapViewToModelPos(TextFrameIndex(0));
519 *aPam
.GetMark() = pTextFrame
->MapViewToModelPos(TextFrameIndex(
520 std::min(rStr
.getLength(), pTextFrame
->GetText().getLength())));
522 if( !GetDoc()->getIDocumentContentOperations().Overwrite( aPam
, rStr
) )
524 OSL_FAIL( "Doc->getIDocumentContentOperations().Overwrite(Str) failed." );
530 OUString
SwEditShell::Calculate()
532 OUStringBuffer aFormel
; // the final formula
533 SwCalc
aCalc( *GetDoc() );
534 const CharClass
& rCC
= GetAppCharClass();
536 for(SwPaM
& rCurrentPaM
: GetCursor()->GetNext()->GetRingContainer())
538 SwTextNode
* pTextNd
= rCurrentPaM
.GetPointNode().GetTextNode();
541 const SwPosition
*pStart
= rCurrentPaM
.Start(), *pEnd
= rCurrentPaM
.End();
542 const sal_Int32 nStt
= pStart
->GetContentIndex();
543 OUString aStr
= pTextNd
->GetExpandText(GetLayout(),
544 nStt
, pEnd
->GetContentIndex() - nStt
);
546 aStr
= rCC
.lowercase( aStr
);
548 bool bValidFields
= false;
551 while( nPos
< aStr
.getLength() )
553 sal_Unicode ch
= aStr
[ nPos
++ ];
554 if( rCC
.isLetter( aStr
, nPos
-1 ) || ch
== '_' )
556 sal_Int32 nTmpStt
= nPos
-1;
557 while( nPos
< aStr
.getLength() &&
558 0 != ( ch
= aStr
[ nPos
++ ]) &&
559 (rCC
.isLetterNumeric( aStr
, nPos
- 1 ) ||
560 ch
== '_'|| ch
== '.' ))
563 if( nPos
< aStr
.getLength() )
566 OUString sVar
= aStr
.copy( nTmpStt
, nPos
- nTmpStt
);
567 if( !::FindOperator( sVar
) &&
568 (aCalc
.GetVarTable().Find(sVar
) ||
569 aCalc
.VarLook( sVar
)) )
573 GetDoc()->getIDocumentFieldsAccess().FieldsToCalc( aCalc
,
574 pStart
->GetNodeIndex(),
575 pStart
->GetContentIndex() );
578 aFormel
.append("(" + aCalc
.GetStrResult( aCalc
.VarLook( sVar
)->nValue
) + ")");
581 aFormel
.append(sVar
);
589 return aCalc
.GetStrResult( aCalc
.Calculate(aFormel
.makeStringAndClear()) );
592 sfx2::LinkManager
& SwEditShell::GetLinkManager()
594 return mxDoc
->getIDocumentLinksAdministration().GetLinkManager();
597 void *SwEditShell::GetIMapInventor() const
599 // The node on which the cursor points should be sufficient as a unique identifier
600 return static_cast<void*>(&(GetCursor()->GetPointNode()));
604 Graphic
SwEditShell::GetIMapGraphic() const
606 // returns always a graphic if the cursor is in a Fly
607 CurrShell
aCurr( const_cast<SwEditShell
*>(this) );
609 SwPaM
* pCursor
= GetCursor();
610 if ( !pCursor
->HasMark() )
612 SwNode
& rNd
=pCursor
->GetPointNode();
613 if( rNd
.IsGrfNode() )
615 SwGrfNode
& rGrfNode(static_cast<SwGrfNode
&>(rNd
));
616 aRet
= rGrfNode
.GetGrf(GraphicType::Default
== rGrfNode
.GetGrf().GetType());
618 else if ( rNd
.IsOLENode() )
620 if (const Graphic
* pGraphic
= static_cast<SwOLENode
&>(rNd
).GetGraphic())
625 SwFlyFrame
* pFlyFrame
= rNd
.GetContentNode()->getLayoutFrame( GetLayout() )->FindFlyFrame();
627 aRet
= pFlyFrame
->GetFormat()->MakeGraphic();
633 bool SwEditShell::InsertURL( const SwFormatINetFormat
& rFormat
, const OUString
& rStr
, bool bKeepSelection
)
635 // URL and hint text (directly or via selection) necessary
636 if( rFormat
.GetValue().isEmpty() || ( rStr
.isEmpty() && !HasSelection() ) )
639 GetDoc()->GetIDocumentUndoRedo().StartUndo( SwUndoId::UI_INSERT_URLTXT
, nullptr);
640 bool bInsText
= true;
642 if( !rStr
.isEmpty() )
644 SwPaM
* pCursor
= GetCursor();
645 if( pCursor
->HasMark() && *pCursor
->GetPoint() != *pCursor
->GetMark() )
647 // Selection existent, multi selection?
648 bool bDelText
= true;
649 if( !pCursor
->IsMultiSelection() )
651 // simple selection -> check the text
652 const OUString
sText(comphelper::string::stripEnd(GetSelText(), ' '));
654 bDelText
= bInsText
= false;
656 else if( rFormat
.GetValue() == rStr
) // Are Name and URL equal?
657 bDelText
= bInsText
= false;
662 else if( pCursor
->IsMultiSelection() && rFormat
.GetValue() == rStr
)
669 ExtendSelection( false, rStr
.getLength() );
675 SetAttrItem( rFormat
);
676 if (bInsText
&& !IsCursorPtAtEnd())
682 GetDoc()->GetIDocumentUndoRedo().EndUndo( SwUndoId::UI_INSERT_URLTXT
, nullptr );
687 void SwEditShell::GetINetAttrs(SwGetINetAttrs
& rArr
, bool bIncludeInToxContent
)
691 const SwCharFormats
* pFormats
= GetDoc()->GetCharFormats();
692 for( auto n
= pFormats
->size(); 1 < n
; )
694 SwIterator
<SwTextINetFormat
,SwCharFormat
> aIter(*(*pFormats
)[--n
]);
695 for( SwTextINetFormat
* pFnd
= aIter
.First(); pFnd
; pFnd
= aIter
.Next() )
697 SwTextNode
const*const pTextNd(pFnd
->GetpTextNode());
698 SwTextFrame
const*const pFrame(pTextNd
699 ? static_cast<SwTextFrame
const*>(pTextNd
->getLayoutFrame(GetLayout()))
701 if (nullptr != pTextNd
&& nullptr != pFrame
702 && pTextNd
->GetNodes().IsDocNodes()
703 // check it's not fully deleted
704 && pFrame
->MapModelToView(pTextNd
, pFnd
->GetStart())
705 != pFrame
->MapModelToView(pTextNd
, *pFnd
->GetEnd()))
707 // tdf#52113, tdf#148312 Don't include table of contents hyperlinks in the
708 // Navigator content tree Hyperlinks entries
709 if (!bIncludeInToxContent
)
711 if(const SwSectionNode
* pSectNd
= pTextNd
->FindSectionNode())
713 SectionType eType
= pSectNd
->GetSection().GetType();
714 if(SectionType::ToxContent
== eType
)
719 SwTextINetFormat
& rAttr
= *pFnd
;
720 OUString
sText( pTextNd
->GetExpandText(GetLayout(),
721 rAttr
.GetStart(), *rAttr
.GetEnd() - rAttr
.GetStart()) );
723 sText
= sText
.replaceAll("\x0a", "");
724 sText
= comphelper::string::strip(sText
, ' ');
726 if( !sText
.isEmpty() )
728 rArr
.emplace_back(sText
, rAttr
);
735 /// If the cursor is in an INetAttribute then it will be deleted completely (incl. hint text, the
736 /// latter is needed for drag & drop)
737 void SwEditShell::DelINetAttrWithText()
739 bool bRet
= SelectTextAttr( RES_TXTATR_INETFMT
, false );
741 DeleteSel(*GetCursor(), true);
744 /// Set the DontExpand flag at the text character attributes
745 bool SwEditShell::DontExpandFormat()
748 if( !IsTableMode() && GetDoc()->DontExpandFormat( *GetCursor()->GetPoint() ))
756 SvNumberFormatter
* SwEditShell::GetNumberFormatter()
758 return GetDoc()->GetNumberFormatter();
761 bool SwEditShell::ConvertFieldsToText()
764 bool bRet
= GetDoc()->ConvertFieldsToText(*GetLayout());
769 void SwEditShell::SetNumberingRestart()
773 // iterate over all text contents - body, frames, header, footer, footnote text
774 SwPaM
* pCursor
= GetCursor();
775 for(int i
= 0; i
< 2; i
++)
778 MakeFindRange(SwDocPositions::Start
, SwDocPositions::End
, pCursor
); // body content
780 MakeFindRange(SwDocPositions::OtherStart
, SwDocPositions::OtherEnd
, pCursor
); // extra content
781 SwPosition
* pSttPos
= pCursor
->Start(), *pEndPos
= pCursor
->End();
782 SwNodeOffset nCurrNd
= pSttPos
->GetNodeIndex();
783 SwNodeOffset nEndNd
= pEndPos
->GetNodeIndex();
784 if( nCurrNd
<= nEndNd
)
786 SwContentFrame
* pContentFrame
;
788 // iterate over all paragraphs
791 SwNode
* pNd
= GetDoc()->GetNodes()[ nCurrNd
];
792 switch( pNd
->GetNodeType() )
794 case SwNodeType::Text
:
795 pContentFrame
= static_cast<SwTextNode
*>(pNd
)->getLayoutFrame( GetLayout() );
796 if( nullptr != pContentFrame
)
798 // skip hidden frames - ignore protection!
799 if( !static_cast<SwTextFrame
*>(pContentFrame
)->IsHiddenNow() )
801 // if the node is numbered and the starting value of the numbering equals the
802 // start value of the numbering rule then set this value as hard starting value
806 SwTextNode
* pTextNd( pNd
->GetTextNode() );
807 SwNumRule
* pNumRule( pTextNd
->GetNumRule() );
809 // sw_redlinehide: not sure what this should do, only called from mail-merge
811 ( pNumRule
&& pTextNd
->GetNum() &&
812 ( pTextNd
->HasNumber() || pTextNd
->HasBullet() ) &&
813 pTextNd
->IsCountedInList() &&
814 !pTextNd
->IsListRestart() );
817 int nListLevel
= pTextNd
->GetActualListLevel();
822 if (nListLevel
>= MAXLEVEL
)
823 nListLevel
= MAXLEVEL
- 1;
825 bIsNodeNum
= pTextNd
->GetNum()->GetNumber() ==
826 pNumRule
->Get( o3tl::narrowing
<sal_uInt16
>(nListLevel
) ).GetStart();
830 // now set the start value as attribute
831 SwPosition
aCurrentNode(*pNd
);
832 GetDoc()->SetNumRuleStart( aCurrentNode
);
837 case SwNodeType::Section
:
838 // skip hidden sections - ignore protection!
839 if(static_cast<SwSectionNode
*>(pNd
)->GetSection().IsHidden() )
840 nCurrNd
= pNd
->EndOfSectionIndex();
845 bGoOn
= nCurrNd
< nEndNd
;
851 Pop(PopMode::DeleteCurrent
);
855 sal_Int32
SwEditShell::GetLineCount()
859 SwPaM
* pPam
= GetCursor();
860 SwNodeIndex
aStart( pPam
->GetPoint()->GetNode() );
862 SwContentFrame
*pContentFrame
= nullptr;
864 aStart
= SwNodeOffset(0);
866 while( nullptr != ( pCNd
= GetDoc()->GetNodes().GoNextSection(
867 &aStart
, true, false )) )
869 if( nullptr != ( pContentFrame
= pCNd
->getLayoutFrame( GetLayout() ) ) && pContentFrame
->IsTextFrame() )
871 SwTextFrame
*const pFrame(static_cast<SwTextFrame
*>(pContentFrame
));
872 nRet
= nRet
+ pFrame
->GetLineCount(TextFrameIndex(COMPLETE_STRING
));
873 if (GetLayout()->HasMergedParas())
875 if (auto const*const pMerged
= pFrame
->GetMergedPara())
877 aStart
= *pMerged
->pLastNode
;
885 tools::Long
SwEditShell::CompareDoc( const SwDoc
& rDoc
)
888 tools::Long nRet
= GetDoc()->CompareDoc( rDoc
);
893 tools::Long
SwEditShell::MergeDoc( const SwDoc
& rDoc
)
896 tools::Long nRet
= GetDoc()->MergeDoc( rDoc
);
901 const SwFootnoteInfo
& SwEditShell::GetFootnoteInfo() const
903 return GetDoc()->GetFootnoteInfo();
906 void SwEditShell::SetFootnoteInfo(const SwFootnoteInfo
& rInfo
)
909 CurrShell
aCurr( this );
910 GetDoc()->SetFootnoteInfo(rInfo
);
915 const SwEndNoteInfo
& SwEditShell::GetEndNoteInfo() const
917 return GetDoc()->GetEndNoteInfo();
920 void SwEditShell::SetEndNoteInfo(const SwEndNoteInfo
& rInfo
)
923 CurrShell
aCurr( this );
924 GetDoc()->SetEndNoteInfo(rInfo
);
928 const SwLineNumberInfo
& SwEditShell::GetLineNumberInfo() const
930 return GetDoc()->GetLineNumberInfo();
933 void SwEditShell::SetLineNumberInfo(const SwLineNumberInfo
& rInfo
)
936 CurrShell
aCurr( this );
937 GetDoc()->SetLineNumberInfo(rInfo
);
938 AddPaintRect( GetLayout()->getFrameArea() );
942 sal_uInt16
SwEditShell::GetLinkUpdMode() const
944 return getIDocumentSettingAccess().getLinkUpdateMode( false );
947 void SwEditShell::SetLinkUpdMode( sal_uInt16 nMode
)
949 getIDocumentSettingAccess().setLinkUpdateMode( nMode
);
952 // Interface for TextInputData - (for text input of japanese/chinese characters)
953 void SwEditShell::CreateExtTextInput(LanguageType eInputLanguage
)
955 SwExtTextInput
* pRet
= GetDoc()->CreateExtTextInput( *GetCursor() );
956 pRet
->SetLanguage(eInputLanguage
);
957 pRet
->SetOverwriteCursor( SwCursorShell::IsOverwriteCursor() );
960 OUString
SwEditShell::DeleteExtTextInput( bool bInsText
)
962 const SwPosition
& rPos
= *GetCursor()->GetPoint();
963 SwExtTextInput
* pDel
= GetDoc()->GetExtTextInput( rPos
.GetNode(),
964 rPos
.GetContentIndex() );
967 //JP 25.10.2001: under UNIX the cursor is moved before the Input-
968 // Engine event comes in. So take any - normally there
969 // exist only one at the time. -- Task 92016
970 pDel
= GetDoc()->GetExtTextInput();
976 SwUnoCursorHelper::GetTextFromPam(*pDel
, sTmp
);
978 CurrShell
aCurr( this );
980 pDel
->SetInsText( bInsText
);
981 SetOverwriteCursor( pDel
->IsOverwriteCursor() );
982 const SwPosition
aPos( *pDel
->GetPoint() );
983 GetDoc()->DeleteExtTextInput( pDel
);
985 // In this case, the "replace" function did not set the cursor
986 // to the original position. Therefore we have to do this manually.
987 if ( ! bInsText
&& IsOverwriteCursor() )
988 *GetCursor()->GetPoint() = aPos
;
995 void SwEditShell::SetExtTextInputData( const CommandExtTextInputData
& rData
)
997 SwPaM
* pCurrentCursor
= GetCursor();
998 const SwPosition
& rPos
= *pCurrentCursor
->GetPoint();
999 SwExtTextInput
* pInput
= GetDoc()->GetExtTextInput( rPos
.GetNode() );
1004 CurrShell
aCurr( this );
1006 if( !rData
.IsOnlyCursorChanged() )
1007 pInput
->SetInputData( rData
);
1009 const SwPosition
& rStt
= *pInput
->Start();
1010 const sal_Int32 nNewCursorPos
= rStt
.GetContentIndex() + rData
.GetCursorPos();
1014 const sal_Int32 nDiff
= nNewCursorPos
- rPos
.GetContentIndex();
1017 bool bLeft
= nDiff
< 0;
1018 sal_Int32 nMaxGuard
= std::abs(nDiff
);
1021 auto nOldPos
= pCurrentCursor
->GetPoint()->GetContentIndex();
1023 Left(1, SwCursorSkipMode::Chars
);
1025 Right(1, SwCursorSkipMode::Chars
);
1026 auto nNewPos
= pCurrentCursor
->GetPoint()->GetContentIndex();
1029 if (nNewPos
== nNewCursorPos
)
1032 if (nNewPos
== nOldPos
)
1034 // if there was no movement, we have failed for some reason
1035 SAL_WARN("sw.core", "IM cursor move failed");
1039 if (--nMaxGuard
== 0)
1041 // if it takes more cursor moves than there are utf-16 chars to move past
1042 // something has probably gone wrong
1043 SAL_WARN("sw.core", "IM abandoning cursor positioning");
1049 SetOverwriteCursor( rData
.IsCursorOverwrite() );
1053 if( !rData
.IsCursorVisible() ) // must be called after the EndAction
1058 void SwEditShell::TransliterateText( TransliterationFlags nType
)
1060 utl::TransliterationWrapper
aTrans( ::comphelper::getProcessComponentContext(), nType
);
1062 CurrShell
aCurr( this );
1064 SwPaM
* pCursor
= GetCursor();
1065 if( pCursor
->GetNext() != pCursor
)
1067 GetDoc()->GetIDocumentUndoRedo().StartUndo(SwUndoId::EMPTY
, nullptr);
1068 for(const SwPaM
& rPaM
: GetCursor()->GetRingContainer())
1070 if( rPaM
.HasMark() )
1071 GetDoc()->getIDocumentContentOperations().TransliterateText( rPaM
, aTrans
);
1073 GetDoc()->GetIDocumentUndoRedo().EndUndo(SwUndoId::EMPTY
, nullptr);
1076 GetDoc()->getIDocumentContentOperations().TransliterateText( *pCursor
, aTrans
);
1081 void SwEditShell::CountWords( SwDocStat
& rStat
) const
1083 for(const SwPaM
& rPaM
: GetCursor()->GetRingContainer())
1085 if( rPaM
.HasMark() )
1086 SwDoc::CountWords( rPaM
, rStat
);
1091 void SwEditShell::ApplyViewOptions( const SwViewOption
&rOpt
)
1093 SwCursorShell::StartAction();
1094 SwViewShell::ApplyViewOptions( rOpt
);
1095 SwEditShell::EndAction();
1098 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */