Version 7.6.3.2-android, tag libreoffice-7.6.3.2-android
[LibreOffice.git] / sw / source / core / edit / editsh.cxx
blob0dcec2742c668962297f9facfe887251868ac2f7
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
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>
30 #include <frmfmt.hxx>
31 #include <charfmt.hxx>
32 #include <doc.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>
39 #include <editsh.hxx>
40 #include <frame.hxx>
41 #include <cntfrm.hxx>
42 #include <pam.hxx>
43 #include <ndtxt.hxx>
44 #include <flyfrm.hxx>
45 #include <swundo.hxx>
46 #include <calc.hxx>
47 #include <ndgrf.hxx>
48 #include <ndole.hxx>
49 #include <txtfrm.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>
58 #include <calbck.hxx>
60 using namespace com::sun::star;
62 void SwEditShell::Insert( sal_Unicode c, bool bOnlyCurrCursor )
64 StartAllAction();
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() );
71 if( bOnlyCurrCursor )
72 break;
76 EndAllAction();
79 void SwEditShell::Insert2(const OUString &rStr, const bool bForceExpandHints )
81 StartAllAction();
83 const SwInsertFlags nInsertFlags =
84 bForceExpandHints
85 ? (SwInsertFlags::FORCEHINTEXPAND | SwInsertFlags::EMPTYEXPAND)
86 : SwInsertFlags::EMPTYEXPAND;
88 for(SwPaM& rCurrentCursor : getShellCursor( true )->GetRingContainer())
90 //OPT: GetSystemCharSet
91 GetDoc()->getIDocumentContentOperations().SetIME(false);
92 const bool bSuccess =
93 GetDoc()->getIDocumentContentOperations().InsertString(rCurrentCursor, rStr, nInsertFlags);
94 OSL_ENSURE( bSuccess, "Doc->Insert() failed." );
96 if (bSuccess)
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();
123 if ( nPrevPos )
124 --nPrevPos;
126 SwTextFrame const* pFrame;
127 SwScriptInfo *const pSI = SwScriptInfo::GetScriptInfo(
128 static_cast<SwTextNode&>(rNode), &pFrame, true);
130 sal_uInt8 nLevel = 0;
131 if ( ! pSI )
133 // seems to be an empty paragraph.
134 Point aPt;
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 );
147 else
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#
165 EndAllAction();
168 void SwEditShell::Overwrite(const OUString &rStr)
170 StartAllAction();
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() );
179 EndAllAction();
182 void SwEditShell::SplitNode( bool bAutoFormat, bool bCheckTableStart )
184 StartAllAction();
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);
196 if( bAutoFormat )
197 AutoFormatBySplitNode();
199 ClearTableBoxContent();
201 EndAllAction();
204 bool SwEditShell::AppendTextNode()
206 bool bRet = false;
207 StartAllAction();
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();
220 EndAllAction();
221 return bRet;
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();
233 return pGrfNode;
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 );
242 if ( pGrfNode )
244 pGrf = &(pGrfNode->GetGrf(bWait && GraphicType::Default == pGrfNode->GetGrf().GetType()));
246 return pGrf;
249 bool SwEditShell::IsLinkedGrfSwapOut() const
251 SwGrfNode *pGrfNode = GetGrfNode_();
252 return pGrfNode &&
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();
287 return true;
289 return false;
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 )
297 StartAllAction();
298 mxDoc->getIDocumentContentOperations().ReRead( *GetCursor(), rGrfName, rFltName, pGraphic );
299 EndAllAction();
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" );
308 if( pFormat )
309 SwDoc::GetGrfNms( *pFormat, pGrfName, pFltName );
310 else
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();
327 StartAllAction();
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();
333 EndAllAction();
336 void SwEditShell::ClearAutomaticContour()
338 SwNoTextNode *pNd = GetCursor()->GetPointNode().GetNoTextNode();
339 OSL_ENSURE( pNd, "is no NoTextNode!" );
340 if( pNd->HasAutomaticContour() )
342 StartAllAction();
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();
348 EndAllAction();
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
374 SwStartNode *pStNd;
375 SwNodeIndex aIdx( *GetNodes().GetEndOfAutotext().StartOfSectionNode(), 1 );
376 while ( nullptr != (pStNd = aIdx.GetNode().GetStartNode()) )
378 ++aIdx;
379 SwNode& rNd = aIdx.GetNode();
380 if( rNd.IsOLENode() &&
381 rName == static_cast<SwOLENode&>(rNd).GetChartTableName() &&
382 static_cast<SwOLENode&>(rNd).getLayoutFrame( GetLayout() ) )
383 return true;
385 aIdx.Assign( *pStNd->EndOfSectionNode(), + 1 );
387 return false;
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();
413 if (!pNd)
415 return OUString();
417 SwTextFrame const*const pFrame(static_cast<SwTextFrame*>(pNd->getLayoutFrame(GetLayout())));
418 if (pFrame)
420 return pFrame->GetCurWord(*rPaM.GetPoint());
422 return OUString();
425 void SwEditShell::UpdateDocStat( )
427 StartAllAction();
428 GetDoc()->getIDocumentStatistics().UpdateDocStat( false, true );
429 EndAllAction();
432 const SwDocStat& SwEditShell::GetUpdatedDocStat()
434 StartAllAction();
435 const SwDocStat &rRet = GetDoc()->getIDocumentStatistics().GetUpdatedDocStat( false, true );
436 EndAllAction();
437 return rRet;
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
460 OUString aText;
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();
467 bool bPrev = true;
468 SwPaM* pLast = pCursor;
469 SwPaM* pTemp = pCursor;
470 while ( bPrev )
472 SwPaM* pPrev2 = pTemp->GetPrev();
473 bPrev = ( pPrev2 && pPrev2 != pLast );
474 if ( bPrev )
476 pTemp = pPrev2;
477 SwNodeOffset nTemp = pPrev2->GetMark()->GetNodeIndex();
478 if ( nTemp < nIndex )
480 nIndex = nTemp;
481 pCursor = pPrev2;
487 SwTextNode const*const pTextNd = pCursor->GetMarkNode().GetTextNode();
488 if( pTextNd )
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?");
493 if (pTextFrame)
495 TextFrameIndex const nDropLen(pTextFrame->GetDropLen(TextFrameIndex(nChars)));
496 aText = pTextFrame->GetText().copy(0, sal_Int32(nDropLen));
500 return aText;
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()) )
508 return;
510 StartAllAction();
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())));
516 if (pTextFrame)
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." );
527 EndAllAction();
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();
539 if(pTextNd)
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;
549 sal_Int32 nPos = 0;
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() )
564 --nPos;
566 OUString sVar = aStr.copy( nTmpStt, nPos - nTmpStt );
567 if( !::FindOperator( sVar ) &&
568 (aCalc.GetVarTable().Find(sVar) ||
569 aCalc.VarLook( sVar )) )
571 if( !bValidFields )
573 GetDoc()->getIDocumentFieldsAccess().FieldsToCalc( aCalc,
574 pStart->GetNodeIndex(),
575 pStart->GetContentIndex() );
576 bValidFields = true;
578 aFormel.append("(" + aCalc.GetStrResult( aCalc.VarLook( sVar )->nValue ) + ")");
580 else
581 aFormel.append(sVar);
583 else
584 aFormel.append(ch);
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()));
603 // #i73788#
604 Graphic SwEditShell::GetIMapGraphic() const
606 // returns always a graphic if the cursor is in a Fly
607 CurrShell aCurr( const_cast<SwEditShell*>(this) );
608 Graphic aRet;
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())
621 aRet = *pGraphic;
623 else
625 SwFlyFrame* pFlyFrame = rNd.GetContentNode()->getLayoutFrame( GetLayout() )->FindFlyFrame();
626 if(pFlyFrame)
627 aRet = pFlyFrame->GetFormat()->MakeGraphic();
630 return aRet;
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() ) )
637 return false;
638 StartAllAction();
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(), ' '));
653 if( sText == rStr )
654 bDelText = bInsText = false;
656 else if( rFormat.GetValue() == rStr ) // Are Name and URL equal?
657 bDelText = bInsText = false;
659 if( bDelText )
660 Delete(true);
662 else if( pCursor->IsMultiSelection() && rFormat.GetValue() == rStr )
663 bInsText = false;
665 if( bInsText )
667 Insert2( rStr );
668 SetMark();
669 ExtendSelection( false, rStr.getLength() );
672 else
673 bInsText = false;
675 SetAttrItem( rFormat );
676 if (bInsText && !IsCursorPtAtEnd())
677 SwapPam();
678 if(!bKeepSelection)
679 ClearMark();
680 if( bInsText )
681 DontExpandFormat();
682 GetDoc()->GetIDocumentUndoRedo().EndUndo( SwUndoId::UI_INSERT_URLTXT, nullptr );
683 EndAllAction();
684 return true;
687 void SwEditShell::GetINetAttrs(SwGetINetAttrs& rArr, bool bIncludeInToxContent)
689 rArr.clear();
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()))
700 : nullptr);
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)
715 continue;
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 );
740 if( bRet )
741 DeleteSel(*GetCursor(), true);
744 /// Set the DontExpand flag at the text character attributes
745 bool SwEditShell::DontExpandFormat()
747 bool bRet = false;
748 if( !IsTableMode() && GetDoc()->DontExpandFormat( *GetCursor()->GetPoint() ))
750 bRet = true;
751 CallChgLnk();
753 return bRet;
756 SvNumberFormatter* SwEditShell::GetNumberFormatter()
758 return GetDoc()->GetNumberFormatter();
761 bool SwEditShell::ConvertFieldsToText()
763 StartAllAction();
764 bool bRet = GetDoc()->ConvertFieldsToText(*GetLayout());
765 EndAllAction();
766 return bRet;
769 void SwEditShell::SetNumberingRestart()
771 StartAllAction();
772 Push();
773 // iterate over all text contents - body, frames, header, footer, footnote text
774 SwPaM* pCursor = GetCursor();
775 for(int i = 0; i < 2; i++)
777 if(!i)
778 MakeFindRange(SwDocPositions::Start, SwDocPositions::End, pCursor); // body content
779 else
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;
787 bool bGoOn = true;
788 // iterate over all paragraphs
789 while( bGoOn )
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
804 // get the node num
805 // OD 2005-11-09
806 SwTextNode* pTextNd( pNd->GetTextNode() );
807 SwNumRule* pNumRule( pTextNd->GetNumRule() );
809 // sw_redlinehide: not sure what this should do, only called from mail-merge
810 bool bIsNodeNum =
811 ( pNumRule && pTextNd->GetNum() &&
812 ( pTextNd->HasNumber() || pTextNd->HasBullet() ) &&
813 pTextNd->IsCountedInList() &&
814 !pTextNd->IsListRestart() );
815 if (bIsNodeNum)
817 int nListLevel = pTextNd->GetActualListLevel();
819 if (nListLevel < 0)
820 nListLevel = 0;
822 if (nListLevel >= MAXLEVEL)
823 nListLevel = MAXLEVEL - 1;
825 bIsNodeNum = pTextNd->GetNum()->GetNumber() ==
826 pNumRule->Get( o3tl::narrowing<sal_uInt16>(nListLevel) ).GetStart();
828 if (bIsNodeNum)
830 // now set the start value as attribute
831 SwPosition aCurrentNode(*pNd);
832 GetDoc()->SetNumRuleStart( aCurrentNode );
836 break;
837 case SwNodeType::Section:
838 // skip hidden sections - ignore protection!
839 if(static_cast<SwSectionNode*>(pNd)->GetSection().IsHidden() )
840 nCurrNd = pNd->EndOfSectionIndex();
841 break;
842 default: break;
845 bGoOn = nCurrNd < nEndNd;
846 ++nCurrNd;
851 Pop(PopMode::DeleteCurrent);
852 EndAllAction();
855 sal_Int32 SwEditShell::GetLineCount()
857 sal_Int32 nRet = 0;
858 CalcLayout();
859 SwPaM* pPam = GetCursor();
860 SwNodeIndex aStart( pPam->GetPoint()->GetNode() );
861 SwContentNode* pCNd;
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;
882 return nRet;
885 tools::Long SwEditShell::CompareDoc( const SwDoc& rDoc )
887 StartAllAction();
888 tools::Long nRet = GetDoc()->CompareDoc( rDoc );
889 EndAllAction();
890 return nRet;
893 tools::Long SwEditShell::MergeDoc( const SwDoc& rDoc )
895 StartAllAction();
896 tools::Long nRet = GetDoc()->MergeDoc( rDoc );
897 EndAllAction();
898 return nRet;
901 const SwFootnoteInfo& SwEditShell::GetFootnoteInfo() const
903 return GetDoc()->GetFootnoteInfo();
906 void SwEditShell::SetFootnoteInfo(const SwFootnoteInfo& rInfo)
908 StartAllAction();
909 CurrShell aCurr( this );
910 GetDoc()->SetFootnoteInfo(rInfo);
911 CallChgLnk();
912 EndAllAction();
915 const SwEndNoteInfo& SwEditShell::GetEndNoteInfo() const
917 return GetDoc()->GetEndNoteInfo();
920 void SwEditShell::SetEndNoteInfo(const SwEndNoteInfo& rInfo)
922 StartAllAction();
923 CurrShell aCurr( this );
924 GetDoc()->SetEndNoteInfo(rInfo);
925 EndAllAction();
928 const SwLineNumberInfo& SwEditShell::GetLineNumberInfo() const
930 return GetDoc()->GetLineNumberInfo();
933 void SwEditShell::SetLineNumberInfo(const SwLineNumberInfo& rInfo)
935 StartAllAction();
936 CurrShell aCurr( this );
937 GetDoc()->SetLineNumberInfo(rInfo);
938 AddPaintRect( GetLayout()->getFrameArea() );
939 EndAllAction();
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() );
965 if( !pDel )
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();
972 OUString sRet;
973 if( pDel )
975 OUString sTmp;
976 SwUnoCursorHelper::GetTextFromPam(*pDel, sTmp);
977 sRet = sTmp;
978 CurrShell aCurr( this );
979 StartAllAction();
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;
990 EndAllAction();
992 return sRet;
995 void SwEditShell::SetExtTextInputData( const CommandExtTextInputData& rData )
997 SwPaM* pCurrentCursor = GetCursor();
998 const SwPosition& rPos = *pCurrentCursor->GetPoint();
999 SwExtTextInput* pInput = GetDoc()->GetExtTextInput( rPos.GetNode() );
1000 if( !pInput )
1001 return;
1003 StartAllAction();
1004 CurrShell aCurr( this );
1006 if( !rData.IsOnlyCursorChanged() )
1007 pInput->SetInputData( rData );
1008 // position cursor
1009 const SwPosition& rStt = *pInput->Start();
1010 const sal_Int32 nNewCursorPos = rStt.GetContentIndex() + rData.GetCursorPos();
1012 // ugly but works
1013 ShowCursor();
1014 const sal_Int32 nDiff = nNewCursorPos - rPos.GetContentIndex();
1015 if( nDiff != 0)
1017 bool bLeft = nDiff < 0;
1018 sal_Int32 nMaxGuard = std::abs(nDiff);
1019 while (true)
1021 auto nOldPos = pCurrentCursor->GetPoint()->GetContentIndex();
1022 if (bLeft)
1023 Left(1, SwCursorSkipMode::Chars);
1024 else
1025 Right(1, SwCursorSkipMode::Chars);
1026 auto nNewPos = pCurrentCursor->GetPoint()->GetContentIndex();
1028 // expected success
1029 if (nNewPos == nNewCursorPos)
1030 break;
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");
1036 break;
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");
1044 break;
1049 SetOverwriteCursor( rData.IsCursorOverwrite() );
1051 EndAllAction();
1053 if( !rData.IsCursorVisible() ) // must be called after the EndAction
1054 HideCursor();
1058 void SwEditShell::TransliterateText( TransliterationFlags nType )
1060 utl::TransliterationWrapper aTrans( ::comphelper::getProcessComponentContext(), nType );
1061 StartAllAction();
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);
1075 else
1076 GetDoc()->getIDocumentContentOperations().TransliterateText( *pCursor, aTrans );
1078 EndAllAction();
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: */