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 <sal/config.h>
22 #include <string_view>
25 #include "itrform2.hxx"
26 #include "porglue.hxx"
27 #include "redlnitr.hxx"
30 #include "pormulti.hxx"
31 #include "pordrop.hxx"
32 #include <breakit.hxx>
33 #include <unicode/uchar.h>
34 #include <com/sun/star/i18n/ScriptType.hpp>
35 #include <com/sun/star/i18n/CharacterIteratorMode.hpp>
36 #include <com/sun/star/i18n/WordType.hpp>
37 #include <com/sun/star/i18n/XBreakIterator.hpp>
39 #include <sal/log.hxx>
41 #include <editeng/adjustitem.hxx>
42 #include <editeng/charhiddenitem.hxx>
43 #include <svl/asiancfg.hxx>
44 #include <svl/languageoptions.hxx>
45 #include <tools/multisel.hxx>
46 #include <unotools/charclass.hxx>
47 #include <charfmt.hxx>
49 #include <fmtanchr.hxx>
50 #include <redline.hxx>
53 #include <swscanner.hxx>
54 #include <txatbase.hxx>
55 #include <IDocumentRedlineAccess.hxx>
56 #include <IDocumentSettingAccess.hxx>
57 #include <IDocumentContentOperations.hxx>
59 #include <sortedobjs.hxx>
60 #include <com/sun/star/frame/XModel.hpp>
61 #include <com/sun/star/text/XBookmarksSupplier.hpp>
62 #include <officecfg/Office/Common.hxx>
63 #include <comphelper/processfactory.hxx>
65 #include <unobookmark.hxx>
66 #include <unocrsrhelper.hxx>
68 #include <vcl/kernarray.hxx>
69 #include <editeng/ulspitem.hxx>
70 #include <com/sun/star/rdf/Statement.hpp>
71 #include <com/sun/star/rdf/URI.hpp>
72 #include <com/sun/star/rdf/URIs.hpp>
73 #include <com/sun/star/rdf/XDocumentMetadataAccess.hpp>
74 #include <com/sun/star/rdf/XLiteral.hpp>
75 #include <com/sun/star/text/XTextContent.hpp>
77 using namespace ::com::sun::star
;
78 using namespace i18n::ScriptType
;
80 #include <unicode/ubidi.h>
81 #include <i18nutil/scripttypedetector.hxx>
82 #include <i18nutil/unicode.hxx>
85 https://www.khtt.net/en/page/1821/the-big-kashida-secret
87 the rules of priorities that govern the addition of kashidas in Arabic text
88 made ... for ... Explorer 5.5 browser.
90 The kashida justification is based on a connection priority scheme that
91 decides where kashidas are put automatically.
93 This is how the software decides on kashida-inserting priorities:
94 1. First it looks for characters with the highest priority in each word,
95 which means kashida-extensions will only been used in one position in each
97 2. The kashida will be connected to the character with the highest priority.
98 3. If kashida connection opportunities are found with an equal level of
99 priority in one word, the kashida will be placed towards the end of the
102 The priority list of characters and the positioning is as follows:
103 1. after a kashida that is manually placed in the text by the user,
104 2. after a Seen or Sad (initial and medial form),
105 3. before the final form of Taa Marbutah, Haa, Dal,
106 4. before the final form of Alef, Tah Lam, Kaf and Gaf,
107 5. before the preceding medial Baa of Ra, Ya and Alef Maqsurah,
108 6. before the final form of Waw, Ain, Qaf and Fa,
109 7. before the final form of other characters that can be connected.
112 #define IS_JOINING_GROUP(c, g) ( u_getIntPropertyValue( (c), UCHAR_JOINING_GROUP ) == U_JG_##g )
113 #define isAinChar(c) IS_JOINING_GROUP((c), AIN)
114 #define isAlefChar(c) IS_JOINING_GROUP((c), ALEF)
115 #define isDalChar(c) IS_JOINING_GROUP((c), DAL)
116 #if U_ICU_VERSION_MAJOR_NUM >= 58
117 #define isFehChar(c) (IS_JOINING_GROUP((c), FEH) || IS_JOINING_GROUP((c), AFRICAN_FEH))
119 #define isFehChar(c) IS_JOINING_GROUP((c), FEH)
121 #define isGafChar(c) IS_JOINING_GROUP((c), GAF)
122 #define isHehChar(c) IS_JOINING_GROUP((c), HEH)
123 #define isKafChar(c) IS_JOINING_GROUP((c), KAF)
124 #define isLamChar(c) IS_JOINING_GROUP((c), LAM)
125 #if U_ICU_VERSION_MAJOR_NUM >= 58
126 #define isQafChar(c) (IS_JOINING_GROUP((c), QAF) || IS_JOINING_GROUP((c), AFRICAN_QAF))
128 #define isQafChar(c) IS_JOINING_GROUP((c), QAF)
130 #define isRehChar(c) IS_JOINING_GROUP((c), REH)
131 #define isTahChar(c) IS_JOINING_GROUP((c), TAH)
132 #define isTehMarbutaChar(c) IS_JOINING_GROUP((c), TEH_MARBUTA)
133 #define isWawChar(c) IS_JOINING_GROUP((c), WAW)
134 #define isSeenOrSadChar(c) (IS_JOINING_GROUP((c), SAD) || IS_JOINING_GROUP((c), SEEN))
136 // Beh and characters that behave like Beh in medial form.
137 static bool isBehChar(sal_Unicode cCh
)
140 switch (u_getIntPropertyValue(cCh
, UCHAR_JOINING_GROUP
))
144 #if U_ICU_VERSION_MAJOR_NUM >= 58
145 case U_JG_AFRICAN_NOON
:
150 case U_JG_BURUSHASKI_YEH_BARREE
:
161 // Yeh and characters that behave like Yeh in final form.
162 static bool isYehChar(sal_Unicode cCh
)
165 switch (u_getIntPropertyValue(cCh
, UCHAR_JOINING_GROUP
))
169 case U_JG_YEH_BARREE
:
170 case U_JG_BURUSHASKI_YEH_BARREE
:
171 case U_JG_YEH_WITH_TAIL
:
182 static bool isTransparentChar ( sal_Unicode cCh
)
184 return u_getIntPropertyValue( cCh
, UCHAR_JOINING_TYPE
) == U_JT_TRANSPARENT
;
187 // Checks if cCh + cNectCh builds a ligature (used for Kashidas)
188 static bool lcl_IsLigature( sal_Unicode cCh
, sal_Unicode cNextCh
)
191 return ( isLamChar ( cCh
) && isAlefChar ( cNextCh
));
194 // Checks if cCh is connectable to cPrevCh (used for Kashidas)
195 static bool lcl_ConnectToPrev( sal_Unicode cCh
, sal_Unicode cPrevCh
)
197 const int32_t nJoiningType
= u_getIntPropertyValue( cPrevCh
, UCHAR_JOINING_TYPE
);
198 bool bRet
= nJoiningType
!= U_JT_RIGHT_JOINING
&& nJoiningType
!= U_JT_NON_JOINING
;
200 // check for ligatures cPrevChar + cChar
202 bRet
= !lcl_IsLigature( cPrevCh
, cCh
);
207 static bool lcl_HasStrongLTR ( std::u16string_view rText
, sal_Int32 nStart
, sal_Int32 nEnd
)
209 for( sal_Int32 nCharIdx
= nStart
; nCharIdx
< nEnd
; ++nCharIdx
)
211 const UCharDirection nCharDir
= u_charDirection ( rText
[ nCharIdx
] );
212 if ( nCharDir
== U_LEFT_TO_RIGHT
||
213 nCharDir
== U_LEFT_TO_RIGHT_EMBEDDING
||
214 nCharDir
== U_LEFT_TO_RIGHT_OVERRIDE
)
220 // This is (meant to be) functionally equivalent to 'delete m_pNext' where
221 // deleting a SwLineLayout recursively deletes the owned m_pNext SwLineLayout.
223 // Here, instead of using a potentially deep stack, iterate over all the
224 // SwLineLayouts that would be deleted recursively and delete them linearly
225 void SwLineLayout::DeleteNext()
229 SwLineLayout
* pNext
= m_pNext
;
232 SwLineLayout
* pLastNext
= pNext
;
233 pNext
= pNext
->GetNext();
234 pLastNext
->SetNext(nullptr);
240 void SwLineLayout::Height(const SwTwips nNew
, const bool bText
)
242 SwPosSize::Height(nNew
);
244 m_nTextHeight
= nNew
;
247 // class SwLineLayout: This is the layout of a single line, which is made
248 // up of its dimension, the character count and the word spacing in the line.
249 // Line objects are managed in an own pool, in order to store them continuously
250 // in memory so that they are paged out together and don't fragment memory.
251 SwLineLayout::~SwLineLayout()
255 m_pLLSpaceAdd
.reset();
259 SwLinePortion
*SwLineLayout::Insert( SwLinePortion
*pIns
)
261 // First attribute change: copy mass and length from *pIns into the first
267 mpNextPortion
= SwTextPortion::CopyLinePortion(*this);
270 SetBlinking( false );
275 SetNextPortion( pIns
);
279 // Call with scope or we'll end up with recursion!
280 return mpNextPortion
->SwLinePortion::Insert( pIns
);
283 SwLinePortion
*SwLineLayout::Append( SwLinePortion
*pIns
)
285 // First attribute change: copy mass and length from *pIns into the first
288 mpNextPortion
= SwTextPortion::CopyLinePortion(*this);
289 // Call with scope or we'll end up with recursion!
290 return mpNextPortion
->SwLinePortion::Append( pIns
);
293 // For special treatment of empty lines
295 bool SwLineLayout::Format( SwTextFormatInfo
&rInf
)
298 return SwTextPortion::Format( rInf
);
300 Height( rInf
.GetTextHeight() );
304 // We collect all FlyPortions at the beginning of the line and make that a
306 SwMarginPortion
*SwLineLayout::CalcLeftMargin()
308 SwMarginPortion
*pLeft
= (GetNextPortion() && GetNextPortion()->IsMarginPortion()) ?
309 static_cast<SwMarginPortion
*>(GetNextPortion()) : nullptr;
310 if( !GetNextPortion() )
311 SetNextPortion(SwTextPortion::CopyLinePortion(*this));
314 pLeft
= new SwMarginPortion
;
315 pLeft
->SetNextPortion( GetNextPortion() );
316 SetNextPortion( pLeft
);
322 pLeft
->SetLen(TextFrameIndex(0));
323 pLeft
->SetAscent( 0 );
324 pLeft
->SetNextPortion( nullptr );
325 pLeft
->SetFixWidth(0);
328 SwLinePortion
*pPos
= pLeft
->GetNextPortion();
331 if( pPos
->IsFlyPortion() )
333 // The FlyPortion gets sucked out...
334 pLeft
->Join( static_cast<SwGluePortion
*>(pPos
) );
335 pPos
= pLeft
->GetNextPortion();
336 if( GetpKanaComp() && !GetKanaComp().empty() )
337 GetKanaComp().pop_front();
345 void SwLineLayout::InitSpaceAdd()
347 if ( !m_pLLSpaceAdd
)
350 SetLLSpaceAdd( 0, 0 );
353 void SwLineLayout::CreateSpaceAdd( const tools::Long nInit
)
355 m_pLLSpaceAdd
.reset( new std::vector
<tools::Long
> );
356 SetLLSpaceAdd( nInit
, 0 );
359 // #i3952# Returns true if there are only blanks in [nStt, nEnd[
360 // Used to implement IgnoreTabsAndBlanksForLineCalculation compat flag
361 static bool lcl_HasOnlyBlanks(std::u16string_view rText
, TextFrameIndex nStt
, TextFrameIndex nEnd
)
363 while ( nStt
< nEnd
)
365 switch (rText
[sal_Int32(nStt
++)])
367 case 0x0020: // SPACE
368 case 0x2002: // EN SPACE
369 case 0x2003: // EM SPACE
370 case 0x2005: // FOUR-PER-EM SPACE
371 case 0x3000: // IDEOGRAPHIC SPACE
380 // Swapped out from FormatLine()
381 void SwLineLayout::CalcLine( SwTextFormatter
&rLine
, SwTextFormatInfo
&rInf
)
383 const sal_uInt16 nLineWidth
= rInf
.RealWidth();
385 sal_uInt16 nFlyAscent
= 0;
386 sal_uInt16 nFlyHeight
= 0;
387 sal_uInt16 nFlyDescent
= 0;
389 // If this line has a clearing break, then this is the portion's height.
390 sal_uInt16 nBreakHeight
= 0;
392 bool bOnlyPostIts
= true;
395 bool bTmpDummy
= !GetLen();
396 SwFlyCntPortion
* pFlyCnt
= nullptr;
405 const bool bIgnoreBlanksAndTabsForLineHeightCalculation
=
406 rInf
.GetTextFrame()->GetDoc().getIDocumentSettingAccess().get(
407 DocumentSettingId::IGNORE_TABS_AND_BLANKS_FOR_LINE_CALCULATION
);
409 bool bHasBlankPortion
= false;
410 bool bHasOnlyBlankPortions
= true;
411 bool bHasFlyPortion
= false;
416 if( mpNextPortion
->IsBreakPortion() )
418 SetLen( mpNextPortion
->GetLen() );
424 const SwTwips nLineHeight
= Height();
425 Init( GetNextPortion() );
426 SwLinePortion
*pPos
= mpNextPortion
;
427 SwLinePortion
*pLast
= this;
428 sal_uInt16 nMaxDescent
= 0;
430 // A group is a segment in the portion chain of pCurr or a fixed
431 // portion spanning to the end or the next fixed portion
434 SAL_WARN_IF( PortionType::NONE
== pPos
->GetWhichPor(),
435 "sw.core", "SwLineLayout::CalcLine: don't use SwLinePortions !" );
437 // Null portions are eliminated. They can form if two FlyFrames
439 // coverity[deref_arg] - "Cut" means next "GetNextPortion" returns a different Portion
440 if( !pPos
->Compress() )
442 // Only take over Height and Ascent if the rest of the line
444 if( !pPos
->GetNextPortion() )
447 Height( pPos
->Height(), false );
449 SetAscent( pPos
->GetAscent() );
451 SwLinePortion
* pPortion
= pLast
->Cut( pPos
);
452 rLine
.ClearIfIsFirstOfBorderMerge(pPortion
);
454 pPos
= pLast
->GetNextPortion();
458 TextFrameIndex
const nPorSttIdx
= rInf
.GetLineStart() + mnLineLength
;
459 mnLineLength
+= pPos
->GetLen();
460 AddPrtWidth( pPos
->Width() );
463 if (bIgnoreBlanksAndTabsForLineHeightCalculation
&& !rInf
.GetLineStart())
465 if ( pPos
->InTabGrp() || pPos
->IsHolePortion() ||
466 ( pPos
->IsTextPortion() &&
467 lcl_HasOnlyBlanks( rInf
.GetText(), nPorSttIdx
, nPorSttIdx
+ pPos
->GetLen() ) ) )
470 pPos
= pPos
->GetNextPortion();
471 bHasBlankPortion
= true;
476 // Ignore drop portion height
477 // tdf#130804 ... and bookmark portions
478 if ((pPos
->IsDropPortion() && static_cast<SwDropPortion
*>(pPos
)->GetLines() > 1)
479 || pPos
->GetWhichPor() == PortionType::Bookmark
)
482 pPos
= pPos
->GetNextPortion();
486 bHasOnlyBlankPortions
= false;
488 // We had an attribute change: Sum up/build maxima of length and mass
490 SwTwips nPosHeight
= pPos
->Height();
491 SwTwips nPosAscent
= pPos
->GetAscent();
493 SAL_WARN_IF( nPosHeight
< nPosAscent
,
494 "sw.core", "SwLineLayout::CalcLine: bad ascent or height" );
496 if( pPos
->IsHangingPortion() )
499 rInf
.GetParaPortion()->SetMargin();
501 else if( !bHasFlyPortion
&& ( pPos
->IsFlyCntPortion() || pPos
->IsFlyPortion() ) )
502 bHasFlyPortion
= true;
504 // A line break portion only influences the height of the line in case it's the only
505 // portion in the line, except when it's a clearing break.
506 bool bClearingBreak
= false;
507 if (pPos
->IsBreakPortion())
509 auto pBreakPortion
= static_cast<SwBreakPortion
*>(pPos
);
510 bClearingBreak
= pBreakPortion
->GetClear() != SwLineBreakClear::NONE
;
511 nBreakHeight
= nPosHeight
;
513 if (!(pPos
->IsBreakPortion() && !bClearingBreak
) || !Height())
515 if (!pPos
->IsPostItsPortion()) bOnlyPostIts
= false;
517 if( bTmpDummy
&& !mnLineLength
)
519 if( pPos
->IsFlyPortion() )
521 if( nFlyHeight
< nPosHeight
)
522 nFlyHeight
= nPosHeight
;
523 if( nFlyAscent
< nPosAscent
)
524 nFlyAscent
= nPosAscent
;
525 if( nFlyDescent
< nPosHeight
- nPosAscent
)
526 nFlyDescent
= nPosHeight
- nPosAscent
;
530 if( pPos
->InNumberGrp() )
532 sal_uInt16 nTmp
= rInf
.GetFont()->GetAscent(
533 rInf
.GetVsh(), *rInf
.GetOut() );
534 if( nTmp
> nPosAscent
)
536 nPosHeight
+= nTmp
- nPosAscent
;
539 nTmp
= rInf
.GetFont()->GetHeight( rInf
.GetVsh(),
541 if( nTmp
> nPosHeight
)
544 Height( nPosHeight
, false );
545 mnAscent
= nPosAscent
;
546 nMaxDescent
= nPosHeight
- nPosAscent
;
549 else if( !pPos
->IsFlyPortion() )
551 if( Height() < nPosHeight
)
553 // Height is set to 0 when Init() is called.
554 if (bIgnoreBlanksAndTabsForLineHeightCalculation
&& pPos
->IsFlyCntPortion())
555 // Compat flag set: take the line height, if it's larger.
556 Height(std::max(nPosHeight
, nLineHeight
), false);
558 // Just care about the portion height.
559 Height(nPosHeight
, pPos
->IsTextPortion());
561 SwFlyCntPortion
* pAsFly(nullptr);
562 if(pPos
->IsFlyCntPortion())
563 pAsFly
= static_cast<SwFlyCntPortion
*>(pPos
);
564 if( pAsFly
|| ( pPos
->IsMultiPortion()
565 && static_cast<SwMultiPortion
*>(pPos
)->HasFlyInContent() ) )
566 rLine
.SetFlyInCntBase();
567 if(pAsFly
&& pAsFly
->GetAlign() != sw::LineAlign::NONE
)
569 pAsFly
->SetMax(false);
570 if( !pFlyCnt
|| pPos
->Height() > pFlyCnt
->Height() )
575 if( mnAscent
< nPosAscent
)
576 mnAscent
= nPosAscent
;
577 if( nMaxDescent
< nPosHeight
- nPosAscent
)
578 nMaxDescent
= nPosHeight
- nPosAscent
;
582 else if( pPos
->GetLen() )
585 if( !HasContent() && !pPos
->InNumberGrp() )
587 if ( pPos
->InExpGrp() )
590 if( pPos
->GetExpText( rInf
, aText
) && !aText
.isEmpty() )
593 else if( ( pPos
->InTextGrp() || pPos
->IsMultiPortion() ) &&
598 bTmpDummy
&= !HasContent() && ( !pPos
->Width() || pPos
->IsFlyPortion() );
601 pPos
= pPos
->GetNextPortion();
606 if( pFlyCnt
->Height() == Height() )
608 pFlyCnt
->SetMax( true );
609 if( Height() > nMaxDescent
+ mnAscent
)
611 if( sw::LineAlign::BOTTOM
== pFlyCnt
->GetAlign() )
612 mnAscent
= Height() - nMaxDescent
;
613 else if( sw::LineAlign::CENTER
== pFlyCnt
->GetAlign() )
614 mnAscent
= ( Height() + mnAscent
- nMaxDescent
) / 2;
616 pFlyCnt
->SetAscent( mnAscent
);
620 if( bTmpDummy
&& nFlyHeight
)
622 mnAscent
= nFlyAscent
;
623 if( nFlyDescent
> nFlyHeight
- nFlyAscent
)
624 Height( nFlyHeight
+ nFlyDescent
, false );
627 if (nBreakHeight
> nFlyHeight
)
629 // The line has no content, but it has a clearing break: then the line
630 // height is not only the intersection of the fly and line's rectangle, but
631 // also includes the clearing break's height.
632 Height(nBreakHeight
, false);
636 Height(nFlyHeight
, false);
640 else if( nMaxDescent
> Height() - mnAscent
)
641 Height( nMaxDescent
+ mnAscent
, false );
643 if( bOnlyPostIts
&& !( bHasBlankPortion
&& bHasOnlyBlankPortions
) )
645 Height( rInf
.GetFont()->GetHeight( rInf
.GetVsh(), *rInf
.GetOut() ) );
646 mnAscent
= rInf
.GetFont()->GetAscent( rInf
.GetVsh(), *rInf
.GetOut() );
652 SetContent( !bTmpDummy
);
655 if ( bIgnoreBlanksAndTabsForLineHeightCalculation
&&
656 lcl_HasOnlyBlanks( rInf
.GetText(), rInf
.GetLineStart(), rInf
.GetLineStart() + GetLen() ) )
658 bHasBlankPortion
= true;
662 // #i3952# Whitespace does not increase line height
663 if ( bHasBlankPortion
&& bHasOnlyBlankPortions
)
665 sal_uInt16 nTmpAscent
= GetAscent();
666 sal_uInt16 nTmpHeight
= Height();
667 rLine
.GetAttrHandler().GetDefaultAscentAndHeight( rInf
.GetVsh(), *rInf
.GetOut(), nTmpAscent
, nTmpHeight
);
668 if (nTmpAscent
< GetAscent() || GetAscent() <= 0)
669 SetAscent(nTmpAscent
);
670 if (nTmpHeight
< Height() || Height() <= 0)
671 Height(nTmpHeight
, false);
675 if( nLineWidth
< Width() )
677 SAL_WARN_IF( nLineWidth
< Width(), "sw.core", "SwLineLayout::CalcLine: line is bursting" );
678 SetDummy( bTmpDummy
);
679 std::pair
<SwTextNode
const*, sal_Int32
> const start(
680 rInf
.GetTextFrame()->MapViewToModel(rLine
.GetStart()));
681 std::pair
<SwTextNode
const*, sal_Int32
> const end(
682 rInf
.GetTextFrame()->MapViewToModel(rLine
.GetEnd()));
683 bool bHasRedline
= rLine
.GetRedln();
686 OUString sRedlineText
;
688 enum RedlineType eRedlineEnd
;
689 bHasRedline
= rLine
.GetRedln()->CheckLine(start
.first
->GetIndex(), start
.second
,
690 end
.first
->GetIndex(), end
.second
, sRedlineText
, bHasRedlineEnd
, eRedlineEnd
);
693 SetRedlineText( sRedlineText
);
695 SetRedlineEnd( bHasRedlineEnd
);
696 if( eRedlineEnd
!= RedlineType::None
)
697 SetRedlineEndType( eRedlineEnd
);
700 SetRedline( bHasRedline
);
702 // redlining: set crossing out for deleted anchored objects
703 if ( !bHasFlyPortion
)
706 SwLinePortion
*pPos
= mpNextPortion
;
707 TextFrameIndex nLineLength
;
710 TextFrameIndex
const nPorSttIdx
= rInf
.GetLineStart() + nLineLength
;
711 nLineLength
+= pPos
->GetLen();
712 // anchored as characters
713 if( pPos
->IsFlyCntPortion() )
715 bool bDeleted
= false;
716 size_t nAuthor
= std::string::npos
;
719 OUString sRedlineText
;
721 enum RedlineType eRedlineEnd
;
722 std::pair
<SwTextNode
const*, sal_Int32
> const flyStart(
723 rInf
.GetTextFrame()->MapViewToModel(nPorSttIdx
));
724 bool bHasFlyRedline
= rLine
.GetRedln()->CheckLine(flyStart
.first
->GetIndex(),
725 flyStart
.second
, flyStart
.first
->GetIndex(), flyStart
.second
, sRedlineText
,
726 bHasRedlineEnd
, eRedlineEnd
, /*pAuthorAtPos=*/&nAuthor
);
727 bDeleted
= bHasFlyRedline
&& eRedlineEnd
== RedlineType::Delete
;
729 static_cast<SwFlyCntPortion
*>(pPos
)->SetDeleted(bDeleted
);
730 static_cast<SwFlyCntPortion
*>(pPos
)->SetAuthor(nAuthor
);
732 // anchored to characters
733 else if ( pPos
->IsFlyPortion() )
735 const IDocumentRedlineAccess
& rIDRA
=
736 rInf
.GetTextFrame()->GetDoc().getIDocumentRedlineAccess();
737 SwSortedObjs
*pObjs
= rInf
.GetTextFrame()->GetDrawObjs();
738 if ( pObjs
&& IDocumentRedlineAccess::IsShowChanges( rIDRA
.GetRedlineFlags() ) )
740 for ( size_t i
= 0; rInf
.GetTextFrame()->GetDrawObjs() && i
< pObjs
->size(); ++i
)
742 SwAnchoredObject
* pAnchoredObj
= (*rInf
.GetTextFrame()->GetDrawObjs())[i
];
743 if ( auto pFly
= pAnchoredObj
->DynCastFlyFrame() )
745 bool bDeleted
= false;
746 size_t nAuthor
= std::string::npos
;
747 const SwFormatAnchor
& rAnchor
= pAnchoredObj
->GetFrameFormat().GetAnchor();
748 if ( rAnchor
.GetAnchorId() == RndStdIds::FLY_AT_CHAR
)
750 SwPosition aAnchor
= *rAnchor
.GetContentAnchor();
751 SwRedlineTable::size_type n
= 0;
752 const SwRangeRedline
* pFnd
=
753 rIDRA
.GetRedlineTable().FindAtPosition( aAnchor
, n
);
754 if ( pFnd
&& RedlineType::Delete
== pFnd
->GetType() )
757 nAuthor
= pFnd
->GetAuthor();
760 pFly
->SetDeleted(bDeleted
);
761 pFly
->SetAuthor(nAuthor
);
766 pPos
= pPos
->GetNextPortion();
770 // #i47162# - add optional parameter <_bNoFlyCntPorAndLinePor>
771 // to control, if the fly content portions and line portion are considered.
772 void SwLineLayout::MaxAscentDescent( SwTwips
& _orAscent
,
774 SwTwips
& _orObjAscent
,
775 SwTwips
& _orObjDescent
,
776 const SwLinePortion
* _pDontConsiderPortion
,
777 const bool _bNoFlyCntPorAndLinePor
) const
784 const SwLinePortion
* pTmpPortion
= this;
785 if ( !pTmpPortion
->GetLen() && pTmpPortion
->GetNextPortion() )
787 pTmpPortion
= pTmpPortion
->GetNextPortion();
790 while ( pTmpPortion
)
792 if ( !pTmpPortion
->IsBreakPortion() && !pTmpPortion
->IsFlyPortion() &&
793 // tdf#130804 ignore bookmark portions
794 pTmpPortion
->GetWhichPor() != PortionType::Bookmark
&&
795 ( !_bNoFlyCntPorAndLinePor
||
796 ( !pTmpPortion
->IsFlyCntPortion() &&
797 !(pTmpPortion
== this && pTmpPortion
->GetNextPortion() ) ) ) )
799 SwTwips nPortionAsc
= pTmpPortion
->GetAscent();
800 SwTwips nPortionDesc
= pTmpPortion
->Height() - nPortionAsc
;
802 const bool bFlyCmp
= pTmpPortion
->IsFlyCntPortion() ?
803 static_cast<const SwFlyCntPortion
*>(pTmpPortion
)->IsMax() :
804 ( pTmpPortion
!= _pDontConsiderPortion
);
808 _orObjAscent
= std::max( _orObjAscent
, nPortionAsc
);
809 _orObjDescent
= std::max( _orObjDescent
, nPortionDesc
);
812 if ( !pTmpPortion
->IsFlyCntPortion() && !pTmpPortion
->IsGrfNumPortion() )
814 _orAscent
= std::max( _orAscent
, nPortionAsc
);
815 _orDescent
= std::max( _orDescent
, nPortionDesc
);
818 pTmpPortion
= pTmpPortion
->GetNextPortion();
822 void SwLineLayout::dumpAsXml(xmlTextWriterPtr pWriter
, const OUString
& rText
,
823 TextFrameIndex
& nOffset
) const
825 (void)xmlTextWriterStartElement(pWriter
, BAD_CAST("SwLineLayout"));
826 dumpAsXmlAttributes(pWriter
, rText
, nOffset
);
829 (void)xmlTextWriterEndElement(pWriter
);
832 void SwLineLayout::ResetFlags()
834 m_bFormatAdj
= m_bDummy
= m_bEndHyph
= m_bMidHyph
= m_bFly
835 = m_bRest
= m_bBlinking
= m_bClipping
= m_bContent
= m_bRedline
836 = m_bRedlineEnd
= m_bForcedLeftMargin
= m_bHanging
= false;
837 m_eRedlineEnd
= RedlineType::None
;
840 SwLineLayout::SwLineLayout()
841 : m_pNext( nullptr ),
844 m_bUnderscore( false )
847 SetWhichPor( PortionType::Lay
);
850 SwLinePortion
*SwLineLayout::GetFirstPortion() const
852 const SwLinePortion
*pRet
= mpNextPortion
? mpNextPortion
: this;
853 return const_cast<SwLinePortion
*>(pRet
);
856 SwCharRange
&SwCharRange::operator+=(const SwCharRange
&rRange
)
858 if (TextFrameIndex(0) != rRange
.m_nLen
)
860 if (TextFrameIndex(0) == m_nLen
) {
861 m_nStart
= rRange
.m_nStart
;
862 m_nLen
= rRange
.m_nLen
;
865 if(rRange
.m_nStart
+ rRange
.m_nLen
> m_nStart
+ m_nLen
) {
866 m_nLen
= rRange
.m_nStart
+ rRange
.m_nLen
- m_nStart
;
868 if(rRange
.m_nStart
< m_nStart
) {
869 m_nLen
+= m_nStart
- rRange
.m_nStart
;
870 m_nStart
= rRange
.m_nStart
;
877 SwScriptInfo::SwScriptInfo()
878 : m_nInvalidityPos(0)
883 SwScriptInfo::~SwScriptInfo()
887 // Converts i18n Script Type (LATIN, ASIAN, COMPLEX, WEAK) to
888 // Sw Script Types (SwFontScript::Latin, SwFontScript::CJK, SwFontScript::CTL), used to identify the font
889 static SwFontScript
lcl_ScriptToFont(sal_uInt16
const nScript
)
892 case i18n::ScriptType::LATIN
: return SwFontScript::Latin
;
893 case i18n::ScriptType::ASIAN
: return SwFontScript::CJK
;
894 case i18n::ScriptType::COMPLEX
: return SwFontScript::CTL
;
897 OSL_FAIL( "Somebody tells lies about the script type!" );
898 return SwFontScript::Latin
;
901 SwFontScript
SwScriptInfo::WhichFont(TextFrameIndex
const nIdx
) const
903 const sal_uInt16
nScript(ScriptType(nIdx
));
904 return lcl_ScriptToFont(nScript
);
907 SwFontScript
SwScriptInfo::WhichFont(sal_Int32 nIdx
, OUString
const& rText
)
909 const sal_uInt16
nScript(g_pBreakIt
->GetRealScriptOfText(rText
, nIdx
));
910 return lcl_ScriptToFont(nScript
);
913 static Color
getBookmarkColor(const SwTextNode
& rNode
, const sw::mark::IBookmark
* pBookmark
)
915 // search custom color in metadata, otherwise use COL_TRANSPARENT;
916 Color c
= COL_TRANSPARENT
;
920 SwDoc
& rDoc
= const_cast<SwDoc
&>(rNode
.GetDoc());
921 const uno::Reference
< text::XTextContent
> xRef
= SwXBookmark::CreateXBookmark(rDoc
,
922 const_cast<sw::mark::IMark
*>(static_cast<const sw::mark::IMark
*>(pBookmark
)));
923 const css::uno::Reference
<css::rdf::XResource
> xSubject(xRef
, uno::UNO_QUERY
);
924 uno::Reference
<frame::XModel
> xModel
= rDoc
.GetDocShell()->GetBaseModel();
926 static uno::Reference
< uno::XComponentContext
> xContext(
927 ::comphelper::getProcessComponentContext());
929 static uno::Reference
< rdf::XURI
> xODF_SHADING(
930 rdf::URI::createKnown(xContext
, rdf::URIs::LO_EXT_SHADING
), uno::UNO_SET_THROW
);
932 uno::Reference
<rdf::XDocumentMetadataAccess
> xDocumentMetadataAccess(
933 rDoc
.GetDocShell()->GetBaseModel(), uno::UNO_QUERY
);
934 const uno::Reference
<rdf::XRepository
>& xRepository
=
935 xDocumentMetadataAccess
->getRDFRepository();
936 const uno::Reference
<container::XEnumeration
> xEnum(
937 xRepository
->getStatements(xSubject
, xODF_SHADING
, nullptr), uno::UNO_SET_THROW
);
940 if ( xEnum
->hasMoreElements() && (xEnum
->nextElement() >>= stmt
) )
942 const uno::Reference
<rdf::XLiteral
> xObject(stmt
.Object
, uno::UNO_QUERY
);
944 c
= Color::STRtoRGB(xObject
->getValue());
947 catch (const lang::IllegalArgumentException
&)
954 static void InitBookmarks(
955 std::optional
<std::vector
<sw::Extent
>::const_iterator
> oPrevIter
,
956 std::vector
<sw::Extent
>::const_iterator iter
,
957 std::vector
<sw::Extent
>::const_iterator
const end
,
958 TextFrameIndex nOffset
,
959 std::vector
<std::pair
<sw::mark::IBookmark
const*, SwScriptInfo::MarkKind
>> & rBookmarks
,
960 std::vector
<std::tuple
<TextFrameIndex
, SwScriptInfo::MarkKind
, Color
, OUString
>> & o_rBookmarks
)
962 SwTextNode
const*const pNode(iter
->pNode
);
963 for (auto const& it
: rBookmarks
)
965 assert(iter
->pNode
== pNode
|| pNode
->GetIndex() < iter
->pNode
->GetIndex());
966 assert(!oPrevIter
|| (*oPrevIter
)->pNode
->GetIndex() <= pNode
->GetIndex());
968 // search for custom bookmark boundary mark color
969 Color c
= getBookmarkColor(*pNode
, it
.first
);
973 case SwScriptInfo::MarkKind::Start
:
975 // SwUndoSaveContent::DelContentIndex() is rather messy but
976 // apparently bookmarks "on the edge" are deleted if
977 // * point: equals start-of-selection (not end-of-selection)
978 // * expanded: one position equals edge of selection
979 // and other does not (is inside)
980 // interesting case: if end[/start] of the mark is on the
981 // start of first[/end of last] extent, and the other one
982 // is outside this merged paragraph, is it deleted or not?
983 // assume "no" because the line break it contains isn't deleted.
984 SwPosition
const& rStart(it
.first
->GetMarkStart());
985 SwPosition
const& rEnd(it
.first
->GetMarkEnd());
986 assert(&rStart
.GetNode() == pNode
);
989 if (&rStart
.GetNode() != iter
->pNode
// iter moved to next node
990 || rStart
.GetContentIndex() < iter
->nStart
)
992 if (rEnd
.GetNodeIndex() < iter
->pNode
->GetIndex()
993 || (&rEnd
.GetNode() == iter
->pNode
&& rEnd
.GetContentIndex() <= iter
->nStart
))
995 break; // deleted - skip it
999 o_rBookmarks
.emplace_back(nOffset
, it
.second
, c
, it
.first
->GetName());
1003 else if (rStart
.GetContentIndex() <= iter
->nEnd
)
1005 auto const iterNext(iter
+ 1);
1006 if (rStart
.GetContentIndex() == iter
->nEnd
1008 ? &rEnd
.GetNode() == iter
->pNode
1009 : (rEnd
.GetNodeIndex() < iterNext
->pNode
->GetIndex()
1010 || (&rEnd
.GetNode() == iterNext
->pNode
&& rEnd
.GetContentIndex() < iterNext
->nStart
))))
1012 break; // deleted - skip it
1016 o_rBookmarks
.emplace_back(
1017 nOffset
+ TextFrameIndex(rStart
.GetContentIndex() - iter
->nStart
),
1018 it
.second
, c
, it
.first
->GetName());
1024 nOffset
+= TextFrameIndex(iter
->nEnd
- iter
->nStart
);
1026 ++iter
; // bookmarks are sorted...
1031 if (pNode
->GetIndex() < rEnd
.GetNodeIndex()) // pNode is last node of merged
1033 break; // deleted - skip it
1037 o_rBookmarks
.emplace_back(nOffset
, it
.second
, c
, it
.first
->GetName());
1042 case SwScriptInfo::MarkKind::End
:
1044 SwPosition
const& rEnd(it
.first
->GetMarkEnd());
1045 assert(&rEnd
.GetNode() == pNode
);
1049 || &rEnd
.GetNode() != iter
->pNode
// iter moved to next node
1050 || rEnd
.GetContentIndex() <= iter
->nStart
)
1052 SwPosition
const& rStart(it
.first
->GetMarkStart());
1053 // oPrevIter may point to pNode or a preceding node
1055 ? ((*oPrevIter
)->pNode
->GetIndex() < rStart
.GetNodeIndex()
1056 || ((*oPrevIter
)->pNode
== &rStart
.GetNode()
1057 && ((iter
!= end
&& &rEnd
.GetNode() == iter
->pNode
&& rEnd
.GetContentIndex() == iter
->nStart
)
1058 ? (*oPrevIter
)->nEnd
< rStart
.GetContentIndex()
1059 : (*oPrevIter
)->nEnd
<= rStart
.GetContentIndex())))
1060 : rStart
.GetNode() == rEnd
.GetNode())
1062 break; // deleted - skip it
1066 o_rBookmarks
.emplace_back(nOffset
, it
.second
, c
, it
.first
->GetName());
1070 else if (rEnd
.GetContentIndex() <= iter
->nEnd
)
1072 o_rBookmarks
.emplace_back(
1073 nOffset
+ TextFrameIndex(rEnd
.GetContentIndex() - iter
->nStart
),
1074 it
.second
, c
, it
.first
->GetName());
1079 nOffset
+= TextFrameIndex(iter
->nEnd
- iter
->nStart
);
1086 case SwScriptInfo::MarkKind::Point
:
1088 SwPosition
const& rPos(it
.first
->GetMarkPos());
1089 assert(&rPos
.GetNode() == pNode
);
1092 if (&rPos
.GetNode() != iter
->pNode
// iter moved to next node
1093 || rPos
.GetContentIndex() < iter
->nStart
)
1095 break; // deleted - skip it
1097 else if (rPos
.GetContentIndex() <= iter
->nEnd
)
1099 if (rPos
.GetContentIndex() == iter
->nEnd
1100 && rPos
.GetContentIndex() != iter
->pNode
->Len())
1102 break; // deleted - skip it
1106 o_rBookmarks
.emplace_back(
1107 nOffset
+ TextFrameIndex(rPos
.GetContentIndex() - iter
->nStart
),
1108 it
.second
, c
, it
.first
->GetName());
1114 nOffset
+= TextFrameIndex(iter
->nEnd
- iter
->nStart
);
1124 break; // remaining marks are hidden
1129 // searches for script changes in rText and stores them
1130 void SwScriptInfo::InitScriptInfo(const SwTextNode
& rNode
,
1131 sw::MergedPara
const*const pMerged
)
1133 InitScriptInfo( rNode
, pMerged
, m_nDefaultDir
== UBIDI_RTL
);
1136 void SwScriptInfo::InitScriptInfo(const SwTextNode
& rNode
,
1137 sw::MergedPara
const*const pMerged
, bool bRTL
)
1139 assert(g_pBreakIt
&& g_pBreakIt
->GetBreakIter().is());
1141 const OUString
& rText(pMerged
? pMerged
->mergedText
: rNode
.GetText());
1143 // HIDDEN TEXT INFORMATION
1145 m_Bookmarks
.clear();
1146 m_HiddenChg
.clear();
1149 SwTextNode
const* pNode(nullptr);
1150 TextFrameIndex
nOffset(0);
1151 std::optional
<std::vector
<sw::Extent
>::const_iterator
> oPrevIter
;
1152 for (auto iter
= pMerged
->extents
.begin(); iter
!= pMerged
->extents
.end();
1155 if (iter
->pNode
== pNode
)
1157 nOffset
+= TextFrameIndex(iter
->nEnd
- iter
->nStart
);
1159 continue; // skip extents at end of previous node
1161 pNode
= iter
->pNode
;
1162 Range
aRange( 0, pNode
->Len() > 0 ? pNode
->Len() - 1 : 0 );
1163 MultiSelection
aHiddenMulti( aRange
);
1164 std::vector
<std::pair
<sw::mark::IBookmark
const*, MarkKind
>> bookmarks
;
1165 CalcHiddenRanges(*pNode
, aHiddenMulti
, &bookmarks
);
1167 InitBookmarks(oPrevIter
, iter
, pMerged
->extents
.end(), nOffset
, bookmarks
, m_Bookmarks
);
1169 for (sal_Int32 i
= 0; i
< aHiddenMulti
.GetRangeCount(); ++i
)
1171 const Range
& rRange
= aHiddenMulti
.GetRange( i
);
1172 const sal_Int32 nStart
= rRange
.Min();
1173 const sal_Int32 nEnd
= rRange
.Max() + 1;
1174 bool isStartHandled(false);
1175 ::std::optional
<sal_Int32
> oExtend
;
1177 if (nEnd
<= iter
->nStart
)
1178 { // entirely in gap, skip this hidden range
1184 if (!isStartHandled
&& nStart
<= iter
->nEnd
)
1186 isStartHandled
= true;
1187 if (nStart
<= iter
->nStart
&& !m_HiddenChg
.empty()
1188 && m_HiddenChg
.back() == nOffset
)
1190 // previous one went until end of extent, extend it
1191 oExtend
.emplace(::std::min(iter
->nEnd
, nEnd
) - ::std::max(iter
->nStart
, nStart
));
1195 m_HiddenChg
.push_back(nOffset
+ TextFrameIndex(::std::max(nStart
- iter
->nStart
, sal_Int32(0))));
1200 *oExtend
+= ::std::min(iter
->nEnd
, nEnd
) - iter
->nStart
;
1202 if (nEnd
<= iter
->nEnd
)
1206 m_HiddenChg
.back() += TextFrameIndex(*oExtend
);
1210 m_HiddenChg
.push_back(nOffset
+ TextFrameIndex(::std::max(nEnd
- iter
->nStart
, sal_Int32(0))));
1212 break; // iterate to next hidden range
1214 nOffset
+= TextFrameIndex(iter
->nEnd
- iter
->nStart
);
1217 while (iter
!= pMerged
->extents
.end() && iter
->pNode
== pNode
);
1218 if (iter
== pMerged
->extents
.end() || iter
->pNode
!= pNode
)
1224 m_HiddenChg
.back() += TextFrameIndex(*oExtend
);
1228 m_HiddenChg
.push_back(nOffset
);
1230 } // else: beyond last extent in node, ignore
1231 break; // skip hidden ranges beyond last extent in node
1238 Range
aRange( 0, !rText
.isEmpty() ? rText
.getLength() - 1 : 0 );
1239 MultiSelection
aHiddenMulti( aRange
);
1240 std::vector
<std::pair
<sw::mark::IBookmark
const*, MarkKind
>> bookmarks
;
1241 CalcHiddenRanges(rNode
, aHiddenMulti
, &bookmarks
);
1243 for (auto const& it
: bookmarks
)
1245 // don't show __RefHeading__ bookmarks, which are hidden in Navigator, too
1246 // (They are inserted automatically e.g. with the ToC at the beginning of
1248 if (it
.first
->GetName().startsWith(
1249 IDocumentMarkAccess::GetCrossRefHeadingBookmarkNamePrefix()))
1254 // search for custom bookmark boundary mark color
1255 Color c
= getBookmarkColor(rNode
, it
.first
);
1259 case MarkKind::Start
:
1260 m_Bookmarks
.emplace_back(TextFrameIndex(it
.first
->GetMarkStart().GetContentIndex()), it
.second
, c
, it
.first
->GetName());
1263 m_Bookmarks
.emplace_back(TextFrameIndex(it
.first
->GetMarkEnd().GetContentIndex()), it
.second
, c
, it
.first
->GetName());
1265 case MarkKind::Point
:
1266 m_Bookmarks
.emplace_back(TextFrameIndex(it
.first
->GetMarkPos().GetContentIndex()), it
.second
, c
, it
.first
->GetName());
1271 m_HiddenChg
.reserve( aHiddenMulti
.GetRangeCount() * 2 );
1272 for (sal_Int32 i
= 0; i
< aHiddenMulti
.GetRangeCount(); ++i
)
1274 const Range
& rRange
= aHiddenMulti
.GetRange( i
);
1275 const sal_Int32 nStart
= rRange
.Min();
1276 const sal_Int32 nEnd
= rRange
.Max() + (rText
.isEmpty() ? 0 : 1);
1278 m_HiddenChg
.push_back( TextFrameIndex(nStart
) );
1279 m_HiddenChg
.push_back( TextFrameIndex(nEnd
) );
1283 // SCRIPT AND SCRIPT RELATED INFORMATION
1285 TextFrameIndex nChg
= m_nInvalidityPos
;
1287 // COMPLETE_STRING means the data structure is up to date
1288 m_nInvalidityPos
= TextFrameIndex(COMPLETE_STRING
);
1290 // this is the default direction
1291 m_nDefaultDir
= static_cast<sal_uInt8
>(bRTL
? UBIDI_RTL
: UBIDI_LTR
);
1293 // counter for script info arrays
1295 // counter for compression information arrays
1296 size_t nCntComp
= 0;
1297 // counter for kashida array
1298 size_t nCntKash
= 0;
1300 sal_Int16 nScript
= i18n::ScriptType::LATIN
;
1303 const CharCompressType aCompEnum
= rNode
.getIDocumentSettingAccess()->getCharacterCompressionType();
1305 auto const& rParaItems((pMerged
? *pMerged
->pParaPropsNode
: rNode
).GetSwAttrSet());
1306 // justification type
1307 const bool bAdjustBlock
= SvxAdjust::Block
== rParaItems
.GetAdjust().GetAdjust();
1309 // FIND INVALID RANGES IN SCRIPT INFO ARRAYS:
1313 // if change position = 0 we do not use any data from the arrays
1314 // because by deleting all characters of the first group at the beginning
1315 // of a paragraph nScript is set to a wrong value
1316 SAL_WARN_IF( !CountScriptChg(), "sw.core", "Where're my changes of script?" );
1317 while( nCnt
< CountScriptChg() )
1319 if ( nChg
> GetScriptChg( nCnt
) )
1323 nScript
= GetScriptType( nCnt
);
1327 if( CharCompressType::NONE
!= aCompEnum
)
1329 while( nCntComp
< CountCompChg() )
1331 if ( nChg
<= GetCompStart( nCntComp
) )
1338 while( nCntKash
< CountKashida() )
1340 if ( nChg
<= GetKashida( nCntKash
) )
1347 // ADJUST nChg VALUE:
1349 // by stepping back one position we know that we are inside a group
1350 // declared as an nScript group
1354 const TextFrameIndex nGrpStart
= nCnt
? GetScriptChg(nCnt
- 1) : TextFrameIndex(0);
1356 // we go back in our group until we reach the first character of
1358 while ( nChg
> nGrpStart
&&
1359 nScript
!= g_pBreakIt
->GetBreakIter()->getScriptType(rText
, sal_Int32(nChg
)))
1362 // If we are at the start of a group, we do not trust nScript,
1363 // we better get nScript from the breakiterator:
1364 if ( nChg
== nGrpStart
)
1365 nScript
= static_cast<sal_uInt8
>(g_pBreakIt
->GetBreakIter()->getScriptType(rText
, sal_Int32(nChg
)));
1367 // INVALID DATA FROM THE SCRIPT INFO ARRAYS HAS TO BE DELETED:
1369 // remove invalid entries from script information arrays
1370 m_ScriptChanges
.erase(m_ScriptChanges
.begin() + nCnt
, m_ScriptChanges
.end());
1372 // get the start of the last compression group
1373 TextFrameIndex nLastCompression
= nChg
;
1377 nLastCompression
= GetCompStart( nCntComp
);
1378 if( nChg
>= nLastCompression
+ GetCompLen( nCntComp
) )
1380 nLastCompression
= nChg
;
1385 // remove invalid entries from compression information arrays
1386 m_CompressionChanges
.erase(m_CompressionChanges
.begin() + nCntComp
,
1387 m_CompressionChanges
.end());
1389 // get the start of the last kashida group
1390 TextFrameIndex nLastKashida
= nChg
;
1391 if( nCntKash
&& i18n::ScriptType::COMPLEX
== nScript
)
1394 nLastKashida
= GetKashida( nCntKash
);
1397 // remove invalid entries from kashida array
1398 m_Kashida
.erase(m_Kashida
.begin() + nCntKash
, m_Kashida
.end());
1400 // TAKE CARE OF WEAK CHARACTERS: WE MUST FIND AN APPROPRIATE
1401 // SCRIPT FOR WEAK CHARACTERS AT THE BEGINNING OF A PARAGRAPH
1403 if (WEAK
== g_pBreakIt
->GetBreakIter()->getScriptType(rText
, sal_Int32(nChg
)))
1405 // If the beginning of the current group is weak, this means that
1406 // all of the characters in this group are weak. We have to assign
1407 // the scripts to these characters depending on the fonts which are
1408 // set for these characters to display them.
1409 TextFrameIndex
nEnd(
1410 g_pBreakIt
->GetBreakIter()->endOfScript(rText
, sal_Int32(nChg
), WEAK
));
1412 if (nEnd
> TextFrameIndex(rText
.getLength()) || nEnd
< TextFrameIndex(0))
1413 nEnd
= TextFrameIndex(rText
.getLength());
1415 nScript
= SvtLanguageOptions::GetI18NScriptTypeOfLanguage( GetAppLanguage() );
1417 SAL_WARN_IF( i18n::ScriptType::LATIN
!= nScript
&&
1418 i18n::ScriptType::ASIAN
!= nScript
&&
1419 i18n::ScriptType::COMPLEX
!= nScript
, "sw.core", "Wrong default language" );
1423 // Get next script type or set to weak in order to exit
1424 sal_uInt8 nNextScript
= (nEnd
< TextFrameIndex(rText
.getLength()))
1425 ? static_cast<sal_uInt8
>(g_pBreakIt
->GetBreakIter()->getScriptType(rText
, sal_Int32(nEnd
)))
1428 if ( nScript
!= nNextScript
)
1430 m_ScriptChanges
.emplace_back(nEnd
, nScript
);
1432 nScript
= nNextScript
;
1436 // UPDATE THE SCRIPT INFO ARRAYS:
1438 while (nChg
< TextFrameIndex(rText
.getLength())
1439 || (m_ScriptChanges
.empty() && rText
.isEmpty()))
1441 SAL_WARN_IF( i18n::ScriptType::WEAK
== nScript
,
1442 "sw.core", "Inserting WEAK into SwScriptInfo structure" );
1444 TextFrameIndex nSearchStt
= nChg
;
1445 nChg
= TextFrameIndex(g_pBreakIt
->GetBreakIter()->endOfScript(
1446 rText
, sal_Int32(nSearchStt
), nScript
));
1448 if (nChg
> TextFrameIndex(rText
.getLength()) || nChg
< TextFrameIndex(0))
1449 nChg
= TextFrameIndex(rText
.getLength());
1451 // special case for dotted circle since it can be used with complex
1452 // before a mark, so we want it associated with the mark's script
1453 // tdf#112594: another soecial case for NNBSP followed by a Mongolian
1454 // character, since NNBSP has special uses in Mongolian (tdf#112594)
1455 auto nPos
= sal_Int32(nChg
);
1456 auto nPrevPos
= nPos
;
1457 auto nPrevChar
= rText
.iterateCodePoints(&nPrevPos
, -1);
1458 if (nChg
< TextFrameIndex(rText
.getLength()) && nChg
> TextFrameIndex(0)
1459 && (i18n::ScriptType::WEAK
==
1460 g_pBreakIt
->GetBreakIter()->getScriptType(rText
, nPrevPos
)))
1462 auto nChar
= rText
.iterateCodePoints(&nPos
, 0);
1463 auto nType
= u_charType(nChar
);
1464 if (nType
== U_NON_SPACING_MARK
||
1465 nType
== U_ENCLOSING_MARK
||
1466 nType
== U_COMBINING_SPACING_MARK
||
1467 (nPrevChar
== CHAR_NNBSP
&&
1468 u_getIntPropertyValue(nChar
, UCHAR_SCRIPT
) == USCRIPT_MONGOLIAN
))
1473 m_ScriptChanges
.emplace_back(TextFrameIndex(nPos
), nScript
);
1476 // if current script is asian, we search for compressible characters
1478 if ( CharCompressType::NONE
!= aCompEnum
&&
1479 i18n::ScriptType::ASIAN
== nScript
)
1481 CompType ePrevState
= NONE
;
1482 CompType eState
= NONE
;
1483 TextFrameIndex nPrevChg
= nLastCompression
;
1485 while ( nLastCompression
< nChg
)
1487 sal_Unicode cChar
= rText
[ sal_Int32(nLastCompression
) ];
1489 // examine current character
1492 // Left punctuation found
1493 case 0x3008: case 0x300A: case 0x300C: case 0x300E:
1494 case 0x3010: case 0x3014: case 0x3016: case 0x3018:
1495 case 0x301A: case 0x301D:
1496 case 0xFF08: case 0xFF3B: case 0xFF5B:
1497 eState
= SPECIAL_LEFT
;
1499 // Right punctuation found
1500 case 0x3009: case 0x300B:
1501 case 0x300D: case 0x300F: case 0x3011: case 0x3015:
1502 case 0x3017: case 0x3019: case 0x301B: case 0x301E:
1504 case 0xFF09: case 0xFF3D: case 0xFF5D:
1505 eState
= SPECIAL_RIGHT
;
1507 case 0x3001: case 0x3002: // Fullstop or comma
1508 case 0xFF0C: case 0xFF0E: case 0xFF1A: case 0xFF1B:
1509 eState
= SPECIAL_MIDDLE
;
1512 eState
= ( 0x3040 <= cChar
&& 0x3100 > cChar
) ? KANA
: NONE
;
1515 // insert range of compressible characters
1516 if( ePrevState
!= eState
)
1518 if ( ePrevState
!= NONE
)
1520 // insert start and type
1521 if ( CharCompressType::PunctuationAndKana
== aCompEnum
||
1522 ePrevState
!= KANA
)
1524 m_CompressionChanges
.emplace_back(nPrevChg
,
1525 nLastCompression
- nPrevChg
, ePrevState
);
1529 ePrevState
= eState
;
1530 nPrevChg
= nLastCompression
;
1536 // we still have to examine last entry
1537 if ( ePrevState
!= NONE
)
1539 // insert start and type
1540 if ( CharCompressType::PunctuationAndKana
== aCompEnum
||
1541 ePrevState
!= KANA
)
1543 m_CompressionChanges
.emplace_back(nPrevChg
,
1544 nLastCompression
- nPrevChg
, ePrevState
);
1549 // we search for connecting opportunities (kashida)
1550 else if ( bAdjustBlock
&& i18n::ScriptType::COMPLEX
== nScript
)
1552 // sw_redlinehide: this is the only place that uses SwScanner with
1553 // frame text, so we convert to sal_Int32 here
1554 std::function
<LanguageType (sal_Int32
, sal_Int32
, bool)> const pGetLangOfCharM(
1555 [&pMerged
](sal_Int32
const nBegin
, sal_uInt16
const script
, bool const bNoChar
)
1557 std::pair
<SwTextNode
const*, sal_Int32
> const pos(
1558 sw::MapViewToModel(*pMerged
, TextFrameIndex(nBegin
)));
1559 return pos
.first
->GetLang(pos
.second
, bNoChar
? 0 : 1, script
);
1561 std::function
<LanguageType (sal_Int32
, sal_Int32
, bool)> const pGetLangOfChar1(
1562 [&rNode
](sal_Int32
const nBegin
, sal_uInt16
const script
, bool const bNoChar
)
1563 { return rNode
.GetLang(nBegin
, bNoChar
? 0 : 1, script
); });
1564 auto pGetLangOfChar(pMerged
? pGetLangOfCharM
: pGetLangOfChar1
);
1565 SwScanner
aScanner( std::move(pGetLangOfChar
), rText
, nullptr, ModelToViewHelper(),
1566 i18n::WordType::DICTIONARY_WORD
,
1567 sal_Int32(nLastKashida
), sal_Int32(nChg
));
1569 // the search has to be performed on a per word base
1570 while ( aScanner
.NextWord() )
1572 const OUString
& rWord
= aScanner
.GetWord();
1574 sal_Int32 nIdx
= 0, nPrevIdx
= 0;
1575 sal_Int32 nKashidaPos
= -1;
1576 sal_Unicode cCh
, cPrevCh
= 0;
1578 int nPriorityLevel
= 7; // 0..6 = level found
1581 sal_Int32 nWordLen
= rWord
.getLength();
1583 // ignore trailing vowel chars
1584 while( nWordLen
&& isTransparentChar( rWord
[ nWordLen
- 1 ] ))
1587 while (nIdx
< nWordLen
)
1589 cCh
= rWord
[ nIdx
];
1592 // after user inserted kashida
1595 nKashidaPos
= aScanner
.GetBegin() + nIdx
;
1600 // after a Seen or Sad
1601 if (nPriorityLevel
>= 1 && nIdx
< nWordLen
- 1)
1603 if( isSeenOrSadChar( cCh
)
1604 && (rWord
[ nIdx
+1 ] != 0x200C) ) // #i98410#: prevent ZWNJ expansion
1606 nKashidaPos
= aScanner
.GetBegin() + nIdx
;
1612 // before final form of Teh Marbuta, Heh, Dal
1613 if ( nPriorityLevel
>= 2 && nIdx
> 0 )
1615 if ( isTehMarbutaChar ( cCh
) || // Teh Marbuta (right joining)
1616 isDalChar ( cCh
) || // Dal (right joining) final form may appear in the middle of word
1617 ( isHehChar ( cCh
) && nIdx
== nWordLen
- 1)) // Heh (dual joining) only at end of word
1620 SAL_WARN_IF( 0 == cPrevCh
, "sw.core", "No previous character" );
1621 // check if character is connectable to previous character,
1622 if ( lcl_ConnectToPrev( cCh
, cPrevCh
) )
1624 nKashidaPos
= aScanner
.GetBegin() + nPrevIdx
;
1631 // before final form of Alef, Tah, Lam, Kaf or Gaf
1632 if ( nPriorityLevel
>= 3 && nIdx
> 0 )
1634 if ( isAlefChar ( cCh
) || // Alef (right joining) final form may appear in the middle of word
1635 (( isLamChar ( cCh
) || // Lam,
1636 isTahChar ( cCh
) || // Tah,
1637 isKafChar ( cCh
) || // Kaf (all dual joining)
1639 && nIdx
== nWordLen
- 1)) // only at end of word
1641 SAL_WARN_IF( 0 == cPrevCh
, "sw.core", "No previous character" );
1642 // check if character is connectable to previous character,
1643 if ( lcl_ConnectToPrev( cCh
, cPrevCh
) )
1645 nKashidaPos
= aScanner
.GetBegin() + nPrevIdx
;
1652 // before medial Beh-like
1653 if ( nPriorityLevel
>= 4 && nIdx
> 0 && nIdx
< nWordLen
- 1 )
1655 if ( isBehChar ( cCh
) )
1657 // check if next character is Reh or Yeh-like
1658 sal_Unicode cNextCh
= rWord
[ nIdx
+ 1 ];
1659 if ( isRehChar ( cNextCh
) || isYehChar ( cNextCh
))
1661 SAL_WARN_IF( 0 == cPrevCh
, "sw.core", "No previous character" );
1662 // check if character is connectable to previous character,
1663 if ( lcl_ConnectToPrev( cCh
, cPrevCh
) )
1665 nKashidaPos
= aScanner
.GetBegin() + nPrevIdx
;
1673 // before the final form of Waw, Ain, Qaf and Feh
1674 if ( nPriorityLevel
>= 5 && nIdx
> 0 )
1676 if ( isWawChar ( cCh
) || // Wav (right joining)
1677 // final form may appear in the middle of word
1678 (( isAinChar ( cCh
) || // Ain (dual joining)
1679 isQafChar ( cCh
) || // Qaf (dual joining)
1680 isFehChar ( cCh
) ) // Feh (dual joining)
1681 && nIdx
== nWordLen
- 1)) // only at end of word
1683 SAL_WARN_IF( 0 == cPrevCh
, "sw.core", "No previous character" );
1684 // check if character is connectable to previous character,
1685 if ( lcl_ConnectToPrev( cCh
, cPrevCh
) )
1687 nKashidaPos
= aScanner
.GetBegin() + nPrevIdx
;
1693 // other connecting possibilities
1694 if ( nPriorityLevel
>= 6 && nIdx
> 0 )
1697 if ( isRehChar ( cCh
) )
1699 SAL_WARN_IF( 0 == cPrevCh
, "sw.core", "No previous character" );
1700 // check if character is connectable to previous character,
1701 if ( lcl_ConnectToPrev( cCh
, cPrevCh
) )
1703 nKashidaPos
= aScanner
.GetBegin() + nPrevIdx
;
1709 // Do not consider vowel marks when checking if a character
1710 // can be connected to previous character.
1711 if ( !isTransparentChar ( cCh
) )
1718 } // end of current word
1720 if ( -1 != nKashidaPos
)
1722 m_Kashida
.insert(m_Kashida
.begin() + nCntKash
, TextFrameIndex(nKashidaPos
));
1725 } // end of kashida search
1728 if (nChg
< TextFrameIndex(rText
.getLength()))
1729 nScript
= static_cast<sal_uInt8
>(g_pBreakIt
->GetBreakIter()->getScriptType(rText
, sal_Int32(nChg
)));
1731 nLastCompression
= nChg
;
1732 nLastKashida
= nChg
;
1735 #if OSL_DEBUG_LEVEL > 0
1736 // check kashida data
1737 TextFrameIndex
nTmpKashidaPos(-1);
1738 bool bWrongKash
= false;
1739 for (size_t i
= 0; i
< m_Kashida
.size(); ++i
)
1741 TextFrameIndex nCurrKashidaPos
= GetKashida( i
);
1742 if ( nCurrKashidaPos
<= nTmpKashidaPos
)
1747 nTmpKashidaPos
= nCurrKashidaPos
;
1749 SAL_WARN_IF( bWrongKash
, "sw.core", "Kashida array contains wrong data" );
1752 // remove invalid entries from direction information arrays
1753 m_DirectionChanges
.clear();
1755 // Perform Unicode Bidi Algorithm for text direction information
1757 UpdateBidiInfo( rText
);
1759 // #i16354# Change script type for RTL text to CTL:
1760 // 1. All text in RTL runs will use the CTL font
1761 // #i89825# change the script type also to CTL (hennerdrewes)
1762 // 2. Text in embedded LTR runs that does not have any strong LTR characters (numbers!)
1763 for (size_t nDirIdx
= 0; nDirIdx
< m_DirectionChanges
.size(); ++nDirIdx
)
1765 const sal_uInt8 nCurrDirType
= GetDirType( nDirIdx
);
1766 // nStart is start of RTL run:
1767 const TextFrameIndex nStart
= nDirIdx
> 0 ? GetDirChg(nDirIdx
- 1) : TextFrameIndex(0);
1768 // nEnd is end of RTL run:
1769 const TextFrameIndex nEnd
= GetDirChg( nDirIdx
);
1771 if ( nCurrDirType
% 2 == UBIDI_RTL
|| // text in RTL run
1772 (nCurrDirType
> UBIDI_LTR
&& // non-strong text in embedded LTR run
1773 !lcl_HasStrongLTR(rText
, sal_Int32(nStart
), sal_Int32(nEnd
))))
1775 // nScriptIdx points into the ScriptArrays:
1776 size_t nScriptIdx
= 0;
1778 // Skip entries in ScriptArray which are not inside the RTL run:
1779 // Make nScriptIdx become the index of the script group with
1780 // 1. nStartPosOfGroup <= nStart and
1781 // 2. nEndPosOfGroup > nStart
1782 while ( GetScriptChg( nScriptIdx
) <= nStart
)
1785 const TextFrameIndex nStartPosOfGroup
= nScriptIdx
1786 ? GetScriptChg(nScriptIdx
- 1)
1787 : TextFrameIndex(0);
1788 const sal_uInt8 nScriptTypeOfGroup
= GetScriptType( nScriptIdx
);
1790 SAL_WARN_IF( nStartPosOfGroup
> nStart
|| GetScriptChg( nScriptIdx
) <= nStart
,
1791 "sw.core", "Script override with CTL font trouble" );
1793 // Check if we have to insert a new script change at
1794 // position nStart. If nStartPosOfGroup < nStart,
1795 // we have to insert a new script change:
1796 if (nStart
> TextFrameIndex(0) && nStartPosOfGroup
< nStart
)
1798 m_ScriptChanges
.insert(m_ScriptChanges
.begin() + nScriptIdx
,
1799 ScriptChangeInfo(nStart
, nScriptTypeOfGroup
) );
1803 // Remove entries in ScriptArray which end inside the RTL run:
1804 while (nScriptIdx
< m_ScriptChanges
.size()
1805 && GetScriptChg(nScriptIdx
) <= nEnd
)
1807 m_ScriptChanges
.erase(m_ScriptChanges
.begin() + nScriptIdx
);
1810 // Insert a new entry in ScriptArray for the end of the RTL run:
1811 m_ScriptChanges
.insert(m_ScriptChanges
.begin() + nScriptIdx
,
1812 ScriptChangeInfo(nEnd
, i18n::ScriptType::COMPLEX
) );
1814 #if OSL_DEBUG_LEVEL > 1
1815 // Check that ScriptChangeInfos are in increasing order of
1816 // position and that we don't have "empty" changes.
1817 sal_uInt8 nLastTyp
= i18n::ScriptType::WEAK
;
1818 TextFrameIndex nLastPos
= TextFrameIndex(0);
1819 for (const auto& rScriptChange
: m_ScriptChanges
)
1821 SAL_WARN_IF( nLastTyp
== rScriptChange
.type
||
1822 nLastPos
>= rScriptChange
.position
,
1823 "sw.core", "Heavy InitScriptType() confusion" );
1824 nLastPos
= rScriptChange
.position
;
1825 nLastTyp
= rScriptChange
.type
;
1833 void SwScriptInfo::UpdateBidiInfo( const OUString
& rText
)
1835 // remove invalid entries from direction information arrays
1836 m_DirectionChanges
.clear();
1838 // Bidi functions from icu 2.0
1840 UErrorCode nError
= U_ZERO_ERROR
;
1841 UBiDi
* pBidi
= ubidi_openSized( rText
.getLength(), 0, &nError
);
1842 nError
= U_ZERO_ERROR
;
1844 ubidi_setPara( pBidi
, reinterpret_cast<const UChar
*>(rText
.getStr()), rText
.getLength(),
1845 m_nDefaultDir
, nullptr, &nError
);
1846 nError
= U_ZERO_ERROR
;
1847 int nCount
= ubidi_countRuns( pBidi
, &nError
);
1850 UBiDiLevel nCurrDir
;
1851 for ( int nIdx
= 0; nIdx
< nCount
; ++nIdx
)
1853 ubidi_getLogicalRun( pBidi
, nStart
, &nEnd
, &nCurrDir
);
1854 m_DirectionChanges
.emplace_back(TextFrameIndex(nEnd
), nCurrDir
);
1858 ubidi_close( pBidi
);
1861 // returns the position of the next character which belongs to another script
1862 // than the character of the actual (input) position.
1863 // If there's no script change until the end of the paragraph, it will return
1865 // Scripts are Asian (Chinese, Japanese, Korean),
1866 // Latin ( English etc.)
1867 // and Complex ( Hebrew, Arabian )
1868 TextFrameIndex
SwScriptInfo::NextScriptChg(const TextFrameIndex nPos
) const
1870 const size_t nEnd
= CountScriptChg();
1871 for( size_t nX
= 0; nX
< nEnd
; ++nX
)
1873 if( nPos
< GetScriptChg( nX
) )
1874 return GetScriptChg( nX
);
1877 return TextFrameIndex(COMPLETE_STRING
);
1880 // returns the script of the character at the input position
1881 sal_Int16
SwScriptInfo::ScriptType(const TextFrameIndex nPos
) const
1883 const size_t nEnd
= CountScriptChg();
1884 for( size_t nX
= 0; nX
< nEnd
; ++nX
)
1886 if( nPos
< GetScriptChg( nX
) )
1887 return GetScriptType( nX
);
1890 // the default is the application language script
1891 return SvtLanguageOptions::GetI18NScriptTypeOfLanguage( GetAppLanguage() );
1894 TextFrameIndex
SwScriptInfo::NextDirChg(const TextFrameIndex nPos
,
1895 const sal_uInt8
* pLevel
) const
1897 const sal_uInt8 nCurrDir
= pLevel
? *pLevel
: 62;
1898 const size_t nEnd
= CountDirChg();
1899 for( size_t nX
= 0; nX
< nEnd
; ++nX
)
1901 if( nPos
< GetDirChg( nX
) &&
1902 ( nX
+ 1 == nEnd
|| GetDirType( nX
+ 1 ) <= nCurrDir
) )
1903 return GetDirChg( nX
);
1906 return TextFrameIndex(COMPLETE_STRING
);
1909 sal_uInt8
SwScriptInfo::DirType(const TextFrameIndex nPos
) const
1911 const size_t nEnd
= CountDirChg();
1912 for( size_t nX
= 0; nX
< nEnd
; ++nX
)
1914 if( nPos
< GetDirChg( nX
) )
1915 return GetDirType( nX
);
1921 TextFrameIndex
SwScriptInfo::NextHiddenChg(TextFrameIndex
const nPos
) const
1923 for (auto const& it
: m_HiddenChg
)
1930 return TextFrameIndex(COMPLETE_STRING
);
1933 TextFrameIndex
SwScriptInfo::NextBookmark(TextFrameIndex
const nPos
) const
1935 for (auto const& it
: m_Bookmarks
)
1937 if (nPos
< std::get
<0>(it
))
1939 return std::get
<0>(it
);
1942 return TextFrameIndex(COMPLETE_STRING
);
1945 std::vector
<std::tuple
<SwScriptInfo::MarkKind
, Color
, OUString
>>
1946 SwScriptInfo::GetBookmarks(TextFrameIndex
const nPos
)
1948 std::vector
<std::tuple
<SwScriptInfo::MarkKind
, Color
, OUString
>> aColors
;
1949 for (auto const& it
: m_Bookmarks
)
1951 if (nPos
== std::get
<0>(it
))
1953 const OUString
& sName
= std::get
<3>(it
);
1954 // filter hidden bookmarks imported from OOXML
1955 // TODO import them as hidden bookmarks
1956 if ( !( sName
.startsWith("_Toc") || sName
.startsWith("_Ref") ) )
1957 aColors
.push_back(std::tuple
<MarkKind
, Color
,
1958 OUString
>(std::get
<1>(it
), std::get
<2>(it
), std::get
<3>(it
)));
1960 else if (nPos
< std::get
<0>(it
))
1966 // sort bookmark boundary marks at the same position
1967 // mark order: ] | [
1968 // color order: [c1 [c2 [c3 ... c3] c2] c1]
1969 sort(aColors
.begin(), aColors
.end(),
1970 [](std::tuple
<MarkKind
, Color
, OUString
> const a
, std::tuple
<MarkKind
, Color
, OUString
> const b
) {
1971 return (MarkKind::End
== std::get
<0>(a
) && MarkKind::End
!= std::get
<0>(b
)) ||
1972 (MarkKind::Point
== std::get
<0>(a
) && MarkKind::Start
== std::get
<0>(b
)) ||
1973 // if both are end or start, order by color
1974 (MarkKind::End
== std::get
<0>(a
) && MarkKind::End
== std::get
<0>(b
) && std::get
<1>(a
) < std::get
<1>(b
)) ||
1975 (MarkKind::Start
== std::get
<0>(a
) && MarkKind::Start
== std::get
<0>(b
) && std::get
<1>(b
) < std::get
<1>(a
));});
1980 // Takes a string and replaced the hidden ranges with cChar.
1981 sal_Int32
SwScriptInfo::MaskHiddenRanges( const SwTextNode
& rNode
, OUStringBuffer
& rText
,
1982 const sal_Int32 nStt
, const sal_Int32 nEnd
,
1983 const sal_Unicode cChar
)
1985 assert(rNode
.GetText().getLength() == rText
.getLength());
1987 std::vector
<sal_Int32
> aList
;
1988 sal_Int32 nHiddenStart
;
1989 sal_Int32 nHiddenEnd
;
1990 sal_Int32 nNumOfHiddenChars
= 0;
1991 GetBoundsOfHiddenRange( rNode
, 0, nHiddenStart
, nHiddenEnd
, &aList
);
1992 auto rFirst( aList
.crbegin() );
1993 auto rLast( aList
.crend() );
1994 while ( rFirst
!= rLast
)
1996 nHiddenEnd
= *(rFirst
++);
1997 nHiddenStart
= *(rFirst
++);
1999 if ( nHiddenEnd
< nStt
|| nHiddenStart
> nEnd
)
2002 while ( nHiddenStart
< nHiddenEnd
&& nHiddenStart
< nEnd
)
2004 if (nHiddenStart
>= nStt
)
2006 rText
[nHiddenStart
] = cChar
;
2007 ++nNumOfHiddenChars
;
2013 return nNumOfHiddenChars
;
2016 // Takes a SwTextNode and deletes the hidden ranges from the node.
2017 void SwScriptInfo::DeleteHiddenRanges( SwTextNode
& rNode
)
2019 std::vector
<sal_Int32
> aList
;
2020 sal_Int32 nHiddenStart
;
2021 sal_Int32 nHiddenEnd
;
2022 GetBoundsOfHiddenRange( rNode
, 0, nHiddenStart
, nHiddenEnd
, &aList
);
2023 auto rFirst( aList
.crbegin() );
2024 auto rLast( aList
.crend() );
2025 while ( rFirst
!= rLast
)
2027 nHiddenEnd
= *(rFirst
++);
2028 nHiddenStart
= *(rFirst
++);
2030 SwPaM
aPam( rNode
, nHiddenStart
, rNode
, nHiddenEnd
);
2031 rNode
.getIDocumentContentOperations().DeleteRange( aPam
);
2035 bool SwScriptInfo::GetBoundsOfHiddenRange( const SwTextNode
& rNode
, sal_Int32 nPos
,
2036 sal_Int32
& rnStartPos
, sal_Int32
& rnEndPos
,
2037 std::vector
<sal_Int32
>* pList
)
2039 rnStartPos
= COMPLETE_STRING
;
2042 bool bNewContainsHiddenChars
= false;
2044 // Optimization: First examine the flags at the text node:
2046 if ( !rNode
.IsCalcHiddenCharFlags() )
2048 bool bWholePara
= rNode
.HasHiddenCharAttribute( true );
2049 bool bContainsHiddenChars
= rNode
.HasHiddenCharAttribute( false );
2050 if ( !bContainsHiddenChars
)
2057 pList
->push_back( 0 );
2058 pList
->push_back(rNode
.GetText().getLength());
2062 rnEndPos
= rNode
.GetText().getLength();
2067 // sw_redlinehide: this won't work if it's merged
2069 const SwScriptInfo
* pSI
= SwScriptInfo::GetScriptInfo( rNode
);
2073 // Check first, if we have a valid SwScriptInfo object for this text node:
2075 bNewContainsHiddenChars
= pSI
->GetBoundsOfHiddenRange( nPos
, rnStartPos
, rnEndPos
, pList
);
2076 const bool bNewHiddenCharsHidePara
=
2077 rnStartPos
== 0 && rnEndPos
>= rNode
.GetText().getLength();
2078 rNode
.SetHiddenCharAttribute( bNewHiddenCharsHidePara
, bNewContainsHiddenChars
);
2084 // No valid SwScriptInfo Object, we have to do it the hard way:
2086 Range
aRange(0, (!rNode
.GetText().isEmpty())
2087 ? rNode
.GetText().getLength() - 1
2089 MultiSelection
aHiddenMulti( aRange
);
2090 SwScriptInfo::CalcHiddenRanges(rNode
, aHiddenMulti
, nullptr);
2091 for( sal_Int32 i
= 0; i
< aHiddenMulti
.GetRangeCount(); ++i
)
2093 const Range
& rRange
= aHiddenMulti
.GetRange( i
);
2094 const sal_Int32 nHiddenStart
= rRange
.Min();
2095 const sal_Int32 nHiddenEnd
= rRange
.Max() + 1;
2097 if ( nHiddenStart
> nPos
)
2099 if (nPos
< nHiddenEnd
)
2101 rnStartPos
= nHiddenStart
;
2102 rnEndPos
= std::min
<sal_Int32
>(nHiddenEnd
,
2103 rNode
.GetText().getLength());
2110 for( sal_Int32 i
= 0; i
< aHiddenMulti
.GetRangeCount(); ++i
)
2112 const Range
& rRange
= aHiddenMulti
.GetRange( i
);
2113 pList
->push_back( rRange
.Min() );
2114 pList
->push_back( rRange
.Max() + 1 );
2118 bNewContainsHiddenChars
= aHiddenMulti
.GetRangeCount() > 0;
2121 return bNewContainsHiddenChars
;
2124 bool SwScriptInfo::GetBoundsOfHiddenRange(TextFrameIndex nPos
,
2125 TextFrameIndex
& rnStartPos
, TextFrameIndex
& rnEndPos
) const
2127 rnStartPos
= TextFrameIndex(COMPLETE_STRING
);
2128 rnEndPos
= TextFrameIndex(0);
2130 const size_t nEnd
= CountHiddenChg();
2131 for( size_t nX
= 0; nX
< nEnd
; ++nX
)
2133 const TextFrameIndex nHiddenStart
= GetHiddenChg( nX
++ );
2134 const TextFrameIndex nHiddenEnd
= GetHiddenChg( nX
);
2136 if ( nHiddenStart
> nPos
)
2138 if (nPos
< nHiddenEnd
)
2140 rnStartPos
= nHiddenStart
;
2141 rnEndPos
= nHiddenEnd
;
2146 return CountHiddenChg() > 0;
2149 bool SwScriptInfo::IsInHiddenRange( const SwTextNode
& rNode
, sal_Int32 nPos
)
2151 sal_Int32 nStartPos
;
2153 SwScriptInfo::GetBoundsOfHiddenRange( rNode
, nPos
, nStartPos
, nEndPos
);
2154 return nStartPos
!= COMPLETE_STRING
;
2158 // returns the type of the compressed character
2159 SwScriptInfo::CompType
SwScriptInfo::DbgCompType(const TextFrameIndex nPos
) const
2161 const size_t nEnd
= CountCompChg();
2162 for( size_t nX
= 0; nX
< nEnd
; ++nX
)
2164 const TextFrameIndex nChg
= GetCompStart(nX
);
2169 if( nPos
< nChg
+ GetCompLen( nX
) )
2170 return GetCompType( nX
);
2176 // returns, if there are compressible kanas or specials
2177 // between nStart and nEnd
2178 size_t SwScriptInfo::HasKana(TextFrameIndex
const nStart
, TextFrameIndex
const nLen
) const
2180 const size_t nCnt
= CountCompChg();
2181 TextFrameIndex nEnd
= nStart
+ nLen
;
2183 for( size_t nX
= 0; nX
< nCnt
; ++nX
)
2185 TextFrameIndex nKanaStart
= GetCompStart(nX
);
2186 TextFrameIndex nKanaEnd
= nKanaStart
+ GetCompLen(nX
);
2188 if ( nKanaStart
>= nEnd
)
2189 return SAL_MAX_SIZE
;
2191 if ( nStart
< nKanaEnd
)
2195 return SAL_MAX_SIZE
;
2198 tools::Long
SwScriptInfo::Compress(KernArray
& rKernArray
, TextFrameIndex nIdx
, TextFrameIndex nLen
,
2199 const sal_uInt16 nCompress
, const sal_uInt16 nFontHeight
,
2201 Point
* pPoint
) const
2203 SAL_WARN_IF( !nCompress
, "sw.core", "Compression without compression?!" );
2204 SAL_WARN_IF( !nLen
, "sw.core", "Compression without text?!" );
2205 const size_t nCompCount
= CountCompChg();
2207 // In asian typography, there are full width and half width characters.
2208 // Full width punctuation characters can be compressed by 50%
2209 // to determine this, we compare the font width with 75% of its height
2210 const tools::Long nMinWidth
= ( 3 * nFontHeight
) / 4;
2212 size_t nCompIdx
= HasKana( nIdx
, nLen
);
2214 if ( SAL_MAX_SIZE
== nCompIdx
)
2217 TextFrameIndex nChg
= GetCompStart( nCompIdx
);
2218 TextFrameIndex nCompLen
= GetCompLen( nCompIdx
);
2224 nI
= sal_Int32(nChg
- nIdx
);
2227 else if( nIdx
< nChg
+ nCompLen
)
2228 nCompLen
-= nIdx
- nChg
;
2230 if( nIdx
> nLen
|| nCompIdx
>= nCompCount
)
2233 tools::Long nSub
= 0;
2234 tools::Long nLast
= nI
? rKernArray
[ nI
- 1 ] : 0;
2237 const CompType nType
= GetCompType( nCompIdx
);
2239 SAL_WARN_IF( nType
!= DbgCompType( nIdx
), "sw.core", "Gimme the right type!" );
2242 if( nCompLen
> nLen
)
2245 // are we allowed to compress the character?
2246 if ( rKernArray
[ nI
] - nLast
< nMinWidth
)
2252 while( nIdx
< nCompLen
)
2254 SAL_WARN_IF( SwScriptInfo::NONE
== nType
, "sw.core", "None compression?!" );
2256 // nLast is width of current character
2257 nLast
-= rKernArray
[ nI
];
2260 tools::Long nMove
= 0;
2261 if( SwScriptInfo::KANA
!= nType
)
2264 if( pPoint
&& SwScriptInfo::SPECIAL_LEFT
== nType
)
2270 pPoint
->AdjustX(nLast
);
2274 else if( bCenter
&& SwScriptInfo::SPECIAL_MIDDLE
== nType
)
2280 nLast
= rKernArray
[ nI
];
2282 rKernArray
.adjust(nI
- 1, nMove
);
2283 rKernArray
.adjust(nI
, -nSub
);
2292 TextFrameIndex nTmpChg
= nLen
;
2293 if( ++nCompIdx
< nCompCount
)
2295 nTmpChg
= GetCompStart( nCompIdx
);
2296 if( nTmpChg
> nLen
)
2298 nCompLen
= GetCompLen( nCompIdx
);
2301 while( nIdx
< nTmpChg
)
2303 nLast
= rKernArray
[ nI
];
2304 rKernArray
.adjust(nI
, -nSub
);
2308 } while( nIdx
< nLen
);
2312 // Note on calling KashidaJustify():
2313 // Kashida positions may be marked as invalid. Therefore KashidaJustify may return the clean
2314 // total number of kashida positions, or the number of kashida positions after some positions
2315 // have been dropped, depending on the state of the m_KashidaInvalid set.
2317 sal_Int32
SwScriptInfo::KashidaJustify( KernArray
* pKernArray
,
2318 sal_Bool
* pKashidaArray
,
2319 TextFrameIndex
const nStt
,
2320 TextFrameIndex
const nLen
,
2321 tools::Long nSpaceAdd
) const
2323 SAL_WARN_IF( !nLen
, "sw.core", "Kashida justification without text?!" );
2325 if( !IsKashidaLine(nStt
))
2328 // evaluate kashida information in collected in SwScriptInfo
2330 size_t nCntKash
= 0;
2331 while( nCntKash
< CountKashida() )
2333 if ( nStt
<= GetKashida( nCntKash
) )
2338 const TextFrameIndex nEnd
= nStt
+ nLen
;
2340 size_t nCntKashEnd
= nCntKash
;
2341 while ( nCntKashEnd
< CountKashida() )
2343 if ( nEnd
<= GetKashida( nCntKashEnd
) )
2348 size_t nActualKashCount
= nCntKashEnd
- nCntKash
;
2349 for (size_t i
= nCntKash
; i
< nCntKashEnd
; ++i
)
2351 if ( nActualKashCount
&& !IsKashidaValid ( i
) )
2356 return nActualKashCount
;
2358 // do nothing if there is no more kashida
2359 if ( nCntKash
< CountKashida() )
2361 // skip any invalid kashidas
2362 while (nCntKash
< nCntKashEnd
&& !IsKashidaValid(nCntKash
))
2365 TextFrameIndex nIdx
= nCntKash
< nCntKashEnd
&& IsKashidaValid(nCntKash
)
2366 ? GetKashida(nCntKash
)
2368 tools::Long nKashAdd
= nSpaceAdd
;
2370 while ( nIdx
< nEnd
)
2372 TextFrameIndex nArrayPos
= nIdx
- nStt
;
2374 // mark Kashida insertion positions, code in VCL will use this
2375 // array to know where to insert Kashida.
2377 pKashidaArray
[sal_Int32(nArrayPos
)] = true;
2379 // next kashida position
2381 while (nCntKash
< nCntKashEnd
&& !IsKashidaValid(nCntKash
))
2384 nIdx
= nCntKash
< nCntKashEnd
&& IsKashidaValid(nCntKash
) ? GetKashida(nCntKash
) : nEnd
;
2388 const TextFrameIndex nArrayEnd
= nIdx
- nStt
;
2390 while ( nArrayPos
< nArrayEnd
)
2392 pKernArray
->adjust(sal_Int32(nArrayPos
), nKashAdd
);
2395 nKashAdd
+= nSpaceAdd
;
2402 // Checks if the current text is 'Arabic' text. Note that only the first
2403 // character has to be checked because a ctl portion only contains one
2404 // script, see NewTextPortion
2405 bool SwScriptInfo::IsArabicText(const OUString
& rText
,
2406 TextFrameIndex
const nStt
, TextFrameIndex
const nLen
)
2408 using namespace ::com::sun::star::i18n
;
2409 static const ScriptTypeList typeList
[] = {
2410 { UnicodeScript_kArabic
, UnicodeScript_kArabic
, sal_Int16(UnicodeScript_kArabic
) }, // 11,
2411 { UnicodeScript_kScriptCount
, UnicodeScript_kScriptCount
, sal_Int16(UnicodeScript_kScriptCount
) } // 88
2414 // go forward if current position does not hold a regular character:
2415 const CharClass
& rCC
= GetAppCharClass();
2416 sal_Int32 nIdx
= sal_Int32(nStt
);
2417 const sal_Int32 nEnd
= sal_Int32(nStt
+ nLen
);
2418 while ( nIdx
< nEnd
&& !rCC
.isLetterNumeric( rText
, nIdx
) )
2425 // no regular character found in this portion. Go backward:
2427 while ( nIdx
>= 0 && !rCC
.isLetterNumeric( rText
, nIdx
) )
2435 const sal_Unicode cCh
= rText
[nIdx
];
2436 const sal_Int16 type
= unicode::getUnicodeScriptType( cCh
, typeList
, sal_Int16(UnicodeScript_kScriptCount
) );
2437 return type
== sal_Int16(UnicodeScript_kArabic
);
2442 bool SwScriptInfo::IsKashidaValid(size_t const nKashPos
) const
2444 return m_KashidaInvalid
.find(nKashPos
) == m_KashidaInvalid
.end();
2447 void SwScriptInfo::ClearKashidaInvalid(size_t const nKashPos
)
2449 m_KashidaInvalid
.erase(nKashPos
);
2453 // marks the first valid kashida in the given text range as invalid
2455 // clears all kashida invalid flags in the given text range
2456 bool SwScriptInfo::MarkOrClearKashidaInvalid(
2457 TextFrameIndex
const nStt
, TextFrameIndex
const nLen
,
2458 bool bMark
, sal_Int32 nMarkCount
)
2460 size_t nCntKash
= 0;
2461 while( nCntKash
< CountKashida() )
2463 if ( nStt
<= GetKashida( nCntKash
) )
2468 const TextFrameIndex nEnd
= nStt
+ nLen
;
2470 while ( nCntKash
< CountKashida() )
2472 if ( nEnd
<= GetKashida( nCntKash
) )
2476 if ( MarkKashidaInvalid ( nCntKash
) )
2485 ClearKashidaInvalid ( nCntKash
);
2492 bool SwScriptInfo::MarkKashidaInvalid(size_t const nKashPos
)
2494 return m_KashidaInvalid
.insert(nKashPos
).second
;
2497 // retrieve the kashida positions in the given text range
2498 void SwScriptInfo::GetKashidaPositions(
2499 TextFrameIndex
const nStt
, TextFrameIndex
const nLen
,
2500 std::vector
<TextFrameIndex
>& rKashidaPosition
)
2502 size_t nCntKash
= 0;
2503 while( nCntKash
< CountKashida() )
2505 if ( nStt
<= GetKashida( nCntKash
) )
2510 const TextFrameIndex nEnd
= nStt
+ nLen
;
2512 size_t nCntKashEnd
= nCntKash
;
2513 while ( nCntKashEnd
< CountKashida() )
2515 if ( nEnd
<= GetKashida( nCntKashEnd
) )
2517 rKashidaPosition
.push_back(GetKashida(nCntKashEnd
));
2522 void SwScriptInfo::SetNoKashidaLine(TextFrameIndex
const nStt
, TextFrameIndex
const nLen
)
2524 m_NoKashidaLine
.push_back( nStt
);
2525 m_NoKashidaLineEnd
.push_back( nStt
+ nLen
);
2528 // determines if the line uses kashida justification
2529 bool SwScriptInfo::IsKashidaLine(TextFrameIndex
const nCharIdx
) const
2531 for (size_t i
= 0; i
< m_NoKashidaLine
.size(); ++i
)
2533 if (nCharIdx
>= m_NoKashidaLine
[i
] && nCharIdx
< m_NoKashidaLineEnd
[i
])
2539 void SwScriptInfo::ClearNoKashidaLine(TextFrameIndex
const nStt
, TextFrameIndex
const nLen
)
2542 while (i
< m_NoKashidaLine
.size())
2544 if (nStt
+ nLen
>= m_NoKashidaLine
[i
] && nStt
< m_NoKashidaLineEnd
[i
])
2546 m_NoKashidaLine
.erase(m_NoKashidaLine
.begin() + i
);
2547 m_NoKashidaLineEnd
.erase(m_NoKashidaLineEnd
.begin() + i
);
2554 // mark the given character indices as invalid kashida positions
2555 void SwScriptInfo::MarkKashidasInvalid(sal_Int32
const nCnt
,
2556 const TextFrameIndex
* pKashidaPositions
)
2558 SAL_WARN_IF( !pKashidaPositions
|| nCnt
== 0, "sw.core", "Where are kashidas?" );
2560 size_t nCntKash
= 0;
2561 sal_Int32 nKashidaPosIdx
= 0;
2563 while (nCntKash
< CountKashida() && nKashidaPosIdx
< nCnt
)
2565 if ( pKashidaPositions
[nKashidaPosIdx
] > GetKashida( nCntKash
) )
2571 if ( pKashidaPositions
[nKashidaPosIdx
] != GetKashida( nCntKash
) || !IsKashidaValid ( nCntKash
) )
2572 return; // something is wrong
2574 MarkKashidaInvalid ( nCntKash
);
2579 TextFrameIndex
SwScriptInfo::ThaiJustify( std::u16string_view aText
, KernArray
* pKernArray
,
2580 TextFrameIndex
const nStt
,
2581 TextFrameIndex
const nLen
,
2582 TextFrameIndex nNumberOfBlanks
,
2583 tools::Long nSpaceAdd
)
2585 SAL_WARN_IF( nStt
+ nLen
> TextFrameIndex(aText
.size()), "sw.core", "String in ThaiJustify too small" );
2587 SwTwips nNumOfTwipsToDistribute
= nSpaceAdd
* sal_Int32(nNumberOfBlanks
) /
2588 SPACING_PRECISION_FACTOR
;
2590 tools::Long nSpaceSum
= 0;
2591 TextFrameIndex
nCnt(0);
2593 for (sal_Int32 nI
= 0; nI
< sal_Int32(nLen
); ++nI
)
2595 const sal_Unicode cCh
= aText
[sal_Int32(nStt
) + nI
];
2597 // check if character is not above or below base
2598 if ( ( 0xE34 > cCh
|| cCh
> 0xE3A ) &&
2599 ( 0xE47 > cCh
|| cCh
> 0xE4E ) && cCh
!= 0xE31 )
2601 if (nNumberOfBlanks
> TextFrameIndex(0))
2603 nSpaceAdd
= nNumOfTwipsToDistribute
/ sal_Int32(nNumberOfBlanks
);
2605 nNumOfTwipsToDistribute
-= nSpaceAdd
;
2607 nSpaceSum
+= nSpaceAdd
;
2612 pKernArray
->adjust(nI
, nSpaceSum
);
2618 SwScriptInfo
* SwScriptInfo::GetScriptInfo( const SwTextNode
& rTNd
,
2619 SwTextFrame
const**const o_ppFrame
,
2620 bool const bAllowInvalid
)
2622 SwIterator
<SwTextFrame
, SwTextNode
, sw::IteratorMode::UnwrapMulti
> aIter(rTNd
);
2623 SwScriptInfo
* pScriptInfo
= nullptr;
2625 for( SwTextFrame
* pLast
= aIter
.First(); pLast
; pLast
= aIter
.Next() )
2627 pScriptInfo
= const_cast<SwScriptInfo
*>(pLast
->GetScriptInfo());
2630 if (bAllowInvalid
||
2631 TextFrameIndex(COMPLETE_STRING
) == pScriptInfo
->GetInvalidityA())
2639 pScriptInfo
= nullptr;
2646 SwParaPortion::SwParaPortion()
2649 m_bFlys
= m_bFootnoteNum
= m_bMargin
= false;
2650 SetWhichPor( PortionType::Para
);
2653 SwParaPortion::~SwParaPortion()
2657 TextFrameIndex
SwParaPortion::GetParLen() const
2659 TextFrameIndex
nLen(0);
2660 const SwLineLayout
*pLay
= this;
2663 nLen
+= pLay
->GetLen();
2664 pLay
= pLay
->GetNext();
2669 bool SwParaPortion::HasNumberingPortion(FootnoteOrNot
const eFootnote
) const
2671 SwLinePortion
const* pPortion(nullptr);
2672 // the first line may contain only fly portion...
2673 for (SwLineLayout
const* pLine
= this; pLine
&& !pPortion
; pLine
= pLine
->GetNext())
2675 pPortion
= pLine
->GetFirstPortion();
2676 while (pPortion
&& (pPortion
->InGlueGrp() || pPortion
->IsKernPortion() || pPortion
->IsFlyPortion()))
2677 { // skip margins and fly spacers - numbering should be first then
2678 pPortion
= pPortion
->GetNextPortion();
2681 if (pPortion
&& pPortion
->InHyphGrp())
2682 { // weird special case, bullet with soft hyphen
2683 pPortion
= pPortion
->GetNextPortion();
2685 return pPortion
&& pPortion
->InNumberGrp()
2686 && (eFootnote
== SwParaPortion::FootnoteToo
|| !pPortion
->IsFootnoteNumPortion());
2689 bool SwParaPortion::HasContentPortions() const
2691 SwLinePortion
const* pPortion(nullptr);
2692 for (SwLineLayout
const* pLine
= this; pLine
&& !pPortion
; pLine
= pLine
->GetNext())
2694 pPortion
= pLine
->GetFirstPortion();
2695 while (pPortion
&& (pPortion
->InGlueGrp() || pPortion
->IsKernPortion() || pPortion
->IsFlyPortion()))
2696 { // skip margins and fly spacers
2697 pPortion
= pPortion
->GetNextPortion();
2700 return pPortion
!= nullptr;
2703 const SwDropPortion
*SwParaPortion::FindDropPortion() const
2705 const SwLineLayout
*pLay
= this;
2706 while( pLay
&& pLay
->IsDummy() )
2707 pLay
= pLay
->GetNext();
2710 const SwLinePortion
*pPos
= pLay
->GetNextPortion();
2711 while ( pPos
&& !pPos
->GetLen() )
2712 pPos
= pPos
->GetNextPortion();
2713 if( pPos
&& pPos
->IsDropPortion() )
2714 return static_cast<const SwDropPortion
*>(pPos
);
2715 pLay
= pLay
->GetLen() ? nullptr : pLay
->GetNext();
2720 void SwParaPortion::dumpAsXml(xmlTextWriterPtr pWriter
, const OUString
& rText
,
2721 TextFrameIndex
& nOffset
) const
2723 (void)xmlTextWriterStartElement(pWriter
, BAD_CAST("SwParaPortion"));
2724 dumpAsXmlAttributes(pWriter
, rText
, nOffset
);
2725 nOffset
+= GetLen();
2727 (void)xmlTextWriterEndElement(pWriter
);
2730 void SwLineLayout::Init( SwLinePortion
* pNextPortion
)
2734 SetLen(TextFrameIndex(0));
2737 SetNextPortion( pNextPortion
);
2740 // looks for hanging punctuation portions in the paragraph
2741 // and return the maximum right offset of them.
2742 // If no such portion is found, the Margin/Hanging-flags will be updated.
2743 SwTwips
SwLineLayout::GetHangingMargin_() const
2745 SwLinePortion
* pPor
= GetNextPortion();
2746 bool bFound
= false;
2750 if( pPor
->IsHangingPortion() )
2752 nDiff
= static_cast<SwHangingPortion
*>(pPor
)->GetInnerWidth() - pPor
->Width();
2756 // the last post its portion
2757 else if ( pPor
->IsPostItsPortion() && ! pPor
->GetNextPortion() )
2760 pPor
= pPor
->GetNextPortion();
2762 if( !bFound
) // update the hanging-flag
2763 const_cast<SwLineLayout
*>(this)->SetHanging( false );
2767 SwTwips
SwTextFrame::HangingMargin() const
2769 SAL_WARN_IF( !HasPara(), "sw.core", "Don't call me without a paraportion" );
2770 if( !GetPara()->IsMargin() )
2772 const SwLineLayout
* pLine
= GetPara();
2776 SwTwips nDiff
= pLine
->GetHangingMargin();
2779 pLine
= pLine
->GetNext();
2781 if( !nRet
) // update the margin-flag
2782 const_cast<SwParaPortion
*>(GetPara())->SetMargin( false );
2786 SwTwips
SwTextFrame::GetLowerMarginForFlyIntersect() const
2788 const IDocumentSettingAccess
& rIDSA
= GetDoc().getIDocumentSettingAccess();
2789 if (!rIDSA
.get(DocumentSettingId::TAB_OVER_MARGIN
))
2791 // Word >= 2013 style or Writer style: lower margin is ignored when determining the text
2796 const SwAttrSet
* pAttrSet
= GetTextNodeForParaProps()->GetpSwAttrSet();
2802 // If it has multiple lines, then probably it already has the needed fly portion.
2803 // Limit this to empty paragraphs for now.
2804 if ((GetPara() && GetPara()->GetNext()) || !GetText().isEmpty())
2809 return pAttrSet
->GetULSpace().GetLower();
2812 void SwScriptInfo::selectHiddenTextProperty(const SwTextNode
& rNode
,
2813 MultiSelection
& rHiddenMulti
,
2814 std::vector
<std::pair
<sw::mark::IBookmark
const*, MarkKind
>> *const pBookmarks
)
2816 assert((rNode
.GetText().isEmpty() && rHiddenMulti
.GetTotalRange().Len() == 1)
2817 || (rNode
.GetText().getLength() == rHiddenMulti
.GetTotalRange().Len()));
2819 const SvxCharHiddenItem
* pItem
= rNode
.GetSwAttrSet().GetItemIfSet( RES_CHRATR_HIDDEN
);
2820 if( pItem
&& pItem
->GetValue() )
2822 rHiddenMulti
.SelectAll();
2825 const SwpHints
* pHints
= rNode
.GetpSwpHints();
2829 for( size_t nTmp
= 0; nTmp
< pHints
->Count(); ++nTmp
)
2831 const SwTextAttr
* pTextAttr
= pHints
->Get( nTmp
);
2832 const SvxCharHiddenItem
* pHiddenItem
= CharFormat::GetItem( *pTextAttr
, RES_CHRATR_HIDDEN
);
2835 const sal_Int32 nSt
= pTextAttr
->GetStart();
2836 const sal_Int32 nEnd
= *pTextAttr
->End();
2839 Range
aTmp( nSt
, nEnd
- 1 );
2840 rHiddenMulti
.Select( aTmp
, pHiddenItem
->GetValue() );
2846 for (const SwContentIndex
* pIndex
= rNode
.GetFirstIndex(); pIndex
; pIndex
= pIndex
->GetNext())
2848 const sw::mark::IMark
* pMark
= pIndex
->GetMark();
2849 const sw::mark::IBookmark
* pBookmark
= dynamic_cast<const sw::mark::IBookmark
*>(pMark
);
2850 if (pBookmarks
&& pBookmark
)
2852 if (!pBookmark
->IsExpanded())
2854 pBookmarks
->emplace_back(pBookmark
, MarkKind::Point
);
2856 else if (pIndex
== &pBookmark
->GetMarkStart().nContent
)
2858 pBookmarks
->emplace_back(pBookmark
, MarkKind::Start
);
2862 assert(pIndex
== &pBookmark
->GetMarkEnd().nContent
);
2863 pBookmarks
->emplace_back(pBookmark
, MarkKind::End
);
2867 // condition is evaluated in DocumentFieldsManager::UpdateExpFields()
2868 if (pBookmark
&& pBookmark
->IsHidden())
2870 // intersect bookmark range with textnode range and add the intersection to rHiddenMulti
2872 const sal_Int32 nSt
= pBookmark
->GetMarkStart().GetContentIndex();
2873 const sal_Int32 nEnd
= pBookmark
->GetMarkEnd().GetContentIndex();
2877 Range
aTmp( nSt
, nEnd
- 1 );
2878 rHiddenMulti
.Select(aTmp
, true);
2884 void SwScriptInfo::selectRedLineDeleted(const SwTextNode
& rNode
, MultiSelection
&rHiddenMulti
, bool bSelect
)
2886 assert((rNode
.GetText().isEmpty() && rHiddenMulti
.GetTotalRange().Len() == 1)
2887 || (rNode
.GetText().getLength() == rHiddenMulti
.GetTotalRange().Len()));
2889 const IDocumentRedlineAccess
& rIDRA
= rNode
.getIDocumentRedlineAccess();
2890 if ( !IDocumentRedlineAccess::IsShowChanges( rIDRA
.GetRedlineFlags() ) )
2893 SwRedlineTable::size_type nAct
= rIDRA
.GetRedlinePos( rNode
, RedlineType::Any
);
2895 for ( ; nAct
< rIDRA
.GetRedlineTable().size(); nAct
++ )
2897 const SwRangeRedline
* pRed
= rIDRA
.GetRedlineTable()[ nAct
];
2899 if (pRed
->Start()->GetNode() > rNode
)
2902 if (pRed
->GetType() != RedlineType::Delete
)
2905 sal_Int32 nRedlStart
;
2906 sal_Int32 nRedlnEnd
;
2907 pRed
->CalcStartEnd( rNode
.GetIndex(), nRedlStart
, nRedlnEnd
);
2908 //clip it if the redline extends past the end of the nodes text
2909 nRedlnEnd
= std::min
<sal_Int32
>(nRedlnEnd
, rNode
.GetText().getLength());
2910 if ( nRedlnEnd
> nRedlStart
)
2912 Range
aTmp( nRedlStart
, nRedlnEnd
- 1 );
2913 rHiddenMulti
.Select( aTmp
, bSelect
);
2918 // Returns a MultiSection indicating the hidden ranges.
2919 void SwScriptInfo::CalcHiddenRanges( const SwTextNode
& rNode
,
2920 MultiSelection
& rHiddenMulti
,
2921 std::vector
<std::pair
<sw::mark::IBookmark
const*, MarkKind
>> *const pBookmarks
)
2923 selectHiddenTextProperty(rNode
, rHiddenMulti
, pBookmarks
);
2925 // If there are any hidden ranges in the current text node, we have
2926 // to unhide the redlining ranges:
2927 selectRedLineDeleted(rNode
, rHiddenMulti
, false);
2929 // We calculated a lot of stuff. Finally we can update the flags at the text node.
2931 const bool bNewContainsHiddenChars
= rHiddenMulti
.GetRangeCount() > 0;
2932 bool bNewHiddenCharsHidePara
= false;
2933 if ( bNewContainsHiddenChars
)
2935 const Range
& rRange
= rHiddenMulti
.GetRange( 0 );
2936 const sal_Int32 nHiddenStart
= rRange
.Min();
2937 const sal_Int32 nHiddenEnd
= rRange
.Max() + 1;
2938 bNewHiddenCharsHidePara
=
2939 (nHiddenStart
== 0 && nHiddenEnd
>= rNode
.GetText().getLength());
2941 rNode
.SetHiddenCharAttribute( bNewHiddenCharsHidePara
, bNewContainsHiddenChars
);
2944 TextFrameIndex
SwScriptInfo::CountCJKCharacters(const OUString
&rText
,
2945 TextFrameIndex nPos
, TextFrameIndex
const nEnd
, LanguageType aLang
)
2947 TextFrameIndex
nCount(0);
2950 sal_Int32 nDone
= 0;
2951 const lang::Locale
&rLocale
= g_pBreakIt
->GetLocale( aLang
);
2952 while ( nPos
< nEnd
)
2954 nPos
= TextFrameIndex(g_pBreakIt
->GetBreakIter()->nextCharacters(
2955 rText
, sal_Int32(nPos
),
2957 i18n::CharacterIteratorMode::SKIPCELL
, 1, nDone
));
2962 nCount
= nEnd
- nPos
;
2967 void SwScriptInfo::CJKJustify( const OUString
& rText
, KernArray
& rKernArray
,
2968 TextFrameIndex
const nStt
,
2969 TextFrameIndex
const nLen
, LanguageType aLang
,
2970 tools::Long nSpaceAdd
, bool bIsSpaceStop
)
2972 assert( sal_Int32(nStt
) >= 0 );
2973 if (sal_Int32(nLen
) <= 0)
2976 tools::Long nSpaceSum
= 0;
2977 const lang::Locale
&rLocale
= g_pBreakIt
->GetLocale( aLang
);
2978 sal_Int32 nDone
= 0;
2979 sal_Int32
nNext(nStt
);
2980 for ( sal_Int32 nI
= 0; nI
< sal_Int32(nLen
); ++nI
)
2982 if (nI
+ sal_Int32(nStt
) == nNext
)
2984 nNext
= g_pBreakIt
->GetBreakIter()->nextCharacters( rText
, nNext
,
2986 i18n::CharacterIteratorMode::SKIPCELL
, 1, nDone
);
2987 if (nNext
< sal_Int32(nStt
+ nLen
) || !bIsSpaceStop
)
2988 nSpaceSum
+= nSpaceAdd
;
2990 rKernArray
.adjust(nI
, nSpaceSum
);
2993 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */