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>
21 #include <sal/log.hxx>
23 #include <DocumentContentOperationsManager.hxx>
24 #include <hintids.hxx>
25 #include <editeng/rsiditem.hxx>
26 #include <osl/diagnose.h>
27 #include <svl/whiter.hxx>
28 #include <svl/itemiter.hxx>
29 #include <editeng/charhiddenitem.hxx>
30 #include <editeng/langitem.hxx>
31 #include <editeng/lrspitem.hxx>
32 #include <txtinet.hxx>
33 #include <txtflcnt.hxx>
35 #include <fmtrfmrk.hxx>
36 #include <fmtanchr.hxx>
37 #include <fmtinfmt.hxx>
39 #include <fchrfmt.hxx>
40 #include <fmtautofmt.hxx>
41 #include <fmtflcnt.hxx>
43 #include <txttxmrk.hxx>
44 #include <txtrfmrk.hxx>
46 #include <textlinebreak.hxx>
48 #include <txtannotationfld.hxx>
49 #include <unotools/fltrcfg.hxx>
50 #include <charfmt.hxx>
53 #include <fmtruby.hxx>
54 #include <fmtmeta.hxx>
55 #include <formatcontentcontrol.hxx>
56 #include <textcontentcontrol.hxx>
57 #include <breakit.hxx>
59 #include <IDocumentUndoRedo.hxx>
60 #include <IDocumentFieldsAccess.hxx>
61 #include <IDocumentLayoutAccess.hxx>
62 #include <IDocumentStylePoolAccess.hxx>
67 #include <rootfrm.hxx>
70 #include <docufld.hxx>
73 #include <poolfmt.hxx>
74 #include <istyleaccess.hxx>
80 #include <rdfhelper.hxx>
84 #define CHECK Check(true);
85 #define CHECK_NOTMERGED Check(false);
87 #define CHECK_NOTMERGED
90 using namespace ::com::sun::star::i18n
;
92 SwpHints::SwpHints(const SwTextNode
& rParent
)
95 , m_bInSplitNode(false)
96 , m_bCalcHiddenParaField(false)
97 , m_bHiddenByParaField(false)
100 , m_bStartMapNeedsSorting(false)
101 , m_bEndMapNeedsSorting(false)
102 , m_bWhichMapNeedsSorting(false)
106 static void TextAttrDelete( SwDoc
& rDoc
, SwTextAttr
* const pAttr
)
108 if (RES_TXTATR_META
== pAttr
->Which() ||
109 RES_TXTATR_METAFIELD
== pAttr
->Which())
111 static_txtattr_cast
<SwTextMeta
*>(pAttr
)->ChgTextNode(nullptr); // prevents ASSERT
113 else if (pAttr
->Which() == RES_TXTATR_CONTENTCONTROL
)
115 static_txtattr_cast
<SwTextContentControl
*>(pAttr
)->ChgTextNode(nullptr);
117 SwTextAttr::Destroy( pAttr
, rDoc
.GetAttrPool() );
120 static bool TextAttrContains(const sal_Int32 nPos
, const SwTextAttrEnd
* const pAttr
)
122 return (pAttr
->GetStart() < nPos
) && (nPos
< *pAttr
->End());
127 // |---| => valid: b before a
128 // |-----| => valid: start == end; b before a
129 // |---------| => invalid: overlap (1)
130 // |-----------| => valid: same end; b around a
131 // |-----------------| => valid: b around a
132 // |---| => valid; same start; b within a
133 // |-----| => valid; same start and end; b around or within a?
134 // |-----------| => valid: same start: b around a
135 // |-| => valid: b within a
136 // |---| => valid: same end; b within a
137 // |---------| => invalid: overlap (2)
138 // |-----| => valid: end == start; b after a
139 // |---| => valid: b after a
140 // ===> 2 invalid overlap cases
142 bool isOverlap(const sal_Int32 nStart1
, const sal_Int32 nEnd1
,
143 const sal_Int32 nStart2
, const sal_Int32 nEnd2
)
146 ((nStart1
> nStart2
) && (nStart1
< nEnd2
) && (nEnd1
> nEnd2
)) // (1)
147 || ((nStart1
< nStart2
) && (nStart2
< nEnd1
) && (nEnd1
< nEnd2
)); // (2)
150 /// #i106930#: now asymmetric: empty hint1 is _not_ nested, but empty hint2 is
152 bool isNestedAny(const sal_Int32 nStart1
, const sal_Int32 nEnd1
,
153 const sal_Int32 nStart2
, const sal_Int32 nEnd2
)
155 return ((nStart1
== nStart2
) || (nEnd1
== nEnd2
))
156 // same start/end: nested except if hint1 empty and hint2 not empty
157 ? (nStart1
!= nEnd1
) || (nStart2
== nEnd2
)
158 : ((nStart1
< nStart2
) ? (nEnd1
>= nEnd2
) : (nEnd1
<= nEnd2
));
162 bool isSelfNestable(const sal_uInt16 nWhich
)
164 if ((RES_TXTATR_INETFMT
== nWhich
) ||
165 (RES_TXTATR_CJK_RUBY
== nWhich
) ||
166 (RES_TXTATR_INPUTFIELD
== nWhich
))
168 assert((RES_TXTATR_META
== nWhich
) ||
169 (RES_TXTATR_METAFIELD
== nWhich
) ||
170 (RES_TXTATR_CONTENTCONTROL
== nWhich
));
175 bool isSplittable(const sal_uInt16 nWhich
)
177 if ((RES_TXTATR_INETFMT
== nWhich
) ||
178 (RES_TXTATR_CJK_RUBY
== nWhich
))
180 assert((RES_TXTATR_META
== nWhich
) ||
181 (RES_TXTATR_METAFIELD
== nWhich
) ||
182 (RES_TXTATR_INPUTFIELD
== nWhich
) ||
183 (RES_TXTATR_CONTENTCONTROL
== nWhich
));
189 enum Split_t
{ FAIL
, SPLIT_NEW
, SPLIT_OTHER
};
194 Calculate splitting policy for overlapping hints, based on what kind of
195 hint is inserted, and what kind of existing hint overlaps.
198 splitPolicy(const sal_uInt16 nWhichNew
, const sal_uInt16 nWhichOther
)
200 if (!isSplittable(nWhichOther
))
202 if (!isSplittable(nWhichNew
))
209 if ( RES_TXTATR_INPUTFIELD
== nWhichNew
)
211 else if ( (RES_TXTATR_INETFMT
== nWhichNew
) &&
212 (RES_TXTATR_CJK_RUBY
== nWhichOther
) )
219 void SwTextINetFormat::InitINetFormat(SwTextNode
& rNode
)
222 SwCharFormat
* const pFormat(
223 rNode
.GetDoc().getIDocumentStylePoolAccess().GetCharFormatFromPool(RES_POOLCHR_INET_NORMAL
) );
224 pFormat
->Add( this );
227 void SwTextRuby::InitRuby(SwTextNode
& rNode
)
230 SwCharFormat
* const pFormat(
231 rNode
.GetDoc().getIDocumentStylePoolAccess().GetCharFormatFromPool(RES_POOLCHR_RUBYTEXT
) );
232 pFormat
->Add( this );
236 Create a new nesting text hint.
238 static SwTextAttrNesting
*
239 MakeTextAttrNesting(SwTextNode
& rNode
, SwTextAttrNesting
& rNesting
,
240 const sal_Int32 nStart
, const sal_Int32 nEnd
)
242 SwTextAttr
* const pNew( MakeTextAttr(
243 rNode
.GetDoc(), rNesting
.GetAttr(), nStart
, nEnd
) );
244 switch (pNew
->Which())
246 case RES_TXTATR_INETFMT
:
248 static_txtattr_cast
<SwTextINetFormat
*>(pNew
)->InitINetFormat(rNode
);
251 case RES_TXTATR_CJK_RUBY
:
253 static_txtattr_cast
<SwTextRuby
*>(pNew
)->InitRuby(rNode
);
257 assert(!"MakeTextAttrNesting: what the hell is that?");
260 return static_txtattr_cast
<SwTextAttrNesting
*>(pNew
);
263 typedef std::vector
<SwTextAttrNesting
*> NestList_t
;
265 static NestList_t::iterator
266 lcl_DoSplitImpl(NestList_t
& rSplits
, SwTextNode
& rNode
,
267 NestList_t::iterator
const iter
, sal_Int32
const nSplitPos
,
268 bool const bSplitAtStart
, bool const bOtherDummy
)
270 const sal_Int32
nStartPos( // skip other's dummy character!
271 (bSplitAtStart
&& bOtherDummy
) ? nSplitPos
+ 1 : nSplitPos
);
272 SwTextAttrNesting
* const pNew( MakeTextAttrNesting(
273 rNode
, **iter
, nStartPos
, *(*iter
)->GetEnd() ) );
274 (*iter
)->SetEnd(nSplitPos
);
275 return rSplits
.insert(iter
+ 1, pNew
);
279 lcl_DoSplitNew(NestList_t
& rSplits
, SwTextNode
& rNode
,
280 const sal_Int32 nNewStart
,
281 const sal_Int32 nOtherStart
, const sal_Int32 nOtherEnd
, bool bOtherDummy
)
283 const bool bSplitAtStart(nNewStart
< nOtherStart
);
284 const sal_Int32
nSplitPos( bSplitAtStart
? nOtherStart
: nOtherEnd
);
285 // first find the portion that is split (not necessarily the last one!)
286 NestList_t::iterator
const iter(
287 std::find_if( rSplits
.begin(), rSplits
.end(),
288 [nSplitPos
](SwTextAttrEnd
* const pAttr
) {
289 return TextAttrContains(nSplitPos
, pAttr
);
291 if (iter
!= rSplits
.end()) // already split here?
293 lcl_DoSplitImpl(rSplits
, rNode
, iter
, nSplitPos
, bSplitAtStart
, bOtherDummy
);
298 Insert nesting hint into the hints array. Also calls NoteInHistory.
299 @param rNewHint the hint to be inserted (must not overlap existing!)
301 void SwpHints::InsertNesting(SwTextAttrNesting
& rNewHint
)
304 NoteInHistory( & rNewHint
, true );
309 The following hints correspond to well-formed XML elements in ODF:
310 RES_TXTATR_INETFMT, RES_TXTATR_CJK_RUBY, RES_TXTATR_META, RES_TXTATR_METAFIELD,
311 RES_TXTATR_CONTENTCONTROL
313 The writer core must ensure that these do not overlap; if they did,
314 the document would not be storable as ODF.
316 Also, a Hyperlink must not be nested within another Hyperlink,
317 and a Ruby must not be nested within another Ruby.
319 The ODF export in xmloff will only put a hyperlink into a ruby, never a ruby
322 Unfortunately the UNO API for Hyperlink and Ruby consists of the properties
323 Hyperlink* and Ruby* of the css.text.CharacterProperties service. In other
324 words, they are treated as formatting attributes, not as content entities.
325 Furthermore, for API users it is not possible to easily test whether a certain
326 range would be overlapping with other nested attributes, and most importantly,
327 <em>which ones</em>, so we can hardly refuse to insert these in cases of
330 It is possible to split Hyperlink and Ruby into multiple portions, such that
331 the result is properly nested.
333 meta and meta-field must not be split, because they have xml:id.
335 content controls should not split, either.
337 These constraints result in the following design:
341 inserts n attributes split at RES_TXTATR_CJK_RUBY, RES_TXTATR_META,
343 may replace existing RES_TXTATR_INETFMT at overlap
346 inserts n attributes split at RES_TXTATR_META, RES_TXTATR_METAFIELD
347 may replace existing RES_TXTATR_CJK_RUBY at overlap
348 may split existing overlapping RES_TXTATR_INETFMT
350 may fail if overlapping existing RES_TXTATR_META/RES_TXTATR_METAFIELD
351 may split existing overlapping RES_TXTATR_INETFMT or RES_TXTATR_CJK_RUBY
353 RES_TXTATR_METAFIELD:
354 may fail if overlapping existing RES_TXTATR_META/RES_TXTATR_METAFIELD
355 may split existing overlapping RES_TXTATR_INETFMT or RES_TXTATR_CJK_RUBY
358 The nesting is expressed by the position of the hints.
359 RES_TXTATR_META and RES_TXTATR_METAFIELD have a CH_TXTATR, and there can
360 only be one such hint starting and ending at a given position.
361 Only RES_TXTATR_INETFMT and RES_TXTATR_CJK_RUBY lack a CH_TXTATR.
362 The interpretation given is that RES_TXTATR_CJK_RUBY is always around
363 a RES_TXTATR_INETFMT at the same start and end position (which corresponds
365 Both of these are always around a nesting hint with CH_TXTATR at the same
366 start and end position (if they should be inside, then the start should be
367 after the CH_TXTATR).
368 It would probably be a bad idea to add another nesting hint without
369 CH_TXTATR; on the other hand, it would be difficult adding a CH_TXTATR to
370 RES_TXTATR_INETFMT and RES_TXTATR_CJK_RUBY, due to the overwriting and
371 splitting of existing hints that is necessary for backward compatibility.
373 @param rNode the text node
374 @param rHint the hint to be inserted
375 @returns true iff hint was successfully inserted
378 SwpHints::TryInsertNesting( SwTextNode
& rNode
, SwTextAttrNesting
& rNewHint
)
380 // INVARIANT: the nestable hints in the array are properly nested
381 const sal_uInt16
nNewWhich( rNewHint
.Which() );
382 const sal_Int32
nNewStart( rNewHint
.GetStart() );
383 const sal_Int32
nNewEnd ( *rNewHint
.GetEnd() );
384 const bool bNewSelfNestable( isSelfNestable(nNewWhich
) );
386 assert( (RES_TXTATR_INETFMT
== nNewWhich
) ||
387 (RES_TXTATR_CJK_RUBY
== nNewWhich
) ||
388 (RES_TXTATR_META
== nNewWhich
) ||
389 (RES_TXTATR_METAFIELD
== nNewWhich
) ||
390 (RES_TXTATR_CONTENTCONTROL
== nNewWhich
) ||
391 (RES_TXTATR_INPUTFIELD
== nNewWhich
));
393 NestList_t OverlappingExisting
; // existing hints to be split
394 NestList_t OverwrittenExisting
; // existing hints to be replaced
395 NestList_t SplitNew
; // new hints to be inserted
397 SplitNew
.push_back(& rNewHint
);
399 // pass 1: split the inserted hint into fragments if necessary
400 for ( size_t i
= 0; i
< Count(); ++i
)
402 SwTextAttr
* const pOther
= GetSortedByEnd(i
);
404 if (pOther
->IsNesting())
406 const sal_uInt16
nOtherWhich( pOther
->Which() );
407 const sal_Int32
nOtherStart( pOther
->GetStart() );
408 const sal_Int32
nOtherEnd ( *pOther
->GetEnd() );
409 if (isOverlap(nNewStart
, nNewEnd
, nOtherStart
, nOtherEnd
))
411 switch (splitPolicy(nNewWhich
, nOtherWhich
))
414 SAL_INFO("sw.core", "cannot insert hint: overlap");
415 for (const auto& aSplit
: SplitNew
)
416 TextAttrDelete(rNode
.GetDoc(), aSplit
);
419 lcl_DoSplitNew(SplitNew
, rNode
, nNewStart
,
420 nOtherStart
, nOtherEnd
, pOther
->HasDummyChar());
423 OverlappingExisting
.push_back(
424 static_txtattr_cast
<SwTextAttrNesting
*>(pOther
));
427 assert(!"bad code monkey");
431 else if (isNestedAny(nNewStart
, nNewEnd
, nOtherStart
, nOtherEnd
))
433 if (!bNewSelfNestable
&& (nNewWhich
== nOtherWhich
))
435 // ruby and hyperlink: if there is nesting, _overwrite_
436 OverwrittenExisting
.push_back(
437 static_txtattr_cast
<SwTextAttrNesting
*>(pOther
));
439 else if ((nNewStart
== nOtherStart
) && pOther
->HasDummyChar())
441 if (rNewHint
.HasDummyChar())
443 assert(!"ERROR: inserting duplicate CH_TXTATR hint");
445 } else if (nNewEnd
< nOtherEnd
) {
446 // other has dummy char, new is inside other, but
447 // new contains the other's dummy char?
448 // should be corrected because it may lead to problems
449 // in SwXMeta::createEnumeration
450 // SplitNew is sorted, so this is the first split
451 assert(SplitNew
.front()->GetStart() == nNewStart
);
452 SplitNew
.front()->SetStart(nNewStart
+ 1);
459 // pass 1b: tragically need to check for fieldmarks here too
460 for (auto iter
= SplitNew
.begin(); iter
!= SplitNew
.end(); ++iter
)
462 SwPaM
const temp(rNode
, (*iter
)->GetStart(), rNode
, *(*iter
)->GetEnd());
463 std::vector
<std::pair
<SwNodeOffset
, sal_Int32
>> Breaks
;
464 sw::CalcBreaks(Breaks
, temp
, true);
467 if (!isSplittable(nNewWhich
))
469 SAL_INFO("sw.core", "cannot insert hint: fieldmark overlap");
470 assert(SplitNew
.size() == 1);
471 TextAttrDelete(rNode
.GetDoc(), &rNewHint
);
476 for (auto const& rPos
: Breaks
)
478 assert(rPos
.first
== rNode
.GetIndex());
479 iter
= lcl_DoSplitImpl(SplitNew
, rNode
, iter
,
480 rPos
.second
, true, true);
486 assert((isSplittable(nNewWhich
) || SplitNew
.size() == 1) &&
487 "splitting the unsplittable ???");
489 // pass 2: split existing hints that overlap/nest with new hint
490 // do not iterate over hints array, but over remembered set of overlapping
491 // hints, to keep things simple w.r.t. insertion/removal
492 // N.B: if there is a hint that splits the inserted hint, then
493 // that hint would also have already split any hint in OverlappingExisting
494 // so any hint in OverlappingExisting can be split at most by one hint
495 // in SplitNew, or even not at all (this is not true for existing hints
496 // that go _around_ new hint, which is the reason d'^etre for pass 4)
497 for (auto& rpOther
: OverlappingExisting
)
499 const sal_Int32
nOtherStart( rpOther
->GetStart() );
500 const sal_Int32
nOtherEnd ( *rpOther
->GetEnd() );
502 for (const auto& rpNew
: SplitNew
)
504 const sal_Int32
nSplitNewStart( rpNew
->GetStart() );
505 const sal_Int32
nSplitNewEnd ( *rpNew
->GetEnd() );
506 // 4 cases: within, around, overlap l, overlap r, (OTHER: no action)
507 const bool bRemoveOverlap(
508 !bNewSelfNestable
&& (nNewWhich
== rpOther
->Which()) );
510 switch (ComparePosition(nSplitNewStart
, nSplitNewEnd
,
511 nOtherStart
, nOtherEnd
))
513 case SwComparePosition::Inside
:
515 assert(!bRemoveOverlap
&&
516 "this one should be in OverwrittenExisting?");
519 case SwComparePosition::Outside
:
520 case SwComparePosition::Equal
:
522 assert(!"existing hint inside new hint: why?");
525 case SwComparePosition::OverlapBefore
:
527 Delete( rpOther
); // this also does NoteInHistory!
528 rpOther
->SetStart(nSplitNewEnd
);
529 InsertNesting( *rpOther
);
532 if ( MAX_HINTS
<= Count() )
534 SAL_INFO("sw.core", "hints array full :-(");
537 SwTextAttrNesting
* const pOtherLeft(
538 MakeTextAttrNesting( rNode
, *rpOther
,
539 nOtherStart
, nSplitNewEnd
) );
540 InsertNesting( *pOtherLeft
);
544 case SwComparePosition::OverlapBehind
:
546 Delete( rpOther
); // this also does NoteInHistory!
547 rpOther
->SetEnd(nSplitNewStart
);
548 InsertNesting( *rpOther
);
551 if ( MAX_HINTS
<= Count() )
553 SAL_INFO("sw.core", "hints array full :-(");
556 SwTextAttrNesting
* const pOtherRight(
557 MakeTextAttrNesting( rNode
, *rpOther
,
558 nSplitNewStart
, nOtherEnd
) );
559 InsertNesting( *pOtherRight
);
564 break; // overlap resolved by splitting new: nothing to do
569 if ( MAX_HINTS
<= Count() || MAX_HINTS
- Count() <= SplitNew
.size() )
571 SAL_INFO("sw.core", "hints array full :-(");
575 // pass 3: insert new hints
576 for (const auto& rpHint
: SplitNew
)
578 InsertNesting(*rpHint
);
581 // pass 4: handle overwritten hints
582 // RES_TXTATR_INETFMT and RES_TXTATR_CJK_RUBY should displace attributes
584 for (auto& rpOther
: OverwrittenExisting
)
586 const sal_Int32
nOtherStart( rpOther
->GetStart() );
587 const sal_Int32
nOtherEnd ( *rpOther
->GetEnd() );
589 // overwritten portion is given by start/end of inserted hint
590 if ((nNewStart
<= nOtherStart
) && (nOtherEnd
<= nNewEnd
))
593 rNode
.DestroyAttr( rpOther
);
597 assert((nOtherStart
< nNewStart
) || (nNewEnd
< nOtherEnd
));
598 // scenario: there is a RUBY, and contained within that a META;
599 // now a RUBY is inserted within the META => the existing RUBY is split:
600 // here it is not possible to simply insert the left/right fragment
601 // of the existing RUBY because they <em>overlap</em> with the META!
602 Delete( rpOther
); // this also does NoteInHistory!
603 if (nNewEnd
< nOtherEnd
)
605 SwTextAttrNesting
* const pOtherRight(
607 rNode
, *rpOther
, nNewEnd
, nOtherEnd
) );
608 bool const bSuccess( TryInsertNesting(rNode
, *pOtherRight
) );
609 SAL_WARN_IF(!bSuccess
, "sw.core", "recursive call 1 failed?");
611 if (nOtherStart
< nNewStart
)
613 rpOther
->SetEnd(nNewStart
);
614 bool const bSuccess( TryInsertNesting(rNode
, *rpOther
) );
615 SAL_WARN_IF(!bSuccess
, "sw.core", "recursive call 2 failed?");
619 rNode
.DestroyAttr(rpOther
);
627 // This function takes care for the following text attribute:
628 // RES_TXTATR_CHARFMT, RES_TXTATR_AUTOFMT
629 // These attributes have to be handled in a special way (Portion building).
631 // The new attribute will be split by any existing RES_TXTATR_AUTOFMT or
632 // RES_TXTATR_CHARFMT. The new attribute itself will
633 // split any existing RES_TXTATR_AUTOFMT or RES_TXTATR_CHARFMT.
635 void SwpHints::BuildPortions( SwTextNode
& rNode
, SwTextAttr
& rNewHint
,
636 const SetAttrMode nMode
)
638 const sal_uInt16 nWhich
= rNewHint
.Which();
640 const sal_Int32 nThisStart
= rNewHint
.GetStart();
641 const sal_Int32 nThisEnd
= *rNewHint
.GetEnd();
642 const bool bNoLengthAttribute
= nThisStart
== nThisEnd
;
644 std::vector
<SwTextAttr
*> aInsDelHints
;
646 assert( RES_TXTATR_CHARFMT
== rNewHint
.Which() ||
647 RES_TXTATR_AUTOFMT
== rNewHint
.Which() );
649 // 2. Find the hints which cover the start and end position
650 // of the new hint. These hints have to be split into two portions:
652 if ( !bNoLengthAttribute
) // nothing to do for no length attributes
654 for ( size_t i
= 0; i
< Count(); ++i
)
656 // we're modifying stuff here which affects the sorting, and we
657 // don't want it changing underneath us
658 SwTextAttr
* pOther
= GetWithoutResorting(i
);
660 if ( RES_TXTATR_CHARFMT
!= pOther
->Which() &&
661 RES_TXTATR_AUTOFMT
!= pOther
->Which() )
664 sal_Int32 nOtherStart
= pOther
->GetStart();
665 const sal_Int32 nOtherEnd
= *pOther
->GetEnd();
667 // Check if start of new attribute overlaps with pOther:
668 // Split pOther if necessary:
669 if ( nOtherStart
< nThisStart
&& nThisStart
< nOtherEnd
)
671 SwTextAttr
* pNewAttr
= MakeTextAttr( rNode
.GetDoc(),
672 pOther
->GetAttr(), nOtherStart
, nThisStart
);
673 if ( RES_TXTATR_CHARFMT
== pOther
->Which() )
675 static_txtattr_cast
<SwTextCharFormat
*>(pNewAttr
)->SetSortNumber(
676 static_txtattr_cast
<SwTextCharFormat
*>(pOther
)->GetSortNumber() );
678 aInsDelHints
.push_back( pNewAttr
);
680 NoteInHistory( pOther
);
681 pOther
->SetStart(nThisStart
);
682 NoteInHistory( pOther
, true );
684 nOtherStart
= nThisStart
;
687 // Check if end of new attribute overlaps with pOther:
688 // Split pOther if necessary:
689 if ( nOtherStart
< nThisEnd
&& nThisEnd
< nOtherEnd
)
691 SwTextAttr
* pNewAttr
= MakeTextAttr( rNode
.GetDoc(),
692 pOther
->GetAttr(), nOtherStart
, nThisEnd
);
693 if ( RES_TXTATR_CHARFMT
== pOther
->Which() )
695 static_txtattr_cast
<SwTextCharFormat
*>(pNewAttr
)->SetSortNumber(
696 static_txtattr_cast
<SwTextCharFormat
*>(pOther
)->GetSortNumber());
698 aInsDelHints
.push_back( pNewAttr
);
700 NoteInHistory( pOther
);
701 pOther
->SetStart(nThisEnd
);
702 NoteInHistory( pOther
, true );
706 // Insert the newly created attributes:
707 for ( const auto& rpHint
: aInsDelHints
)
710 NoteInHistory( rpHint
, true );
715 if( !rNode
.GetDoc().IsInReading() )
716 CHECK_NOTMERGED
; // ignore flags not set properly yet, don't check them
719 // 4. Split rNewHint into 1 ... n new hints:
721 o3tl::sorted_vector
<sal_Int32
> aBounds
;
722 aBounds
.insert( nThisStart
);
723 aBounds
.insert( nThisEnd
);
725 if ( !bNoLengthAttribute
) // nothing to do for no length attributes
727 for ( size_t i
= 0; i
< Count(); ++i
)
729 const SwTextAttr
* pOther
= Get(i
);
731 if ( RES_TXTATR_CHARFMT
!= pOther
->Which() &&
732 RES_TXTATR_AUTOFMT
!= pOther
->Which() )
735 const sal_Int32 nOtherStart
= pOther
->GetStart();
736 const sal_Int32 nOtherEnd
= *pOther
->End();
738 if (nThisStart
<= nOtherStart
&& nOtherStart
<= nThisEnd
)
739 aBounds
.insert( nOtherStart
);
740 if (nThisStart
<= nOtherEnd
&& nOtherEnd
<= nThisEnd
)
741 aBounds
.insert( nOtherEnd
);
745 auto aStartIter
= aBounds
.lower_bound( nThisStart
);
746 auto aEndIter
= aBounds
.upper_bound( nThisEnd
);
747 sal_Int32 nPorStart
= *aStartIter
;
749 bool bDestroyHint
= true;
751 // Insert the 1...n new parts of the new attribute:
753 while ( aStartIter
!= aEndIter
|| bNoLengthAttribute
)
755 OSL_ENSURE( bNoLengthAttribute
|| nPorStart
< *aStartIter
, "AUTOSTYLES: BuildPortion trouble" );
757 const sal_Int32 nPorEnd
= bNoLengthAttribute
? nPorStart
: *aStartIter
;
758 aInsDelHints
.clear();
760 // Get all hints that are in [nPorStart, nPorEnd[:
761 for ( size_t i
= 0; i
< Count(); ++i
)
763 // we get called from TryInsertHint, which changes ordering
764 SwTextAttr
*pOther
= GetWithoutResorting(i
);
766 if ( RES_TXTATR_CHARFMT
!= pOther
->Which() &&
767 RES_TXTATR_AUTOFMT
!= pOther
->Which() )
770 const sal_Int32 nOtherStart
= pOther
->GetStart();
772 if ( nOtherStart
> nPorStart
)
775 if ( pOther
->GetEnd() && *pOther
->GetEnd() == nPorEnd
&& nOtherStart
== nPorStart
)
777 OSL_ENSURE( *pOther
->GetEnd() == nPorEnd
, "AUTOSTYLES: BuildPortion trouble" );
778 aInsDelHints
.push_back( pOther
);
782 SwTextAttr
* pNewAttr
= nullptr;
783 if ( RES_TXTATR_CHARFMT
== nWhich
)
785 // pNewHint can be inserted after calculating the sort value.
786 // This should ensure, that pNewHint comes behind the already present
788 sal_uInt16 nCharStyleCount
= 0;
789 for ( const auto& rpHint
: aInsDelHints
)
791 if ( RES_TXTATR_CHARFMT
== rpHint
->Which() )
794 const SwFormatCharFormat
& rOtherCharFormat
= rpHint
->GetCharFormat();
795 const SwFormatCharFormat
& rThisCharFormat
= rNewHint
.GetCharFormat();
796 const bool bSameCharFormat
= rOtherCharFormat
.GetCharFormat() == rThisCharFormat
.GetCharFormat();
799 // Do not remove existing character format hint during XML import
800 if ( !rNode
.GetDoc().IsInXMLImport() &&
801 ( !( SetAttrMode::DONTREPLACE
& nMode
) ||
802 bNoLengthAttribute
||
807 rNode
.DestroyAttr( rpHint
);
814 // remove all attributes from auto styles, which are explicitly set in
815 // the new character format:
816 OSL_ENSURE( RES_TXTATR_AUTOFMT
== rpHint
->Which(), "AUTOSTYLES - Misc trouble" );
817 SwTextAttr
* pOther
= rpHint
;
818 const std::shared_ptr
<SfxItemSet
> & pOldStyle
= static_cast<const SwFormatAutoFormat
&>(pOther
->GetAttr()).GetStyleHandle();
820 // For each attribute in the automatic style check if it
821 // is also set the new character style:
822 SfxItemSet
aNewSet( *pOldStyle
->GetPool(),
823 aCharAutoFormatSetRange
);
824 SfxItemIter
aItemIter( *pOldStyle
);
825 const SfxPoolItem
* pItem
= aItemIter
.GetCurItem();
828 if ( !CharFormat::IsItemIncluded( pItem
->Which(), &rNewHint
) )
830 aNewSet
.Put( *pItem
);
833 pItem
= aItemIter
.NextItem();
838 rNode
.DestroyAttr( pOther
);
840 // Create new AutoStyle
841 if ( aNewSet
.Count() )
843 pNewAttr
= MakeTextAttr( rNode
.GetDoc(),
844 aNewSet
, nPorStart
, nPorEnd
);
846 NoteInHistory( pNewAttr
, true );
851 // If there is no current hint and start and end of rNewHint
852 // is ok, we do not need to create a new txtattr.
853 if ( nPorStart
== nThisStart
&&
854 nPorEnd
== nThisEnd
&&
857 pNewAttr
= &rNewHint
;
858 bDestroyHint
= false;
862 pNewAttr
= MakeTextAttr( rNode
.GetDoc(), rNewHint
.GetAttr(),
863 nPorStart
, nPorEnd
);
864 static_txtattr_cast
<SwTextCharFormat
*>(pNewAttr
)->SetSortNumber(nCharStyleCount
);
869 // Find the current autostyle. Mix attributes if necessary.
870 SwTextAttr
* pCurrentAutoStyle
= nullptr;
871 SwTextAttr
* pCurrentCharFormat
= nullptr;
872 for ( const auto& rpHint
: aInsDelHints
)
874 if ( RES_TXTATR_AUTOFMT
== rpHint
->Which() )
875 pCurrentAutoStyle
= rpHint
;
876 else if ( RES_TXTATR_CHARFMT
== rpHint
->Which() )
877 pCurrentCharFormat
= rpHint
;
880 std::shared_ptr
<SfxItemSet
> pNewStyle
= static_cast<const SwFormatAutoFormat
&>(rNewHint
.GetAttr()).GetStyleHandle();
881 if ( pCurrentAutoStyle
)
883 const std::shared_ptr
<SfxItemSet
> & pCurrentStyle
= static_cast<const SwFormatAutoFormat
&>(pCurrentAutoStyle
->GetAttr()).GetStyleHandle();
886 SfxItemSet
aNewSet( *pCurrentStyle
);
887 aNewSet
.Put( *pNewStyle
);
889 // #i75750# Remove attributes already set at whole paragraph
890 // #i81764# This should not be applied for no length attributes!!! <--
891 if ( !bNoLengthAttribute
&& rNode
.HasSwAttrSet() && aNewSet
.Count() )
893 SfxItemIter
aIter2( aNewSet
);
894 const SfxPoolItem
* pItem
= aIter2
.GetCurItem();
895 const SfxItemSet
& rWholeParaAttrSet
= rNode
.GetSwAttrSet();
899 const SfxPoolItem
* pTmpItem
= nullptr;
900 if ( SfxItemState::SET
== rWholeParaAttrSet
.GetItemState( pItem
->Which(), false, &pTmpItem
) &&
903 // Do not clear item if the attribute is set in a character format:
904 if ( !pCurrentCharFormat
|| nullptr == CharFormat::GetItem( *pCurrentCharFormat
, pItem
->Which() ) )
908 while ((pItem
= aIter2
.NextItem()));
912 Delete( pCurrentAutoStyle
);
913 rNode
.DestroyAttr( pCurrentAutoStyle
);
915 // Create new AutoStyle
916 if ( aNewSet
.Count() )
917 pNewAttr
= MakeTextAttr( rNode
.GetDoc(), aNewSet
,
918 nPorStart
, nPorEnd
);
922 // Remove any attributes which are already set at the whole paragraph:
923 bool bOptimizeAllowed
= true;
925 // #i75750# Remove attributes already set at whole paragraph
926 // #i81764# This should not be applied for no length attributes!!! <--
927 if ( !bNoLengthAttribute
&& rNode
.HasSwAttrSet() && pNewStyle
->Count() )
929 std::unique_ptr
<SfxItemSet
> pNewSet
;
931 SfxItemIter
aIter2( *pNewStyle
);
932 const SfxPoolItem
* pItem
= aIter2
.GetCurItem();
933 const SfxItemSet
& rWholeParaAttrSet
= rNode
.GetSwAttrSet();
937 const SfxPoolItem
* pTmpItem
= nullptr;
938 if ( SfxItemState::SET
== rWholeParaAttrSet
.GetItemState( pItem
->Which(), false, &pTmpItem
) &&
941 // Do not clear item if the attribute is set in a character format:
942 if ( !pCurrentCharFormat
|| nullptr == CharFormat::GetItem( *pCurrentCharFormat
, pItem
->Which() ) )
945 pNewSet
= pNewStyle
->Clone();
946 pNewSet
->ClearItem( pItem
->Which() );
950 while ((pItem
= aIter2
.NextItem()));
954 bOptimizeAllowed
= false;
955 if ( pNewSet
->Count() )
956 pNewStyle
= rNode
.getIDocumentStyleAccess().getAutomaticStyle( *pNewSet
, IStyleAccess::AUTO_STYLE_CHAR
);
962 // Create new AutoStyle
963 // If there is no current hint and start and end of rNewHint
964 // is ok, we do not need to create a new txtattr.
965 if ( bOptimizeAllowed
&&
966 nPorStart
== nThisStart
&&
967 nPorEnd
== nThisEnd
)
969 pNewAttr
= &rNewHint
;
970 bDestroyHint
= false;
972 else if ( pNewStyle
)
974 pNewAttr
= MakeTextAttr( rNode
.GetDoc(), *pNewStyle
,
975 nPorStart
, nPorEnd
);
983 // if ( bDestroyHint )
984 NoteInHistory( pNewAttr
, true );
987 if ( !bNoLengthAttribute
)
989 nPorStart
= *aStartIter
;
997 rNode
.DestroyAttr( &rNewHint
);
1000 SwTextAttr
* MakeRedlineTextAttr( SwDoc
& rDoc
, SfxPoolItem
const & rAttr
)
1002 // this is intended _only_ for special-purpose redline attributes!
1003 switch (rAttr
.Which())
1005 case RES_CHRATR_COLOR
:
1006 case RES_CHRATR_WEIGHT
:
1007 case RES_CHRATR_CJK_WEIGHT
:
1008 case RES_CHRATR_CTL_WEIGHT
:
1009 case RES_CHRATR_POSTURE
:
1010 case RES_CHRATR_CJK_POSTURE
:
1011 case RES_CHRATR_CTL_POSTURE
:
1012 case RES_CHRATR_UNDERLINE
:
1013 case RES_CHRATR_CROSSEDOUT
:
1014 case RES_CHRATR_CASEMAP
:
1015 case RES_CHRATR_BACKGROUND
:
1018 assert(!"unsupported redline attribute");
1022 // Put new attribute into pool
1023 // FIXME: this const_cast is evil!
1025 const_cast<SfxPoolItem
&>( rDoc
.GetAttrPool().Put( rAttr
) );
1026 return new SwTextAttrEnd( rNew
, 0, 0 );
1029 // create new text attribute
1030 SwTextAttr
* MakeTextAttr(
1033 sal_Int32
const nStt
,
1034 sal_Int32
const nEnd
,
1035 CopyOrNewType
const bIsCopy
,
1036 SwTextNode
*const pTextNode
)
1038 if ( isCHRATR(rAttr
.Which()) )
1040 // Somebody wants to build a SwTextAttr for a character attribute.
1041 // Sorry, this is not allowed any longer.
1042 // You'll get a brand new autostyle attribute:
1043 SfxItemSetFixed
<RES_CHRATR_BEGIN
, RES_CHRATR_END
> aItemSet( rDoc
.GetAttrPool() );
1044 aItemSet
.Put( rAttr
);
1045 return MakeTextAttr( rDoc
, aItemSet
, nStt
, nEnd
);
1047 else if ( RES_TXTATR_AUTOFMT
== rAttr
.Which() &&
1048 static_cast<const SwFormatAutoFormat
&>(rAttr
).GetStyleHandle()->
1049 GetPool() != &rDoc
.GetAttrPool() )
1051 // If the attribute is an auto-style which refers to a pool that is
1052 // different from rDoc's pool, we have to correct this:
1053 const std::shared_ptr
<SfxItemSet
> & pAutoStyle
= static_cast<const SwFormatAutoFormat
&>(rAttr
).GetStyleHandle();
1054 std::unique_ptr
<const SfxItemSet
> pNewSet(
1055 pAutoStyle
->SfxItemSet::Clone( true, &rDoc
.GetAttrPool() ));
1056 SwTextAttr
* pNew
= MakeTextAttr( rDoc
, *pNewSet
, nStt
, nEnd
);
1060 // Put new attribute into pool
1061 // FIXME: this const_cast is evil!
1063 const_cast<SfxPoolItem
&>( rDoc
.GetAttrPool().Put( rAttr
) );
1065 SwTextAttr
* pNew
= nullptr;
1066 switch( rNew
.Which() )
1068 case RES_TXTATR_CHARFMT
:
1070 SwFormatCharFormat
&rFormatCharFormat
= static_cast<SwFormatCharFormat
&>(rNew
);
1071 if( !rFormatCharFormat
.GetCharFormat() )
1073 rFormatCharFormat
.SetCharFormat( rDoc
.GetDfltCharFormat() );
1076 pNew
= new SwTextCharFormat( rFormatCharFormat
, nStt
, nEnd
);
1079 case RES_TXTATR_INETFMT
:
1080 pNew
= new SwTextINetFormat( static_cast<SwFormatINetFormat
&>(rNew
), nStt
, nEnd
);
1083 case RES_TXTATR_FIELD
:
1084 pNew
= new SwTextField( static_cast<SwFormatField
&>(rNew
), nStt
,
1085 rDoc
.IsClipBoard() );
1088 case RES_TXTATR_ANNOTATION
:
1090 pNew
= new SwTextAnnotationField( static_cast<SwFormatField
&>(rNew
), nStt
, rDoc
.IsClipBoard() );
1091 if (bIsCopy
== CopyOrNewType::Copy
)
1093 // On copy of the annotation field do not keep the annotated text range by removing
1094 // the relation to its annotation mark (relation established via annotation field's name).
1095 // If the annotation mark is also copied, the relation and thus the annotated text range will be reestablished,
1096 // when the annotation mark is created and inserted into the document.
1097 auto& pField
= const_cast<SwPostItField
&>(dynamic_cast<const SwPostItField
&>(*(pNew
->GetFormatField().GetField())));
1098 pField
.SetName(OUString());
1099 pField
.SetPostItId();
1104 case RES_TXTATR_INPUTFIELD
:
1105 pNew
= new SwTextInputField( static_cast<SwFormatField
&>(rNew
), nStt
, nEnd
,
1106 rDoc
.IsClipBoard() );
1109 case RES_TXTATR_FLYCNT
:
1111 // finally, copy the frame format (with content)
1112 pNew
= new SwTextFlyCnt( static_cast<SwFormatFlyCnt
&>(rNew
), nStt
);
1113 if ( static_cast<const SwFormatFlyCnt
&>(rAttr
).GetTextFlyCnt() )
1115 // if it has an existing attr then the format must be copied
1116 static_cast<SwTextFlyCnt
*>(pNew
)->CopyFlyFormat( rDoc
);
1120 case RES_TXTATR_FTN
:
1121 pNew
= new SwTextFootnote( static_cast<SwFormatFootnote
&>(rNew
), nStt
);
1122 // copy note's SeqNo
1123 if( static_cast<SwFormatFootnote
&>(rAttr
).GetTextFootnote() )
1124 static_cast<SwTextFootnote
*>(pNew
)->SetSeqNo( static_cast<SwFormatFootnote
&>(rAttr
).GetTextFootnote()->GetSeqRefNo() );
1126 case RES_TXTATR_REFMARK
:
1128 ? new SwTextRefMark( static_cast<SwFormatRefMark
&>(rNew
), nStt
)
1129 : new SwTextRefMark( static_cast<SwFormatRefMark
&>(rNew
), nStt
, &nEnd
);
1131 case RES_TXTATR_TOXMARK
:
1133 SwTOXMark
& rMark
= static_cast<SwTOXMark
&>(rNew
);
1135 // tdf#98868 if the SwTOXType is from a different document that the
1136 // target, re-register the TOXMark against a matching SwTOXType from
1137 // the target document instead
1138 const SwTOXType
* pTOXType
= rMark
.GetTOXType();
1139 if (pTOXType
&& &pTOXType
->GetDoc() != &rDoc
)
1141 SwTOXType
* pToxType
= SwHistorySetTOXMark::GetSwTOXType(rDoc
, pTOXType
->GetType(),
1142 pTOXType
->GetTypeName());
1143 rMark
.RegisterToTOXType(*pToxType
);
1146 pNew
= new SwTextTOXMark(rMark
, nStt
, &nEnd
);
1149 case RES_TXTATR_CJK_RUBY
:
1150 pNew
= new SwTextRuby( static_cast<SwFormatRuby
&>(rNew
), nStt
, nEnd
);
1152 case RES_TXTATR_META
:
1153 case RES_TXTATR_METAFIELD
:
1154 pNew
= SwTextMeta::CreateTextMeta( rDoc
.GetMetaFieldManager(), pTextNode
,
1155 static_cast<SwFormatMeta
&>(rNew
), nStt
, nEnd
, bIsCopy
== CopyOrNewType::Copy
);
1157 case RES_TXTATR_LINEBREAK
:
1158 pNew
= new SwTextLineBreak(static_cast<SwFormatLineBreak
&>(rNew
), nStt
);
1160 case RES_TXTATR_CONTENTCONTROL
:
1161 pNew
= SwTextContentControl::CreateTextContentControl(
1162 rDoc
, pTextNode
, static_cast<SwFormatContentControl
&>(rNew
), nStt
, nEnd
,
1163 bIsCopy
== CopyOrNewType::Copy
);
1166 assert(RES_TXTATR_AUTOFMT
== rNew
.Which());
1167 pNew
= new SwTextAttrEnd( rNew
, nStt
, nEnd
);
1174 SwTextAttr
* MakeTextAttr( SwDoc
& rDoc
, const SfxItemSet
& rSet
,
1175 sal_Int32 nStt
, sal_Int32 nEnd
)
1177 IStyleAccess
& rStyleAccess
= rDoc
.GetIStyleAccess();
1178 const std::shared_ptr
<SfxItemSet
> pAutoStyle
= rStyleAccess
.getAutomaticStyle( rSet
, IStyleAccess::AUTO_STYLE_CHAR
);
1179 SwFormatAutoFormat aNewAutoFormat
;
1180 aNewAutoFormat
.SetStyleHandle( pAutoStyle
);
1181 SwTextAttr
* pNew
= MakeTextAttr( rDoc
, aNewAutoFormat
, nStt
, nEnd
);
1185 // delete the text attribute and unregister its item at the pool
1186 void SwTextNode::DestroyAttr( SwTextAttr
* pAttr
)
1191 // some things need to be done before deleting the formatting attribute
1192 SwDoc
& rDoc
= GetDoc();
1193 switch( pAttr
->Which() )
1195 case RES_TXTATR_FLYCNT
:
1197 SwFrameFormat
* pFormat
= pAttr
->GetFlyCnt().GetFrameFormat();
1198 if( pFormat
) // set to 0 by Undo?
1199 rDoc
.getIDocumentLayoutAccess().DelLayoutFormat( pFormat
);
1203 case RES_CHRATR_HIDDEN
:
1204 SetCalcHiddenCharFlags();
1207 case RES_TXTATR_FTN
:
1208 static_cast<SwTextFootnote
*>(pAttr
)->SetStartNode( nullptr );
1209 static_cast<SwFormatFootnote
&>(pAttr
->GetAttr()).InvalidateFootnote();
1212 case RES_TXTATR_FIELD
:
1213 case RES_TXTATR_ANNOTATION
:
1214 case RES_TXTATR_INPUTFIELD
:
1215 if( !rDoc
.IsInDtor() )
1217 SwTextField
*const pTextField(static_txtattr_cast
<SwTextField
*>(pAttr
));
1218 SwFieldType
* pFieldType
= pAttr
->GetFormatField().GetField()->GetTyp();
1220 if (SwFieldIds::Dde
!= pFieldType
->Which()
1221 && !pTextField
->GetpTextNode())
1223 break; // was not yet inserted
1226 //JP 06-08-95: DDE-fields are an exception
1227 assert(SwFieldIds::Dde
== pFieldType
->Which() ||
1228 this == pTextField
->GetpTextNode());
1230 // certain fields must update the SwDoc's calculation flags
1232 // Certain fields (like HiddenParaField) must trigger recalculation of visible flag
1233 if (GetDoc().FieldCanHideParaWeight(pFieldType
->Which()))
1234 SetCalcHiddenParaField();
1236 switch( pFieldType
->Which() )
1238 case SwFieldIds::HiddenPara
:
1239 case SwFieldIds::DbSetNumber
:
1240 case SwFieldIds::GetExp
:
1241 case SwFieldIds::Database
:
1242 case SwFieldIds::SetExp
:
1243 case SwFieldIds::HiddenText
:
1244 case SwFieldIds::DbNumSet
:
1245 case SwFieldIds::DbNextSet
:
1246 if( !rDoc
.getIDocumentFieldsAccess().IsNewFieldLst() && GetNodes().IsDocNodes() )
1247 rDoc
.getIDocumentFieldsAccess().InsDelFieldInFieldLst(false, *pTextField
);
1249 case SwFieldIds::Dde
:
1250 if (GetNodes().IsDocNodes() && pTextField
->GetpTextNode())
1251 static_cast<SwDDEFieldType
*>(pFieldType
)->DecRefCnt();
1253 case SwFieldIds::Postit
:
1255 const_cast<SwFormatField
&>(pAttr
->GetFormatField()).Broadcast(
1256 SwFormatFieldHint(&pTextField
->GetFormatField(), SwFormatFieldHintWhich::REMOVED
));
1262 static_cast<SwFormatField
&>(pAttr
->GetAttr()).InvalidateField();
1265 case RES_TXTATR_TOXMARK
:
1266 static_cast<SwTOXMark
&>(pAttr
->GetAttr()).InvalidateTOXMark();
1269 case RES_TXTATR_REFMARK
:
1270 static_cast<SwFormatRefMark
&>(pAttr
->GetAttr()).InvalidateRefMark();
1273 case RES_TXTATR_META
:
1274 case RES_TXTATR_METAFIELD
:
1276 auto pTextMeta
= static_txtattr_cast
<SwTextMeta
*>(pAttr
);
1277 SwFormatMeta
& rFormatMeta( static_cast<SwFormatMeta
&>(pTextMeta
->GetAttr()) );
1278 if (::sw::Meta
* pMeta
= rFormatMeta
.GetMeta())
1280 if (SwDocShell
* pDocSh
= rDoc
.GetDocShell())
1282 static constexpr OUStringLiteral
metaNS(u
"urn:bails");
1283 const css::uno::Reference
<css::rdf::XResource
> xSubject
= pMeta
->MakeUnoObject();
1284 uno::Reference
<frame::XModel
> xModel
= pDocSh
->GetBaseModel();
1285 SwRDFHelper::clearStatements(xModel
, metaNS
, xSubject
);
1289 static_txtattr_cast
<SwTextMeta
*>(pAttr
)->ChgTextNode(nullptr);
1292 case RES_TXTATR_CONTENTCONTROL
:
1294 static_txtattr_cast
<SwTextContentControl
*>(pAttr
)->ChgTextNode(nullptr);
1302 SwTextAttr::Destroy( pAttr
, rDoc
.GetAttrPool() );
1305 SwTextAttr
* SwTextNode::InsertItem(
1307 const sal_Int32 nStart
,
1308 const sal_Int32 nEnd
,
1309 const SetAttrMode nMode
)
1311 // character attributes will be inserted as automatic styles:
1312 assert( !isCHRATR(rAttr
.Which()) && "AUTOSTYLES - "
1313 "SwTextNode::InsertItem should not be called with character attributes");
1315 SwTextAttr
*const pNew
=
1321 (nMode
& SetAttrMode::IS_COPY
) ? CopyOrNewType::Copy
: CopyOrNewType::New
,
1326 const bool bSuccess( InsertHint( pNew
, nMode
) );
1327 // N.B.: also check that the hint is actually in the hints array,
1328 // because hints of certain types may be merged after successful
1329 // insertion, and thus destroyed!
1330 if (!bSuccess
|| !m_pSwpHints
->Contains( pNew
))
1339 // take ownership of pAttr; if insertion fails, delete pAttr
1340 bool SwTextNode::InsertHint( SwTextAttr
* const pAttr
, const SetAttrMode nMode
)
1342 bool bHiddenPara
= false;
1344 assert(pAttr
&& pAttr
->GetStart() <= Len());
1345 assert(!pAttr
->GetEnd() || (*pAttr
->GetEnd() <= Len()));
1347 // translate from SetAttrMode to InsertMode (for hints with CH_TXTATR)
1348 const SwInsertFlags nInsertFlags
=
1349 (nMode
& SetAttrMode::NOHINTEXPAND
)
1350 ? SwInsertFlags::NOHINTEXPAND
1351 : (nMode
& SetAttrMode::FORCEHINTEXPAND
)
1352 ? (SwInsertFlags::FORCEHINTEXPAND
| SwInsertFlags::EMPTYEXPAND
)
1353 : SwInsertFlags::EMPTYEXPAND
;
1355 // need this after TryInsertHint, when pAttr may be deleted
1356 const sal_Int32
nStart( pAttr
->GetStart() );
1357 const bool bDummyChar( pAttr
->HasDummyChar() );
1360 SetAttrMode nInsMode
= nMode
;
1361 switch( pAttr
->Which() )
1363 case RES_TXTATR_FLYCNT
:
1365 SwTextFlyCnt
*pFly
= static_cast<SwTextFlyCnt
*>(pAttr
);
1366 SwFrameFormat
* pFormat
= pAttr
->GetFlyCnt().GetFrameFormat();
1367 if( !(SetAttrMode::NOTXTATRCHR
& nInsMode
) )
1369 // Need to insert char first, because SetAnchor() reads
1371 //JP 11.05.98: if the anchor is already set correctly,
1372 // fix it after inserting the char, so that clients don't
1373 // have to worry about it.
1374 const SwFormatAnchor
* pAnchor
= pFormat
->GetItemIfSet( RES_ANCHOR
, false );
1376 SwContentIndex
aIdx( this, pAttr
->GetStart() );
1377 const OUString
c(GetCharOfTextAttr(*pAttr
));
1378 OUString
const ins( InsertText(c
, aIdx
, nInsertFlags
) );
1381 // do not record deletion of Format!
1382 ::sw::UndoGuard
const ug(
1383 pFormat
->GetDoc()->GetIDocumentUndoRedo());
1385 return false; // text node full :(
1387 nInsMode
|= SetAttrMode::NOTXTATRCHR
;
1390 (RndStdIds::FLY_AS_CHAR
== pAnchor
->GetAnchorId()) &&
1391 pAnchor
->GetAnchorNode() &&
1392 *pAnchor
->GetAnchorNode() == *this &&
1393 pAnchor
->GetAnchorContentOffset() == aIdx
.GetIndex() )
1395 const_cast<SwPosition
*>(pAnchor
->GetContentAnchor())->AdjustContent(-1);
1398 pFly
->SetAnchor( this );
1400 // format pointer could have changed in SetAnchor,
1401 // when copying to other docs!
1402 pFormat
= pAttr
->GetFlyCnt().GetFrameFormat();
1403 SwDoc
*pDoc
= pFormat
->GetDoc();
1405 // OD 26.06.2003 - allow drawing objects in header/footer.
1406 // But don't allow control objects in header/footer
1407 if( RES_DRAWFRMFMT
== pFormat
->Which() &&
1408 pDoc
->IsInHeaderFooter( *pFormat
->GetAnchor().GetAnchorNode() ) )
1410 bool bCheckControlLayer
= false;
1411 pFormat
->CallSwClientNotify(sw::CheckDrawFrameFormatLayerHint(&bCheckControlLayer
));
1412 if( bCheckControlLayer
)
1414 // This should not be allowed, prevent it here.
1415 // The dtor of the SwTextAttr does not delete the
1416 // char, so delete it explicitly here.
1417 if( SetAttrMode::NOTXTATRCHR
& nInsMode
)
1419 // delete the char from the string
1420 assert(CH_TXTATR_BREAKWORD
== m_Text
[pAttr
->GetStart()]
1421 || CH_TXTATR_INWORD
== m_Text
[pAttr
->GetStart()]);
1422 m_Text
= m_Text
.replaceAt(pAttr
->GetStart(), 1, u
"");
1424 SwContentIndex
aTmpIdx( this, pAttr
->GetStart() );
1425 Update(aTmpIdx
, 1, UpdateMode::Negative
);
1427 // do not record deletion of Format!
1428 ::sw::UndoGuard
const ug(pDoc
->GetIDocumentUndoRedo());
1429 DestroyAttr( pAttr
);
1436 case RES_TXTATR_FTN
:
1438 // Footnotes: create text node and put it into Inserts-section
1439 SwDoc
& rDoc
= GetDoc();
1440 SwNodes
&rNodes
= rDoc
.GetNodes();
1442 // check that footnote is inserted into body or redline section
1443 if( StartOfSectionIndex() < rNodes
.GetEndOfAutotext().GetIndex() )
1445 // This should not be allowed, prevent it here.
1446 // The dtor of the SwTextAttr does not delete the
1447 // char, so delete it explicitly here.
1448 if( SetAttrMode::NOTXTATRCHR
& nInsMode
)
1450 // delete the char from the string
1451 assert(CH_TXTATR_BREAKWORD
== m_Text
[pAttr
->GetStart()]
1452 || CH_TXTATR_INWORD
== m_Text
[pAttr
->GetStart()]);
1453 m_Text
= m_Text
.replaceAt(pAttr
->GetStart(), 1, u
"");
1455 SwContentIndex
aTmpIdx( this, pAttr
->GetStart() );
1456 Update(aTmpIdx
, 1, UpdateMode::Negative
);
1458 DestroyAttr( pAttr
);
1462 // is a new footnote being inserted?
1463 bool bNewFootnote
= nullptr == static_cast<SwTextFootnote
*>(pAttr
)->GetStartNode();
1466 static_cast<SwTextFootnote
*>(pAttr
)->MakeNewTextSection( GetNodes() );
1467 SwRegHistory
* pHist
= GetpSwpHints()
1468 ? GetpSwpHints()->GetHistory() : nullptr;
1470 pHist
->ChangeNodeIndex( GetIndex() );
1472 else if ( !GetpSwpHints() || !GetpSwpHints()->IsInSplitNode() )
1474 // existing footnote: delete all layout frames of its
1476 SwNodeOffset nSttIdx
=
1477 static_cast<SwTextFootnote
*>(pAttr
)->GetStartNode()->GetIndex();
1478 SwNodeOffset nEndIdx
= rNodes
[ nSttIdx
++ ]->EndOfSectionIndex();
1479 for( ; nSttIdx
< nEndIdx
; ++nSttIdx
)
1481 SwContentNode
* pCNd
= rNodes
[ nSttIdx
]->GetContentNode();
1482 if( nullptr != pCNd
)
1483 pCNd
->DelFrames(nullptr);
1484 else if (SwTableNode
*const pTable
= rNodes
[nSttIdx
]->GetTableNode())
1486 pTable
->DelFrames();
1491 if( !(SetAttrMode::NOTXTATRCHR
& nInsMode
) )
1493 // must insert first, to prevent identical indexes
1494 // that could later prevent insertion into SwDoc's
1496 SwContentIndex
aNdIdx( this, pAttr
->GetStart() );
1497 const OUString
c(GetCharOfTextAttr(*pAttr
));
1498 OUString
const ins( InsertText(c
, aNdIdx
, nInsertFlags
) );
1502 return false; // text node full :(
1504 nInsMode
|= SetAttrMode::NOTXTATRCHR
;
1507 // insert into SwDoc's footnote index array
1508 SwTextFootnote
* pTextFootnote
= nullptr;
1511 // moving an existing footnote (e.g. SplitNode)
1512 for( size_t n
= 0; n
< rDoc
.GetFootnoteIdxs().size(); ++n
)
1513 if( pAttr
== rDoc
.GetFootnoteIdxs()[n
] )
1515 // assign new index by removing and re-inserting
1516 pTextFootnote
= rDoc
.GetFootnoteIdxs()[n
];
1517 rDoc
.GetFootnoteIdxs().erase( rDoc
.GetFootnoteIdxs().begin() + n
);
1520 // if the Undo set the StartNode, the Index isn't
1521 // in the doc's array yet!
1523 if( !pTextFootnote
)
1524 pTextFootnote
= static_cast<SwTextFootnote
*>(pAttr
);
1526 // to update the numbers and for sorting, the Node must be set
1527 static_cast<SwTextFootnote
*>(pAttr
)->ChgTextNode( this );
1529 // do not insert footnote in redline section into footnote array
1530 if( StartOfSectionIndex() > rNodes
.GetEndOfRedlines().GetIndex() )
1532 const bool bSuccess
= rDoc
.GetFootnoteIdxs().insert(pTextFootnote
).second
;
1533 OSL_ENSURE( bSuccess
, "FootnoteIdx not inserted." );
1535 rDoc
.GetFootnoteIdxs().UpdateFootnote( *this );
1536 static_cast<SwTextFootnote
*>(pAttr
)->SetSeqRefNo();
1540 case RES_TXTATR_FIELD
:
1542 // trigger notification for relevant fields, like HiddenParaFields
1543 if (GetDoc().FieldCanHideParaWeight(
1544 pAttr
->GetFormatField().GetField()->GetTyp()->Which()))
1550 case RES_TXTATR_LINEBREAK
:
1552 static_cast<SwTextLineBreak
*>(pAttr
)->SetTextNode(this);
1557 // CH_TXTATR_* are inserted for SwTextHints without EndIndex
1558 // If the caller is SwTextNode::Copy, the char has already been copied,
1559 // and SETATTR_NOTXTATRCHR prevents inserting it again here.
1560 if( !(SetAttrMode::NOTXTATRCHR
& nInsMode
) )
1562 SwContentIndex
aIdx( this, pAttr
->GetStart() );
1563 OUString
const ins( InsertText(OUString(GetCharOfTextAttr(*pAttr
)),
1564 aIdx
, nInsertFlags
) );
1568 return false; // text node full :(
1571 // adjust end of hint to account for inserted CH_TXTATR
1572 const sal_Int32
* pEnd(pAttr
->GetEnd());
1575 pAttr
->SetEnd(*pEnd
+ 1);
1578 if (pAttr
->Which() == RES_TXTATR_CONTENTCONTROL
)
1580 // Content controls have a dummy character at their end as well.
1581 SwContentIndex
aEndIdx(this, *pAttr
->GetEnd());
1583 = InsertText(OUString(GetCharOfTextAttr(*pAttr
)), aEndIdx
, nInsertFlags
);
1590 pEnd
= pAttr
->GetEnd();
1593 pAttr
->SetEnd(*pEnd
+ 1);
1599 // handle attributes which provide content
1600 sal_Int32 nEnd
= nStart
;
1601 bool bInputFieldStartCharInserted
= false;
1602 bool bInputFieldEndCharInserted
= false;
1603 const bool bHasContent( pAttr
->HasContent() );
1606 switch( pAttr
->Which() )
1608 case RES_TXTATR_INPUTFIELD
:
1610 SwTextInputField
* pTextInputField
= dynamic_cast<SwTextInputField
*>(pAttr
);
1611 if ( pTextInputField
)
1613 if( !(SetAttrMode::NOTXTATRCHR
& nMode
) )
1615 SwContentIndex
aIdx( this, pAttr
->GetStart() );
1616 const OUString aContent
= OUStringChar(CH_TXT_ATR_INPUTFIELDSTART
)
1617 + pTextInputField
->GetFieldContent() + OUStringChar(CH_TXT_ATR_INPUTFIELDEND
);
1618 InsertText( aContent
, aIdx
, nInsertFlags
);
1620 const sal_Int32
* const pEnd(pAttr
->GetEnd());
1621 assert(pEnd
!= nullptr);
1622 pAttr
->SetEnd(*pEnd
+ aContent
.getLength());
1623 nEnd
= *pAttr
->GetEnd();
1627 // assure that CH_TXT_ATR_INPUTFIELDSTART and CH_TXT_ATR_INPUTFIELDEND are inserted.
1628 if ( m_Text
[ pAttr
->GetStart() ] != CH_TXT_ATR_INPUTFIELDSTART
)
1630 SwContentIndex
aIdx( this, pAttr
->GetStart() );
1631 InsertText( OUString(CH_TXT_ATR_INPUTFIELDSTART
), aIdx
, nInsertFlags
);
1632 bInputFieldStartCharInserted
= true;
1633 const sal_Int32
* const pEnd(pAttr
->GetEnd());
1634 assert(pEnd
!= nullptr);
1635 pAttr
->SetEnd(*pEnd
+ 1);
1636 nEnd
= *pAttr
->GetEnd();
1639 const sal_Int32
* const pEnd(pAttr
->GetEnd());
1640 assert(pEnd
!= nullptr);
1641 if (m_Text
[ *pEnd
- 1 ] != CH_TXT_ATR_INPUTFIELDEND
)
1643 SwContentIndex
aIdx( this, *pEnd
);
1644 InsertText( OUString(CH_TXT_ATR_INPUTFIELDEND
), aIdx
, nInsertFlags
);
1645 bInputFieldEndCharInserted
= true;
1646 pAttr
->SetEnd(*pEnd
+ 1);
1647 nEnd
= *pAttr
->GetEnd();
1658 GetOrCreateSwpHints();
1660 // handle overlap with an existing InputField
1661 bool bInsertHint
= true;
1663 const SwTextInputField
* pTextInputField
= GetOverlappingInputField( *pAttr
);
1664 if ( pTextInputField
!= nullptr )
1666 if ( pAttr
->End() == nullptr )
1668 bInsertHint
= false;
1673 if ( pAttr
->GetStart() > pTextInputField
->GetStart() )
1675 pAttr
->SetStart( pTextInputField
->GetStart() );
1677 if ( *(pAttr
->End()) < *(pTextInputField
->End()) )
1679 pAttr
->SetEnd(*(pTextInputField
->End()));
1687 // Handle the invariant that a plain text content control has the same character formatting
1688 // for all of its content.
1689 auto* pTextContentControl
= static_txtattr_cast
<SwTextContentControl
*>(
1690 GetTextAttrAt(pAttr
->GetStart(), RES_TXTATR_CONTENTCONTROL
, ::sw::GetTextAttrMode::Parent
));
1691 if (pTextContentControl
)
1693 auto& rFormatContentControl
1694 = static_cast<SwFormatContentControl
&>(pTextContentControl
->GetAttr());
1695 std::shared_ptr
<SwContentControl
> pContentControl
1696 = rFormatContentControl
.GetContentControl();
1697 if (pAttr
->End() != nullptr && pContentControl
->GetPlainText())
1699 if (pAttr
->GetStart() > pTextContentControl
->GetStart())
1701 pAttr
->SetStart(pTextContentControl
->GetStart());
1703 if (*pAttr
->End() < *pTextContentControl
->End())
1705 pAttr
->SetEnd(*pTextContentControl
->End());
1711 const bool bRet
= bInsertHint
1712 && m_pSwpHints
->TryInsertHint( pAttr
, *this, nMode
);
1717 && !(SetAttrMode::NOTXTATRCHR
& nMode
) )
1719 // undo insertion of dummy character
1720 // N.B. cannot insert the dummy character after inserting the hint,
1721 // because if the hint has no extent it will be moved in InsertText,
1722 // resulting in infinite recursion
1723 assert((CH_TXTATR_BREAKWORD
== m_Text
[nStart
] ||
1724 CH_TXTATR_INWORD
== m_Text
[nStart
] ));
1725 SwContentIndex
aIdx( this, nStart
);
1726 EraseText( aIdx
, 1 );
1731 if ( !(SetAttrMode::NOTXTATRCHR
& nMode
)
1732 && (nEnd
- nStart
) > 0 )
1734 SwContentIndex
aIdx( this, nStart
);
1735 EraseText( aIdx
, (nEnd
- nStart
) );
1739 if ( bInputFieldEndCharInserted
1740 && (nEnd
- nStart
) > 0 )
1742 SwContentIndex
aIdx( this, nEnd
- 1 );
1743 EraseText( aIdx
, 1 );
1746 if ( bInputFieldStartCharInserted
)
1748 SwContentIndex
aIdx( this, nStart
);
1749 EraseText( aIdx
, 1 );
1757 SetCalcHiddenParaField();
1763 void SwTextNode::DeleteAttribute( SwTextAttr
* const pAttr
)
1767 OSL_FAIL("DeleteAttribute called, but text node without hints?");
1771 if ( pAttr
->HasDummyChar() )
1774 const SwContentIndex
aIdx( this, pAttr
->GetStart() );
1775 // erase the CH_TXTATR, which will also delete pAttr
1776 EraseText( aIdx
, 1 );
1778 else if ( pAttr
->HasContent() )
1780 const SwContentIndex
aIdx( this, pAttr
->GetStart() );
1781 assert(pAttr
->End() != nullptr);
1782 EraseText( aIdx
, *pAttr
->End() - pAttr
->GetStart() );
1786 // create MsgHint before start/end become invalid
1792 m_pSwpHints
->Delete( pAttr
);
1793 SwTextAttr::Destroy( pAttr
, GetDoc().GetAttrPool() );
1794 CallSwClientNotify(sw::LegacyModifyHint(nullptr, &aHint
));
1796 TryDeleteSwpHints();
1800 //FIXME: this does NOT respect SORT NUMBER (for CHARFMT)!
1801 void SwTextNode::DeleteAttributes(
1802 const sal_uInt16 nWhich
,
1803 const sal_Int32 nStart
,
1804 const sal_Int32 nEnd
)
1809 for ( size_t nPos
= 0; m_pSwpHints
&& nPos
< m_pSwpHints
->Count(); ++nPos
)
1811 SwTextAttr
* const pTextHt
= m_pSwpHints
->Get( nPos
);
1812 const sal_Int32 nHintStart
= pTextHt
->GetStart();
1813 if (nStart
< nHintStart
)
1815 break; // sorted by start
1817 else if ( (nStart
== nHintStart
) && (nWhich
== pTextHt
->Which()) )
1819 if ( nWhich
== RES_CHRATR_HIDDEN
)
1821 assert(!"hey, that's a CHRATR! how did that get in?");
1822 SetCalcHiddenCharFlags();
1824 else if ( nWhich
== RES_TXTATR_CHARFMT
)
1826 // Check if character format contains hidden attribute:
1827 const SwCharFormat
* pFormat
= pTextHt
->GetCharFormat().GetCharFormat();
1828 if ( SfxItemState::SET
== pFormat
->GetItemState( RES_CHRATR_HIDDEN
) )
1829 SetCalcHiddenCharFlags();
1831 // #i75430# Recalc hidden flags if necessary
1832 else if ( nWhich
== RES_TXTATR_AUTOFMT
)
1834 // Check if auto style contains hidden attribute:
1835 const SfxPoolItem
* pHiddenItem
= CharFormat::GetItem( *pTextHt
, RES_CHRATR_HIDDEN
);
1837 SetCalcHiddenCharFlags();
1838 // for auto styles DeleteAttributes is only called from Undo
1839 // so it shouldn't need to care about ignore start/end flags
1842 sal_Int32
const * const pEndIdx
= pTextHt
->GetEnd();
1844 if ( pTextHt
->HasDummyChar() )
1847 const SwContentIndex
aIdx( this, nStart
);
1848 // erase the CH_TXTATR, which will also delete pTextHt
1849 EraseText( aIdx
, 1 );
1851 else if ( pTextHt
->HasContent() )
1853 const SwContentIndex
aIdx( this, nStart
);
1854 OSL_ENSURE( pTextHt
->End() != nullptr, "<SwTextNode::DeleteAttributes(..)> - missing End() at <SwTextAttr> instance which has content" );
1855 EraseText( aIdx
, *pTextHt
->End() - nStart
);
1857 else if( *pEndIdx
== nEnd
)
1859 // Create MsgHint before Start and End are gone.
1860 // For HiddenParaFields it's not necessary to call
1861 // SetCalcHiddenParaField because the dtor does that.
1867 m_pSwpHints
->DeleteAtPos( nPos
);
1868 SwTextAttr::Destroy( pTextHt
, GetDoc().GetAttrPool() );
1869 CallSwClientNotify(sw::LegacyModifyHint(nullptr, &aHint
));
1873 TryDeleteSwpHints();
1876 void SwTextNode::DelSoftHyph( const sal_Int32 nStt
, const sal_Int32 nEnd
)
1878 sal_Int32 nFndPos
= nStt
;
1879 sal_Int32 nEndPos
= nEnd
;
1882 nFndPos
= m_Text
.indexOf(CHAR_SOFTHYPHEN
, nFndPos
);
1883 if (nFndPos
<0 || nFndPos
>=nEndPos
)
1887 const SwContentIndex
aIdx( this, nFndPos
);
1888 EraseText( aIdx
, 1 );
1893 bool SwTextNode::IsIgnoredCharFormatForNumbering(const sal_uInt16 nWhich
, bool bIsCharStyle
)
1895 // LO can save the char background as either shading or highlight, so check which mode is currently chosen.
1896 // Shading does not affect the numbering. Highlighting does (but isn't allowed in a char style).
1897 if (nWhich
== RES_CHRATR_BACKGROUND
)
1898 return bIsCharStyle
|| SvtFilterOptions::Get().IsCharBackground2Shading();
1900 return (nWhich
== RES_CHRATR_UNDERLINE
1901 || nWhich
== RES_CHRATR_ESCAPEMENT
);
1904 // Set these attributes on SwTextNode. If they apply to the entire paragraph
1905 // text, set them in the SwTextNode's item set (SwContentNode::SetAttr).
1906 bool SwTextNode::SetAttr(
1907 const SfxItemSet
& rSet
,
1908 const sal_Int32 nStt
,
1909 const sal_Int32 nEnd
,
1910 const SetAttrMode nMode
,
1911 SwTextAttr
**ppNewTextAttr
)
1916 // split sets (for selection in nodes)
1917 const SfxItemSet
* pSet
= &rSet
;
1918 SfxItemSetFixed
<RES_TXTATR_BEGIN
, RES_TXTATR_END
-1> aTextSet( *rSet
.GetPool() );
1921 if ( !nStt
&& (nEnd
== m_Text
.getLength()) &&
1922 !(nMode
& SetAttrMode::NOFORMATATTR
) )
1924 // if the node already has CharFormat hints, the new attributes must
1925 // be set as hints too to override those.
1926 bool bHasCharFormats
= false;
1929 for ( size_t n
= 0; n
< m_pSwpHints
->Count(); ++n
)
1931 if ( m_pSwpHints
->Get( n
)->IsCharFormatAttr() )
1933 bHasCharFormats
= true;
1939 if( !bHasCharFormats
)
1941 aTextSet
.Put( rSet
);
1942 // If there are any character attributes in rSet,
1943 // we want to set them at the paragraph:
1944 if( aTextSet
.Count() != rSet
.Count() )
1946 const bool bRet
= SetAttr( rSet
);
1947 if( !aTextSet
.Count() )
1951 // check for auto style:
1952 if ( const SwFormatAutoFormat
* pItem
= aTextSet
.GetItemIfSet( RES_TXTATR_AUTOFMT
, false ) )
1954 const bool bRet
= SetAttr( *pItem
->GetStyleHandle() );
1955 if( 1 == aTextSet
.Count() )
1959 // Continue with the text attributes:
1964 GetOrCreateSwpHints();
1966 SfxItemSet
aCharSet( *rSet
.GetPool(), aCharAutoFormatSetRange
);
1969 SfxItemIter
aIter( *pSet
);
1970 const SfxPoolItem
* pItem
= aIter
.GetCurItem();
1974 if (!IsInvalidItem(pItem
))
1976 const sal_uInt16 nWhich
= pItem
->Which();
1977 OSL_ENSURE( isCHRATR(nWhich
) || isTXTATR(nWhich
),
1978 "SwTextNode::SetAttr(): unknown attribute" );
1979 if ( isCHRATR(nWhich
) || isTXTATR(nWhich
) )
1981 if ((RES_TXTATR_CHARFMT
== nWhich
) &&
1982 (GetDoc().GetDfltCharFormat() ==
1983 static_cast<const SwFormatCharFormat
*>(pItem
)->GetCharFormat()))
1985 RstTextAttr( nStt
, nEnd
- nStt
, RES_TXTATR_CHARFMT
);
1986 DontExpandFormat( nStt
);
1990 if (isCHRATR(nWhich
) ||
1991 (RES_TXTATR_UNKNOWN_CONTAINER
== nWhich
))
1993 aCharSet
.Put( *pItem
);
1998 SwTextAttr
*const pNew
= MakeTextAttr( GetDoc(),
1999 const_cast<SfxPoolItem
&>(*pItem
), nStt
, nEnd
);
2002 // store the first one we create into the pp
2003 if (ppNewTextAttr
&& !*ppNewTextAttr
)
2004 *ppNewTextAttr
= pNew
;
2005 if ( nEnd
!= nStt
&& !pNew
->GetEnd() )
2007 OSL_FAIL("Attribute without end, but area marked");
2008 DestroyAttr( pNew
); // do not insert
2010 else if ( InsertHint( pNew
, nMode
) )
2019 pItem
= aIter
.NextItem();
2022 if ( aCharSet
.Count() )
2024 SwTextAttr
* pTmpNew
= MakeTextAttr( GetDoc(), aCharSet
, nStt
, nEnd
);
2025 if ( InsertHint( pTmpNew
, nMode
) )
2031 TryDeleteSwpHints();
2036 static void lcl_MergeAttr( SfxItemSet
& rSet
, const SfxPoolItem
& rAttr
)
2038 if ( RES_TXTATR_AUTOFMT
== rAttr
.Which() )
2040 const SfxItemSet
* pCFSet
= CharFormat::GetItemSet( rAttr
);
2043 SfxWhichIter
aIter( *pCFSet
);
2044 sal_uInt16 nWhich
= aIter
.FirstWhich();
2047 const SfxPoolItem
* pItem
= nullptr;
2048 if( ( nWhich
< RES_CHRATR_END
||
2049 RES_TXTATR_UNKNOWN_CONTAINER
== nWhich
) &&
2050 ( SfxItemState::SET
== aIter
.GetItemState( true, &pItem
) ) )
2052 nWhich
= aIter
.NextWhich();
2059 static void lcl_MergeAttr_ExpandChrFormat( SfxItemSet
& rSet
, const SfxPoolItem
& rAttr
)
2061 if( RES_TXTATR_CHARFMT
== rAttr
.Which() ||
2062 RES_TXTATR_INETFMT
== rAttr
.Which() ||
2063 RES_TXTATR_AUTOFMT
== rAttr
.Which() )
2065 const SfxItemSet
* pCFSet
= CharFormat::GetItemSet( rAttr
);
2069 SfxWhichIter
aIter( *pCFSet
);
2070 sal_uInt16 nWhich
= aIter
.FirstWhich();
2073 const SfxPoolItem
* pItem
= nullptr;
2074 if( ( nWhich
< RES_CHRATR_END
||
2075 ( RES_TXTATR_AUTOFMT
== rAttr
.Which() && RES_TXTATR_UNKNOWN_CONTAINER
== nWhich
) ) &&
2076 ( SfxItemState::SET
== aIter
.GetItemState( true, &pItem
) ) )
2078 nWhich
= aIter
.NextWhich();
2083 /* If multiple attributes overlap, the last one wins!
2084 Probably this can only happen between a RES_TXTATR_INETFMT and one of the
2085 other hints, because BuildPortions ensures that CHARFMT/AUTOFMT don't
2086 overlap. But there may be multiple CHARFMT/AUTOFMT with exactly the same
2087 start/end, sorted by BuildPortions, in which case the same logic applies.
2090 |------------| Font1
2093 |--| query range: -> Font2
2101 struct SwPoolItemEndPair
2104 const SfxPoolItem
* mpItem
;
2107 SwPoolItemEndPair() : mpItem( nullptr ), mnEndPos( 0 ) {};
2112 static void lcl_MergeListLevelIndentAsLRSpaceItem( const SwTextNode
& rTextNode
,
2115 ::sw::ListLevelIndents
const indents(rTextNode
.AreListLevelIndentsApplicable());
2116 if (indents
== ::sw::ListLevelIndents::No
)
2119 const SwNumRule
* pRule
= rTextNode
.GetNumRule();
2120 if ( pRule
&& rTextNode
.GetActualListLevel() >= 0 )
2122 const SwNumFormat
& rFormat
= pRule
->Get(o3tl::narrowing
<sal_uInt16
>(rTextNode
.GetActualListLevel()));
2123 if ( rFormat
.GetPositionAndSpaceMode() == SvxNumberFormat::LABEL_ALIGNMENT
)
2125 if (indents
& ::sw::ListLevelIndents::FirstLine
)
2127 SvxFirstLineIndentItem
const firstLine(static_cast<short>(rFormat
.GetFirstLineIndent()), RES_MARGIN_FIRSTLINE
);
2128 rSet
.Put(firstLine
);
2130 if (indents
& ::sw::ListLevelIndents::LeftMargin
)
2132 SvxTextLeftMarginItem
const leftMargin(rFormat
.GetIndentAt(), RES_MARGIN_TEXTLEFT
);
2133 rSet
.Put(leftMargin
);
2139 // request the attributes of the TextNode at the range
2140 bool SwTextNode::GetParaAttr(SfxItemSet
& rSet
, sal_Int32 nStt
, sal_Int32 nEnd
,
2141 const bool bOnlyTextAttr
, const bool bGetFromChrFormat
,
2142 const bool bMergeIndentValuesOfNumRule
,
2143 SwRootFrame
const*const pLayout
) const
2145 assert(!rSet
.Count()); // handled inconsistently, typically an error?
2147 if (pLayout
&& pLayout
->HasMergedParas())
2149 if (GetRedlineMergeFlag() == SwNode::Merge::Hidden
)
2151 return false; // ignore deleted node
2155 // get the node's automatic attributes
2156 SfxItemSet
aFormatSet( *rSet
.GetPool(), rSet
.GetRanges() );
2159 SwTextNode
const& rParaPropsNode(
2160 sw::GetAttrMerged(aFormatSet
, *this, pLayout
));
2161 if (bMergeIndentValuesOfNumRule
)
2163 lcl_MergeListLevelIndentAsLRSpaceItem(rParaPropsNode
, aFormatSet
);
2169 // First, check which text attributes are valid in the range.
2172 // * the attribute is wholly contained in the range
2173 // * the attribute end is in the range
2174 // * the attribute start is in the range
2175 // Unambiguous (merge into set), if
2176 // * the attribute wholly contains the range
2178 // * the attribute is wholly outside the range
2180 void (*fnMergeAttr
)( SfxItemSet
&, const SfxPoolItem
& )
2181 = bGetFromChrFormat
? &lcl_MergeAttr_ExpandChrFormat
2184 const size_t nSize
= m_pSwpHints
->Count();
2186 if (nStt
== nEnd
) // no range:
2188 for (size_t n
= 0; n
< nSize
; ++n
)
2190 const SwTextAttr
* pHt
= m_pSwpHints
->Get(n
);
2191 const sal_Int32 nAttrStart
= pHt
->GetStart();
2192 if (nAttrStart
> nEnd
) // behind the range
2195 const sal_Int32
* pAttrEnd
= pHt
->End();
2196 if ( ! pAttrEnd
) // no attributes without end
2199 if( ( nAttrStart
< nStt
&&
2200 ( pHt
->DontExpand() ? nStt
< *pAttrEnd
2201 : nStt
<= *pAttrEnd
)) ||
2202 ( nStt
== nAttrStart
&&
2203 ( nAttrStart
== *pAttrEnd
|| !nStt
)))
2204 (*fnMergeAttr
)( rSet
, pHt
->GetAttr() );
2207 else // a query range is defined
2210 std::optional
< std::vector
< SwPoolItemEndPair
> > pAttrArr
;
2212 const size_t coArrSz
= RES_TXTATR_WITHEND_END
- RES_CHRATR_BEGIN
;
2214 for (size_t n
= 0; n
< nSize
; ++n
)
2216 const SwTextAttr
* pHt
= m_pSwpHints
->Get(n
);
2217 const sal_Int32 nAttrStart
= pHt
->GetStart();
2218 if (nAttrStart
> nEnd
) // outside, behind
2221 const sal_Int32
* pAttrEnd
= pHt
->End();
2222 if ( ! pAttrEnd
) // no attributes without end
2225 bool bChkInvalid
= false;
2226 if (nAttrStart
<= nStt
) // before or exactly Start
2228 if (*pAttrEnd
<= nStt
) // outside, before
2231 if (nEnd
<= *pAttrEnd
) // behind or exactly End
2232 (*fnMergeAttr
)( aFormatSet
, pHt
->GetAttr() );
2234 // else if( pHt->GetAttr() != aFormatSet.Get( pHt->Which() ) )
2238 else if (nAttrStart
< nEnd
// starts in the range
2239 )// && pHt->GetAttr() != aFormatSet.Get( pHt->Which() ) )
2245 std::optional
< SfxItemIter
> oItemIter
;
2246 const SfxPoolItem
* pItem
= nullptr;
2248 if ( RES_TXTATR_AUTOFMT
== pHt
->Which() )
2250 const SfxItemSet
* pAutoSet
= CharFormat::GetItemSet( pHt
->GetAttr() );
2253 oItemIter
.emplace( *pAutoSet
);
2254 pItem
= oItemIter
->GetCurItem();
2258 pItem
= &pHt
->GetAttr();
2260 const sal_Int32 nHintEnd
= *pAttrEnd
;
2262 for (; pItem
; pItem
= oItemIter
? oItemIter
->NextItem() : nullptr)
2264 const sal_uInt16 nHintWhich
= pItem
->Which();
2265 OSL_ENSURE(!isUNKNOWNATR(nHintWhich
),
2266 "SwTextNode::GetAttr(): unknown attribute?");
2270 pAttrArr
= std::vector
< SwPoolItemEndPair
>(coArrSz
);
2273 std::vector
< SwPoolItemEndPair
>::iterator pPrev
= pAttrArr
->begin();
2274 if (isCHRATR(nHintWhich
) ||
2275 isTXTATR_WITHEND(nHintWhich
))
2277 pPrev
+= nHintWhich
- RES_CHRATR_BEGIN
;
2281 pPrev
= pAttrArr
->end();
2284 if( pPrev
!= pAttrArr
->end() )
2286 if( !pPrev
->mpItem
)
2288 if ( bOnlyTextAttr
|| *pItem
!= aFormatSet
.Get( nHintWhich
) )
2290 if( nAttrStart
> nStt
)
2292 rSet
.InvalidateItem( nHintWhich
);
2293 pPrev
->mpItem
= INVALID_POOL_ITEM
;
2297 pPrev
->mpItem
= pItem
;
2298 pPrev
->mnEndPos
= nHintEnd
;
2302 else if( !IsInvalidItem(pPrev
->mpItem
) )
2304 if( pPrev
->mnEndPos
== nAttrStart
&&
2305 *pPrev
->mpItem
== *pItem
)
2307 pPrev
->mpItem
= pItem
;
2308 pPrev
->mnEndPos
= nHintEnd
;
2312 rSet
.InvalidateItem( nHintWhich
);
2313 pPrev
->mpItem
= INVALID_POOL_ITEM
;
2323 for (size_t n
= 0; n
< coArrSz
; ++n
)
2325 const SwPoolItemEndPair
& rItemPair
= (*pAttrArr
)[ n
];
2326 if( rItemPair
.mpItem
&& !IsInvalidItem(rItemPair
.mpItem
) )
2328 const sal_uInt16 nWh
=
2329 o3tl::narrowing
<sal_uInt16
>(n
+ RES_CHRATR_BEGIN
);
2331 if (nEnd
<= rItemPair
.mnEndPos
) // behind or exactly end
2333 if( *rItemPair
.mpItem
!= aFormatSet
.Get( nWh
) )
2334 (*fnMergeAttr
)( rSet
, *rItemPair
.mpItem
);
2338 rSet
.InvalidateItem( nWh
);
2343 if( aFormatSet
.Count() )
2345 // remove all from the format-set that are also set in the text-set
2346 aFormatSet
.Differentiate( rSet
);
2350 if (aFormatSet
.Count())
2352 // now "merge" everything
2353 rSet
.Put( aFormatSet
);
2356 return rSet
.Count() != 0;
2362 typedef std::pair
<sal_Int32
, sal_Int32
> AttrSpan_t
;
2363 typedef std::multimap
<AttrSpan_t
, const SwTextAttr
*> AttrSpanMap_t
;
2368 operator()(const AttrSpanMap_t::value_type
& i_rAttrSpan
)
2371 return i_rAttrSpan
.second
&& i_rAttrSpan
.second
->Which() == RES_TXTATR_AUTOFMT
;
2375 /** Removes from io_rAttrSet all items that are set by style on the
2378 struct RemovePresentAttrs
2380 explicit RemovePresentAttrs(SfxItemSet
& io_rAttrSet
)
2381 : m_rAttrSet(io_rAttrSet
)
2386 operator()(const AttrSpanMap_t::value_type
& i_rAttrSpan
)
2389 if (!i_rAttrSpan
.second
)
2394 const SwTextAttr
* const pAutoStyle(i_rAttrSpan
.second
);
2395 SfxItemIter
aIter(m_rAttrSet
);
2396 for (const SfxPoolItem
* pItem(aIter
.GetCurItem()); pItem
; pItem
= aIter
.NextItem())
2398 const sal_uInt16
nWhich(pItem
->Which());
2399 if (CharFormat::IsItemIncluded(nWhich
, pAutoStyle
))
2407 SfxItemSet
& m_rAttrSet
;
2410 /** Collects all style-covered spans from i_rHints to o_rSpanMap. In
2411 addition inserts dummy spans with pointer to format equal to 0 for
2412 all gaps (i.e. spans not covered by any style). This simplifies
2413 creation of autostyles for all needed spans, but it means all code
2414 that tries to access the pointer has to check if it's non-null!
2417 lcl_CollectHintSpans(const SwpHints
& i_rHints
, const sal_Int32 nLength
,
2418 AttrSpanMap_t
& o_rSpanMap
)
2420 sal_Int32
nLastEnd(0);
2422 for (size_t i
= 0; i
< i_rHints
.Count(); ++i
)
2424 const SwTextAttr
* pHint
= i_rHints
.Get(i
);
2425 const sal_uInt16
nWhich(pHint
->Which());
2426 if (nWhich
== RES_TXTATR_CHARFMT
|| nWhich
== RES_TXTATR_AUTOFMT
)
2428 const AttrSpan_t
aSpan(pHint
->GetStart(), *pHint
->End());
2429 o_rSpanMap
.emplace(aSpan
, pHint
);
2431 // < not != because there may be multiple CHARFMT at same range
2432 if (nLastEnd
< aSpan
.first
)
2434 // insert dummy span covering the gap
2435 o_rSpanMap
.emplace( AttrSpan_t(nLastEnd
, aSpan
.first
), nullptr );
2438 nLastEnd
= aSpan
.second
;
2442 // no hints at the end (special case: no hints at all in i_rHints)
2443 if (nLastEnd
!= nLength
&& nLength
!= 0)
2445 o_rSpanMap
.emplace(AttrSpan_t(nLastEnd
, nLength
), nullptr);
2450 lcl_FillWhichIds(const SfxItemSet
& i_rAttrSet
, std::vector
<sal_uInt16
>& o_rClearIds
)
2452 o_rClearIds
.reserve(i_rAttrSet
.Count());
2453 SfxItemIter
aIter(i_rAttrSet
);
2454 for (const SfxPoolItem
* pItem
= aIter
.GetCurItem(); pItem
; pItem
= aIter
.NextItem())
2456 o_rClearIds
.push_back(pItem
->Which());
2460 struct SfxItemSetClearer
2462 SfxItemSet
& m_rItemSet
;
2463 explicit SfxItemSetClearer(SfxItemSet
& rItemSet
) : m_rItemSet(rItemSet
) { }
2464 void operator()(sal_uInt16
const nWhich
) { m_rItemSet
.ClearItem(nWhich
); }
2469 /** Does the hard work of SwTextNode::FormatToTextAttr: the real conversion
2470 of items to automatic styles.
2473 SwTextNode::impl_FormatToTextAttr(const SfxItemSet
& i_rAttrSet
)
2475 typedef AttrSpanMap_t::iterator AttrSpanMap_iterator_t
;
2476 AttrSpanMap_t aAttrSpanMap
;
2478 if (i_rAttrSet
.Count() == 0)
2483 // 1. Identify all spans in hints' array
2485 lcl_CollectHintSpans(*m_pSwpHints
, m_Text
.getLength(), aAttrSpanMap
);
2487 // 2. Go through all spans and insert new attrs
2489 AttrSpanMap_iterator_t
aCurRange(aAttrSpanMap
.begin());
2490 const AttrSpanMap_iterator_t
aEnd(aAttrSpanMap
.end());
2491 while (aCurRange
!= aEnd
)
2493 typedef std::pair
<AttrSpanMap_iterator_t
, AttrSpanMap_iterator_t
>
2495 AttrSpanMapRange_t
aRange(aAttrSpanMap
.equal_range(aCurRange
->first
));
2497 // 2a. Collect attributes to insert
2499 SfxItemSet
aCurSet(i_rAttrSet
);
2500 std::for_each(aRange
.first
, aRange
.second
, RemovePresentAttrs(aCurSet
));
2502 // 2b. Insert automatic style containing the collected attributes
2504 if (aCurSet
.Count() != 0)
2506 AttrSpanMap_iterator_t
aAutoStyleIt(
2507 std::find_if(aRange
.first
, aRange
.second
, IsAutoStyle()));
2508 if (aAutoStyleIt
!= aRange
.second
)
2510 // there already is an automatic style on that span:
2511 // create new one and remove the original one
2512 SwTextAttr
* const pAutoStyle(const_cast<SwTextAttr
*>(aAutoStyleIt
->second
));
2513 const std::shared_ptr
<SfxItemSet
> pOldStyle(
2514 static_cast<const SwFormatAutoFormat
&>(
2515 pAutoStyle
->GetAttr()).GetStyleHandle());
2516 aCurSet
.Put(*pOldStyle
);
2518 // remove the old hint
2519 m_pSwpHints
->Delete(pAutoStyle
);
2520 DestroyAttr(pAutoStyle
);
2522 m_pSwpHints
->Insert(
2523 MakeTextAttr(GetDoc(), aCurSet
,
2524 aCurRange
->first
.first
, aCurRange
->first
.second
));
2527 aCurRange
= aRange
.second
;
2530 // hints were directly inserted, so need to fix the Ignore flags now
2531 m_pSwpHints
->MergePortions(*this);
2533 // 3. Clear items from the node
2534 std::vector
<sal_uInt16
> aClearedIds
;
2535 lcl_FillWhichIds(i_rAttrSet
, aClearedIds
);
2536 ClearItemsFromAttrSet(aClearedIds
);
2539 void SwTextNode::FormatToTextAttr( SwTextNode
* pNd
)
2541 SfxItemSet
aThisSet( GetDoc().GetAttrPool(), aCharFormatSetRange
);
2542 if( HasSwAttrSet() && GetpSwAttrSet()->Count() )
2543 aThisSet
.Put( *GetpSwAttrSet() );
2545 GetOrCreateSwpHints();
2549 impl_FormatToTextAttr(aThisSet
);
2553 // There are five possible combinations of items from this and
2554 // pNd (pNd is the 'main' node):
2556 // case pNd this action
2559 // 2 - a convert item to attr of this
2560 // 3 a - convert item to attr of pNd
2561 // 4 a a clear item in this
2562 // 5 a b convert item to attr of this
2564 SfxItemSet
aNdSet( pNd
->GetDoc().GetAttrPool(), aCharFormatSetRange
);
2565 if( pNd
->HasSwAttrSet() && pNd
->GetpSwAttrSet()->Count() )
2566 aNdSet
.Put( *pNd
->GetpSwAttrSet() );
2568 pNd
->GetOrCreateSwpHints();
2570 std::vector
<sal_uInt16
> aProcessedIds
;
2572 if( aThisSet
.Count() )
2574 SfxItemIter
aIter( aThisSet
);
2575 const SfxPoolItem
* pItem
= aIter
.GetCurItem(), *pNdItem
= nullptr;
2576 SfxItemSet
aConvertSet( GetDoc().GetAttrPool(), aCharFormatSetRange
);
2577 std::vector
<sal_uInt16
> aClearWhichIds
;
2581 if( SfxItemState::SET
== aNdSet
.GetItemState( pItem
->Which(), false, &pNdItem
) )
2583 if (*pItem
== *pNdItem
) // 4
2585 aClearWhichIds
.push_back( pItem
->Which() );
2589 aConvertSet
.Put(*pItem
);
2591 aProcessedIds
.push_back(pItem
->Which());
2595 aConvertSet
.Put(*pItem
);
2598 pItem
= aIter
.NextItem();
2601 // 4/ clear items of this that are set with the same value on pNd
2602 ClearItemsFromAttrSet( aClearWhichIds
);
2604 // 2, 5/ convert all other items to attrs
2605 impl_FormatToTextAttr(aConvertSet
);
2609 std::for_each(aProcessedIds
.begin(), aProcessedIds
.end(),
2610 SfxItemSetClearer(aNdSet
));
2612 // 3/ convert items to attrs
2613 pNd
->impl_FormatToTextAttr(aNdSet
);
2615 if( aNdSet
.Count() )
2617 SwFormatChg
aTmp1( pNd
->GetFormatColl() );
2618 pNd
->CallSwClientNotify(sw::LegacyModifyHint(&aTmp1
, &aTmp1
));
2623 SetCalcHiddenCharFlags();
2625 pNd
->TryDeleteSwpHints();
2628 void SwpHints::CalcFlags()
2630 m_bDDEFields
= m_bFootnote
= false;
2631 const size_t nSize
= Count();
2632 for( size_t nPos
= 0; nPos
< nSize
; ++nPos
)
2634 const SwTextAttr
* pAttr
= Get( nPos
);
2635 switch( pAttr
->Which() )
2637 case RES_TXTATR_FTN
:
2642 case RES_TXTATR_FIELD
:
2644 const SwField
* pField
= pAttr
->GetFormatField().GetField();
2645 if( SwFieldIds::Dde
== pField
->GetTyp()->Which() )
2647 m_bDDEFields
= true;
2657 bool SwpHints::CalcHiddenParaField() const
2659 m_bCalcHiddenParaField
= false;
2660 const bool bOldHiddenByParaField
= m_bHiddenByParaField
;
2661 bool bNewHiddenByParaField
= false;
2662 int nNewResultWeight
= 0;
2663 const size_t nSize
= Count();
2664 const SwTextAttr
* pTextHt
;
2666 for (size_t nPos
= 0; nPos
< nSize
; ++nPos
)
2668 pTextHt
= Get(nPos
);
2669 const sal_uInt16 nWhich
= pTextHt
->Which();
2671 if (RES_TXTATR_FIELD
== nWhich
)
2673 // see also SwTextFrame::IsHiddenNow()
2674 const SwFormatField
& rField
= pTextHt
->GetFormatField();
2675 int nCurWeight
= m_rParent
.GetDoc().FieldCanHideParaWeight(rField
.GetField()->GetTyp()->Which());
2676 if (nCurWeight
> nNewResultWeight
)
2678 nNewResultWeight
= nCurWeight
;
2679 bNewHiddenByParaField
= m_rParent
.GetDoc().FieldHidesPara(*rField
.GetField());
2681 else if (nCurWeight
== nNewResultWeight
&& bNewHiddenByParaField
)
2683 // Currently, for both supported hiding types (HiddenPara, Database), "Don't hide"
2684 // takes precedence - i.e., if there's a "Don't hide" field of that weight, we only
2685 // care about fields of higher weight.
2686 bNewHiddenByParaField
= m_rParent
.GetDoc().FieldHidesPara(*rField
.GetField());
2690 SetHiddenByParaField(bNewHiddenByParaField
);
2691 return bOldHiddenByParaField
!= bNewHiddenByParaField
;
2694 void SwpHints::NoteInHistory( SwTextAttr
*pAttr
, const bool bNew
)
2696 if ( m_pHistory
) { m_pHistory
->AddHint( pAttr
, bNew
); }
2701 SwTextAttr
* pTextAttr
;
2703 bool isRsidOnlyAutoFormat
;
2705 typedef std::vector
< Portion
> PortionMap
;
2706 enum MergeResult
{ MATCH
, DIFFER_ONLY_RSID
, DIFFER
};
2709 static MergeResult
lcl_Compare_Attributes(
2711 const std::pair
< PortionMap::const_iterator
, PortionMap::const_iterator
>& aRange1
,
2712 const std::pair
< PortionMap::const_iterator
, PortionMap::const_iterator
>& aRange2
,
2713 std::vector
<bool>& RsidOnlyAutoFormatFlagMap
);
2715 bool SwpHints::MergePortions( SwTextNode
& rNode
)
2720 // sort before merging
2724 PortionMap aPortionMap
;
2725 aPortionMap
.reserve(Count() + 1);
2726 std::vector
<bool> RsidOnlyAutoFormatFlagMap
;
2727 RsidOnlyAutoFormatFlagMap
.resize(Count() + 1);
2728 sal_Int32 nLastPorStart
= COMPLETE_STRING
;
2731 // get portions by start position:
2732 for ( size_t i
= 0; i
< Count(); ++i
)
2734 SwTextAttr
*pHt
= Get( i
);
2735 if ( RES_TXTATR_CHARFMT
!= pHt
->Which() &&
2736 RES_TXTATR_AUTOFMT
!= pHt
->Which() )
2738 //RES_TXTATR_INETFMT != pHt->Which() )
2741 bool isRsidOnlyAutoFormat(false);
2742 // check for RSID-only AUTOFMT
2743 if (RES_TXTATR_AUTOFMT
== pHt
->Which())
2745 std::shared_ptr
<SfxItemSet
> const & pSet(
2746 pHt
->GetAutoFormat().GetStyleHandle());
2747 if ((pSet
->Count() == 1) && pSet
->GetItem(RES_CHRATR_RSID
, false))
2749 // fdo#70201: eliminate no-extent RSID-only AUTOFMT
2750 // could be produced by ReplaceText or (maybe?) RstAttr
2751 if (pHt
->GetStart() == *pHt
->GetEnd())
2753 DeleteAtPos(i
); // kill it without History!
2754 SwTextAttr::Destroy(pHt
, rNode
.GetDoc().GetAttrPool());
2758 // fdo#52028: this one has _only_ RSID => ignore it completely
2759 if (!pHt
->IsFormatIgnoreStart() || !pHt
->IsFormatIgnoreEnd())
2762 pHt
->SetFormatIgnoreStart(true);
2763 pHt
->SetFormatIgnoreEnd (true);
2764 NoteInHistory(pHt
, true);
2766 isRsidOnlyAutoFormat
= true;
2770 if (pHt
->GetStart() == *pHt
->GetEnd())
2772 // no-length hints are a disease. ignore them here.
2773 // the SwAttrIter::SeekFwd will not call Rst/Chg for them.
2777 const sal_Int32 nPorStart
= pHt
->GetStart();
2778 if (nPorStart
!= nLastPorStart
)
2780 nLastPorStart
= nPorStart
;
2781 aPortionMap
.push_back(Portion
{pHt
, nKey
, isRsidOnlyAutoFormat
});
2782 RsidOnlyAutoFormatFlagMap
[nKey
] = isRsidOnlyAutoFormat
;
2785 // we add data strictly in-order, so we can forward-search the vector
2786 auto equal_range
= [](PortionMap::const_iterator startIt
, PortionMap::const_iterator endIt
, int i
)
2789 while (it1
!= endIt
&& it1
->nKey
< i
)
2792 while (it2
!= endIt
&& it2
->nKey
== i
)
2794 return std::pair
<PortionMap::const_iterator
, PortionMap::const_iterator
>{ it1
, it2
};
2797 // check if portion i can be merged with portion i+1:
2798 // note: need to include i=0 to set IgnoreStart and j=nKey+1 to reset
2799 // IgnoreEnd at first / last portion
2802 // Store this outside the loop, because we limit the search area on subsequent searches.
2803 std::pair
< PortionMap::const_iterator
, PortionMap::const_iterator
> aRange1
{ aPortionMap
.begin(), aPortionMap
.begin() + aPortionMap
.size() };
2806 aRange1
= equal_range( aRange1
.first
, aPortionMap
.begin() + aPortionMap
.size(), i
);
2807 // start the search for this one from where the first search ended.
2808 std::pair
< PortionMap::const_iterator
, PortionMap::const_iterator
> aRange2
2809 = equal_range( aRange1
.second
, aPortionMap
.begin() + aPortionMap
.size(), j
);
2811 MergeResult eMerge
= lcl_Compare_Attributes(i
, j
, aRange1
, aRange2
, RsidOnlyAutoFormatFlagMap
);
2813 if (MATCH
== eMerge
)
2815 // important: delete second range so any IgnoreStart on the first
2816 // range is still valid
2817 // erase all elements with key i + 1
2818 sal_Int32 nNewPortionEnd
= 0;
2819 for ( auto aIter2
= aRange2
.first
; aIter2
!= aRange2
.second
; ++aIter2
)
2821 SwTextAttr
*const p2
= aIter2
->pTextAttr
;
2822 nNewPortionEnd
= *p2
->GetEnd();
2824 const size_t nCountBeforeDelete
= Count();
2827 // robust: check if deletion actually took place before destroying attribute:
2828 if ( Count() < nCountBeforeDelete
)
2829 rNode
.DestroyAttr( p2
);
2831 aPortionMap
.erase( aRange2
.first
, aRange2
.second
);
2834 // change all attributes with key i
2835 aRange1
= equal_range( aRange1
.first
, aPortionMap
.begin() + aPortionMap
.size(), i
);
2836 for ( auto aIter1
= aRange1
.first
; aIter1
!= aRange1
.second
; ++aIter1
)
2838 SwTextAttr
*const p1
= aIter1
->pTextAttr
;
2839 NoteInHistory( p1
);
2840 p1
->SetEnd(nNewPortionEnd
);
2841 NoteInHistory( p1
, true );
2852 // when not merging the ignore flags need to be either set or reset
2853 // (reset too in case one of the autofmts was recently changed)
2854 bool const bSetIgnoreFlag(DIFFER_ONLY_RSID
== eMerge
);
2855 for (auto aIter1
= aRange1
.first
; aIter1
!= aRange1
.second
; ++aIter1
)
2857 if (!aIter1
->isRsidOnlyAutoFormat
) // already set above, don't change
2859 SwTextAttr
*const pCurrent(aIter1
->pTextAttr
);
2860 if (pCurrent
->IsFormatIgnoreEnd() != bSetIgnoreFlag
)
2862 NoteInHistory(pCurrent
);
2863 pCurrent
->SetFormatIgnoreEnd(bSetIgnoreFlag
);
2864 NoteInHistory(pCurrent
, true);
2868 for (auto aIter2
= aRange2
.first
; aIter2
!= aRange2
.second
; ++aIter2
)
2870 if (!aIter2
->isRsidOnlyAutoFormat
) // already set above, don't change
2872 SwTextAttr
*const pCurrent(aIter2
->pTextAttr
);
2873 if (pCurrent
->IsFormatIgnoreStart() != bSetIgnoreFlag
)
2875 NoteInHistory(pCurrent
);
2876 pCurrent
->SetFormatIgnoreStart(bSetIgnoreFlag
);
2877 NoteInHistory(pCurrent
, true);
2881 i
= j
; // ++i not enough: i + 1 may have been deleted (MATCH)!
2890 static MergeResult
lcl_Compare_Attributes(
2892 const std::pair
< PortionMap::const_iterator
, PortionMap::const_iterator
>& aRange1
,
2893 const std::pair
< PortionMap::const_iterator
, PortionMap::const_iterator
>& aRange2
,
2894 std::vector
<bool>& RsidOnlyAutoFormatFlagMap
)
2896 PortionMap::const_iterator aIter1
= aRange1
.first
;
2897 PortionMap::const_iterator aIter2
= aRange2
.first
;
2899 size_t const nAttributesInPor1
= std::distance(aRange1
.first
, aRange1
.second
);
2900 size_t const nAttributesInPor2
= std::distance(aRange2
.first
, aRange2
.second
);
2901 bool const isRsidOnlyAutoFormat1
= i
< sal_Int32(RsidOnlyAutoFormatFlagMap
.size()) && RsidOnlyAutoFormatFlagMap
[i
];
2902 bool const isRsidOnlyAutoFormat2
= j
< sal_Int32(RsidOnlyAutoFormatFlagMap
.size()) && RsidOnlyAutoFormatFlagMap
[j
];
2904 // if both have one they could be equal, but not if only one has it
2905 bool const bSkipRsidOnlyAutoFormat(nAttributesInPor1
!= nAttributesInPor2
);
2907 // this loop needs to handle the case where one has a CHARFMT and the
2908 // other CHARFMT + RSID-only AUTOFMT, so...
2909 // want to skip over RSID-only AUTOFMT here, hence the -1
2910 if (!((nAttributesInPor1
- (isRsidOnlyAutoFormat1
? 1 : 0)) ==
2911 (nAttributesInPor2
- (isRsidOnlyAutoFormat2
? 1 : 0))
2912 && (nAttributesInPor1
!= 0 || nAttributesInPor2
!= 0)))
2917 MergeResult
eMerge(MATCH
);
2919 // _if_ there is one element more either in aRange1 or aRange2
2920 // it _must_ be an RSID-only AUTOFMT, which can be ignored here...
2921 // But if both have RSID-only AUTOFMT they could be equal, no skip!
2922 while (aIter1
!= aRange1
.second
|| aIter2
!= aRange2
.second
)
2924 // first of all test if there's no gap (before skipping stuff!)
2925 if (aIter1
!= aRange1
.second
&& aIter2
!= aRange2
.second
&&
2926 *aIter1
->pTextAttr
->End() < aIter2
->pTextAttr
->GetStart())
2930 // skip it - cannot be equal if bSkipRsidOnlyAutoFormat is set
2931 if (bSkipRsidOnlyAutoFormat
2932 && aIter1
!= aRange1
.second
&& aIter1
->isRsidOnlyAutoFormat
)
2934 assert(DIFFER
!= eMerge
);
2935 eMerge
= DIFFER_ONLY_RSID
;
2939 if (bSkipRsidOnlyAutoFormat
2940 && aIter2
!= aRange2
.second
&& aIter2
->isRsidOnlyAutoFormat
)
2942 assert(DIFFER
!= eMerge
);
2943 eMerge
= DIFFER_ONLY_RSID
;
2947 assert(aIter1
!= aRange1
.second
&& aIter2
!= aRange2
.second
);
2948 SwTextAttr
const*const p1
= aIter1
->pTextAttr
;
2949 SwTextAttr
const*const p2
= aIter2
->pTextAttr
;
2950 if (p1
->Which() != p2
->Which())
2956 // fdo#52028: for auto styles, check if they differ only
2957 // in the RSID, which should have no effect on text layout
2958 if (RES_TXTATR_AUTOFMT
!= p1
->Which())
2961 const SfxItemSet
& rSet1
= *p1
->GetAutoFormat().GetStyleHandle();
2962 const SfxItemSet
& rSet2
= *p2
->GetAutoFormat().GetStyleHandle();
2964 // sadly SfxItemSet::operator== does not seem to work?
2965 SfxItemIter
iter1(rSet1
);
2966 SfxItemIter
iter2(rSet2
);
2967 for (SfxPoolItem
const* pItem1
= iter1
.GetCurItem(),
2968 * pItem2
= iter2
.GetCurItem();;)
2970 if (pItem1
&& pItem1
->Which() == RES_CHRATR_RSID
)
2971 pItem1
= iter1
.NextItem();
2972 if (pItem2
&& pItem2
->Which() == RES_CHRATR_RSID
)
2973 pItem2
= iter2
.NextItem();
2974 if (!pItem1
&& !pItem2
)
2976 eMerge
= DIFFER_ONLY_RSID
;
2979 if (!pItem1
|| !pItem2
)
2983 if (pItem1
!= pItem2
) // all are poolable
2985 assert(IsInvalidItem(pItem1
) || IsInvalidItem(pItem2
) || pItem1
->Which() != pItem2
->Which() || *pItem1
!= *pItem2
);
2988 pItem1
= iter1
.NextItem();
2989 pItem2
= iter2
.NextItem();
2999 // check if there is already a character format and adjust the sort numbers
3000 static void lcl_CheckSortNumber( const SwpHints
& rHints
, SwTextCharFormat
& rNewCharFormat
)
3002 const sal_Int32 nHtStart
= rNewCharFormat
.GetStart();
3003 const sal_Int32 nHtEnd
= *rNewCharFormat
.GetEnd();
3004 sal_uInt16 nSortNumber
= 0;
3006 for ( size_t i
= 0; i
< rHints
.Count(); ++i
)
3008 const SwTextAttr
* pOtherHt
= rHints
.Get(i
);
3010 const sal_Int32 nOtherStart
= pOtherHt
->GetStart();
3012 if ( nOtherStart
> nHtStart
)
3015 if ( RES_TXTATR_CHARFMT
== pOtherHt
->Which() )
3017 const sal_Int32 nOtherEnd
= *pOtherHt
->End();
3019 if ( nOtherStart
== nHtStart
&& nOtherEnd
== nHtEnd
)
3021 nSortNumber
= static_txtattr_cast
<const SwTextCharFormat
*>(pOtherHt
)->GetSortNumber() + 1;
3026 if ( nSortNumber
> 0 )
3027 rNewCharFormat
.SetSortNumber( nSortNumber
);
3031 * Try to insert the new hint.
3032 * Depending on the type of the hint, this either always succeeds, or may fail.
3033 * Depending on the type of the hint, other hints may be deleted or
3035 * The return value indicates successful insertion.
3037 bool SwpHints::TryInsertHint(
3038 SwTextAttr
* const pHint
,
3040 const SetAttrMode nMode
)
3042 if ( MAX_HINTS
<= Count() ) // we're sorry, this flight is overbooked...
3044 OSL_FAIL("hints array full :-(");
3045 rNode
.DestroyAttr(pHint
);
3049 const sal_Int32
*pHtEnd
= pHint
->GetEnd();
3050 const sal_uInt16 nWhich
= pHint
->Which();
3051 std::vector
<sal_uInt16
> aWhichSublist
;
3055 case RES_TXTATR_CHARFMT
:
3057 // Check if character format contains hidden attribute:
3058 const SwCharFormat
* pFormat
= pHint
->GetCharFormat().GetCharFormat();
3059 if ( SfxItemState::SET
== pFormat
->GetItemState( RES_CHRATR_HIDDEN
) )
3060 rNode
.SetCalcHiddenCharFlags();
3062 static_txtattr_cast
<SwTextCharFormat
*>(pHint
)->ChgTextNode( &rNode
);
3065 // #i75430# Recalc hidden flags if necessary
3066 case RES_TXTATR_AUTOFMT
:
3068 std::shared_ptr
<SfxItemSet
> const & pSet( pHint
->GetAutoFormat().GetStyleHandle() );
3069 if (pHint
->GetStart() == *pHint
->GetEnd())
3071 if (pSet
->Count() == 1 && pSet
->GetItem(RES_CHRATR_RSID
, false))
3072 { // empty range RSID-only hints could cause trouble, there's no
3073 rNode
.DestroyAttr(pHint
); // need for them so don't insert
3077 // Check if auto style contains hidden attribute:
3078 const SfxPoolItem
* pHiddenItem
= CharFormat::GetItem( *pHint
, RES_CHRATR_HIDDEN
);
3080 rNode
.SetCalcHiddenCharFlags();
3082 // fdo#71556: populate aWhichFormatAttr member of SwMsgPoolItem
3083 const WhichRangesContainer
& pRanges
= pSet
->GetRanges();
3084 for(auto const & rPair
: pRanges
)
3086 const sal_uInt16 nBeg
= rPair
.first
;
3087 const sal_uInt16 nEnd
= rPair
.second
;
3088 for( sal_uInt16 nSubElem
= nBeg
; nSubElem
<= nEnd
; ++nSubElem
)
3089 if( pSet
->HasItem( nSubElem
) )
3090 aWhichSublist
.push_back( nSubElem
);
3094 case RES_TXTATR_INETFMT
:
3095 static_txtattr_cast
<SwTextINetFormat
*>(pHint
)->InitINetFormat(rNode
);
3098 case RES_TXTATR_FIELD
:
3099 case RES_TXTATR_ANNOTATION
:
3100 case RES_TXTATR_INPUTFIELD
:
3102 SwTextField
*const pTextField(static_txtattr_cast
<SwTextField
*>(pHint
));
3103 bool bDelFirst
= nullptr != pTextField
->GetpTextNode();
3104 pTextField
->ChgTextNode( &rNode
);
3105 SwDoc
& rDoc
= rNode
.GetDoc();
3106 const SwField
* pField
= pTextField
->GetFormatField().GetField();
3108 if( !rDoc
.getIDocumentFieldsAccess().IsNewFieldLst() )
3110 // certain fields must update the SwDoc's calculation flags
3111 switch( pField
->GetTyp()->Which() )
3113 case SwFieldIds::Database
:
3114 case SwFieldIds::SetExp
:
3115 case SwFieldIds::HiddenPara
:
3116 case SwFieldIds::HiddenText
:
3117 case SwFieldIds::DbNumSet
:
3118 case SwFieldIds::DbNextSet
:
3121 rDoc
.getIDocumentFieldsAccess().InsDelFieldInFieldLst(false, *pTextField
);
3122 if( rNode
.GetNodes().IsDocNodes() )
3123 rDoc
.getIDocumentFieldsAccess().InsDelFieldInFieldLst(true, *pTextField
);
3126 case SwFieldIds::Dde
:
3127 if( rNode
.GetNodes().IsDocNodes() )
3128 static_cast<SwDDEFieldType
*>(pField
->GetTyp())->IncRefCnt();
3134 // insert into real document's nodes-array?
3135 if( rNode
.GetNodes().IsDocNodes() )
3137 bool bInsFieldType
= false;
3138 switch( pField
->GetTyp()->Which() )
3140 case SwFieldIds::SetExp
:
3141 bInsFieldType
= static_cast<SwSetExpFieldType
*>(pField
->GetTyp())->IsDeleted();
3142 if( nsSwGetSetExpType::GSE_SEQ
& static_cast<SwSetExpFieldType
*>(pField
->GetTyp())->GetType() )
3144 // register the field at its FieldType before setting
3145 // the sequence reference number!
3146 SwSetExpFieldType
* pFieldType
= static_cast<SwSetExpFieldType
*>(
3147 rDoc
.getIDocumentFieldsAccess().InsertFieldType( *pField
->GetTyp() ) );
3148 if( pFieldType
!= pField
->GetTyp() )
3150 SwFormatField
* pFormatField
= const_cast<SwFormatField
*>(&pTextField
->GetFormatField());
3151 pFormatField
->RegisterToFieldType( *pFieldType
);
3152 pFormatField
->GetField()->ChgTyp( pFieldType
);
3154 pFieldType
->SetSeqRefNo( *const_cast<SwSetExpField
*>(static_cast<const SwSetExpField
*>(pField
)) );
3157 case SwFieldIds::User
:
3158 bInsFieldType
= static_cast<SwUserFieldType
*>(pField
->GetTyp())->IsDeleted();
3161 case SwFieldIds::Dde
:
3162 if( rDoc
.getIDocumentFieldsAccess().IsNewFieldLst() )
3163 static_cast<SwDDEFieldType
*>(pField
->GetTyp())->IncRefCnt();
3164 bInsFieldType
= static_cast<SwDDEFieldType
*>(pField
->GetTyp())->IsDeleted();
3167 case SwFieldIds::Postit
:
3168 if ( rDoc
.GetDocShell() )
3170 rDoc
.GetDocShell()->Broadcast( SwFormatFieldHint(
3171 &pTextField
->GetFormatField(), SwFormatFieldHintWhich::INSERTED
));
3177 rDoc
.getIDocumentFieldsAccess().InsDeletedFieldType( *pField
->GetTyp() );
3181 case RES_TXTATR_FTN
:
3182 static_cast<SwTextFootnote
*>(pHint
)->ChgTextNode( &rNode
);
3184 case RES_TXTATR_REFMARK
:
3185 static_txtattr_cast
<SwTextRefMark
*>(pHint
)->ChgTextNode( &rNode
);
3186 if( rNode
.GetNodes().IsDocNodes() )
3188 // search for a reference with the same name
3190 for( size_t n
= 0, nEnd
= Count(); n
< nEnd
; ++n
)
3192 const sal_Int32
*pTmpHtEnd
;
3193 const sal_Int32
*pTmpHintEnd
;
3194 if (RES_TXTATR_REFMARK
== (pTmpHt
= Get(n
))->Which() &&
3195 pHint
->GetAttr() == pTmpHt
->GetAttr() &&
3196 nullptr != ( pTmpHtEnd
= pTmpHt
->GetEnd() ) &&
3197 nullptr != ( pTmpHintEnd
= pHint
->GetEnd() ) )
3199 SwComparePosition eCmp
= ::ComparePosition(
3200 pTmpHt
->GetStart(), *pTmpHtEnd
,
3201 pHint
->GetStart(), *pTmpHintEnd
);
3202 bool bDelOld
= true, bChgStart
= false, bChgEnd
= false;
3205 case SwComparePosition::Before
:
3206 case SwComparePosition::Behind
: bDelOld
= false; break;
3208 case SwComparePosition::Outside
: bChgStart
= bChgEnd
= true; break;
3210 case SwComparePosition::CollideEnd
:
3211 case SwComparePosition::OverlapBefore
: bChgStart
= true; break;
3212 case SwComparePosition::CollideStart
:
3213 case SwComparePosition::OverlapBehind
: bChgEnd
= true; break;
3219 pHint
->SetStart( pTmpHt
->GetStart() );
3222 pHint
->SetEnd(*pTmpHtEnd
);
3226 NoteInHistory( pTmpHt
);
3227 rNode
.DestroyAttr( Cut( n
-- ) );
3234 case RES_TXTATR_TOXMARK
:
3235 static_txtattr_cast
<SwTextTOXMark
*>(pHint
)->ChgTextNode( &rNode
);
3238 case RES_TXTATR_CJK_RUBY
:
3239 static_txtattr_cast
<SwTextRuby
*>(pHint
)->InitRuby(rNode
);
3242 case RES_TXTATR_META
:
3243 case RES_TXTATR_METAFIELD
:
3244 static_txtattr_cast
<SwTextMeta
*>(pHint
)->ChgTextNode( &rNode
);
3247 case RES_TXTATR_CONTENTCONTROL
:
3248 static_txtattr_cast
<SwTextContentControl
*>(pHint
)->ChgTextNode( &rNode
);
3251 case RES_CHRATR_HIDDEN
:
3252 rNode
.SetCalcHiddenCharFlags();
3256 if( SetAttrMode::DONTEXPAND
& nMode
)
3257 pHint
->SetDontExpand( true );
3259 // special handling for SwTextAttrs without end:
3260 // 1) they cannot overlap
3261 // 2) if two fields are adjacent, they must not be merged into one
3262 // this is guaranteed by inserting a CH_TXTATR_* into the paragraph text!
3263 sal_Int32 nHtStart
= pHint
->GetStart();
3267 NoteInHistory(pHint
, true);
3270 if( !rNode
.GetDoc().IsInReading() )
3273 // ... and notify listeners
3274 if(rNode
.HasWriterListeners())
3281 rNode
.TriggerNodeUpdate(sw::LegacyModifyHint(&aHint
, &aHint
));
3287 // from here on, pHint is known to have an end index!
3289 if( *pHtEnd
< nHtStart
)
3291 assert(*pHtEnd
>= nHtStart
);
3293 // just swap the nonsense:
3294 pHint
->SetStart(*pHtEnd
);
3295 pHint
->SetEnd(nHtStart
);
3296 nHtStart
= pHint
->GetStart();
3299 // I need this value later on for notification but the pointer may become invalid
3300 const sal_Int32 nHintEnd
= *pHtEnd
;
3301 const bool bNoHintAdjustMode
= bool(SetAttrMode::NOHINTADJUST
& nMode
);
3303 // handle nesting attributes: inserting may fail due to overlap!
3304 if (pHint
->IsNesting())
3307 TryInsertNesting(rNode
, *static_txtattr_cast
<SwTextAttrNesting
*>(pHint
)));
3308 if (!bRet
) return false;
3310 // Currently REFMARK and TOXMARK have OverlapAllowed set to true.
3311 // These attributes may be inserted directly.
3312 // Also attributes without length may be inserted directly.
3313 // SETATTR_NOHINTADJUST is set e.g., during undo.
3314 // Portion building in not necessary during XML import.
3315 else if ( !bNoHintAdjustMode
&&
3316 !pHint
->IsOverlapAllowedAttr() &&
3317 !rNode
.GetDoc().IsInXMLImport() &&
3318 ( RES_TXTATR_AUTOFMT
== nWhich
||
3319 RES_TXTATR_CHARFMT
== nWhich
) )
3321 assert( nWhich
!= RES_TXTATR_AUTOFMT
||
3322 static_cast<const SwFormatAutoFormat
&>(pHint
->GetAttr()).GetStyleHandle()->GetPool() ==
3323 &rNode
.GetDoc().GetAttrPool());
3325 BuildPortions( rNode
, *pHint
, nMode
);
3327 if ( nHtStart
< nHintEnd
) // skip merging for 0-length attributes
3328 MergePortions( rNode
);
3332 // There may be more than one character style at the current position.
3333 // Take care of the sort number.
3334 // FME 2007-11-08 #i82989# in NOHINTADJUST mode, we want to insert
3335 // character attributes directly
3336 if (!bNoHintAdjustMode
3337 && ( (RES_TXTATR_CHARFMT
== nWhich
)
3338 // tdf#149978 also for import of automatic styles, could be produced by non-LO application
3339 || (RES_TXTATR_AUTOFMT
== nWhich
&& rNode
.GetDoc().IsInXMLImport())))
3341 BuildPortions( rNode
, *pHint
, nMode
);
3345 // #i82989# Check sort numbers in NoHintAdjustMode
3346 if ( RES_TXTATR_CHARFMT
== nWhich
)
3347 lcl_CheckSortNumber(*this, *static_txtattr_cast
<SwTextCharFormat
*>(pHint
));
3350 NoteInHistory( pHint
, true );
3354 // ... and notify listeners
3355 if ( rNode
.HasWriterListeners() )
3357 const SwUpdateAttr
aHint(nHtStart
, nHintEnd
, nWhich
, std::move(aWhichSublist
));
3358 rNode
.TriggerNodeUpdate(sw::LegacyModifyHint(&aHint
, &aHint
));
3362 if( !bNoHintAdjustMode
&& !rNode
.GetDoc().IsInReading() )
3369 void SwpHints::DeleteAtPos( const size_t nPos
)
3371 assert(!m_bStartMapNeedsSorting
&& "deleting at pos and the list needs sorting?");
3373 SwTextAttr
*pHint
= Get(nPos
);
3374 assert( pHint
->m_pHints
== this );
3375 // ChainDelete( pHint );
3376 NoteInHistory( pHint
);
3378 // optimization: nPos is the position in the Starts array
3379 SwTextAttr
*pHt
= m_HintsByStart
[ nPos
];
3380 m_HintsByStart
.erase( m_HintsByStart
.begin() + nPos
);
3382 if (m_bStartMapNeedsSorting
)
3384 if (m_bEndMapNeedsSorting
)
3386 if (m_bWhichMapNeedsSorting
)
3389 auto findIt
= std::lower_bound(m_HintsByEnd
.begin(), m_HintsByEnd
.end(), pHt
, CompareSwpHtEnd());
3390 assert(*findIt
== pHt
);
3391 m_HintsByEnd
.erase(findIt
);
3393 auto findIt2
= std::lower_bound(m_HintsByWhichAndStart
.begin(), m_HintsByWhichAndStart
.end(), pHt
, CompareSwpHtWhichStart());
3394 assert(*findIt2
== pHt
);
3395 m_HintsByWhichAndStart
.erase(findIt2
);
3397 pHt
->m_pHints
= nullptr;
3399 if( pHint
->Which() == RES_TXTATR_FIELD
)
3401 SwTextField
*const pTextField(static_txtattr_cast
<SwTextField
*>(pHint
));
3402 const SwFieldType
* pFieldTyp
= pTextField
->GetFormatField().GetField()->GetTyp();
3403 if( SwFieldIds::Dde
== pFieldTyp
->Which() )
3405 const SwTextNode
* pNd
= pTextField
->GetpTextNode();
3406 if( pNd
&& pNd
->GetNodes().IsDocNodes() )
3407 const_cast<SwDDEFieldType
*>(static_cast<const SwDDEFieldType
*>(pFieldTyp
))->DecRefCnt();
3408 pTextField
->ChgTextNode(nullptr);
3410 else if (m_bHiddenByParaField
3411 && m_rParent
.GetDoc().FieldCanHideParaWeight(pFieldTyp
->Which()))
3413 m_bCalcHiddenParaField
= true;
3416 else if ( pHint
->Which() == RES_TXTATR_ANNOTATION
)
3418 SwTextField
*const pTextField(static_txtattr_cast
<SwTextField
*>(pHint
));
3419 const_cast<SwFormatField
&>(pTextField
->GetFormatField()).Broadcast(
3420 SwFormatFieldHint(&pTextField
->GetFormatField(), SwFormatFieldHintWhich::REMOVED
));
3424 CHECK_NOTMERGED
; // called from BuildPortions
3428 /// precondition: pTextHt must be in this array
3429 void SwpHints::Delete( SwTextAttr
const * pTextHt
)
3431 const size_t nPos
= GetIndexOf( pTextHt
);
3432 assert(SAL_MAX_SIZE
!= nPos
);
3433 if( SAL_MAX_SIZE
!= nPos
)
3434 DeleteAtPos( nPos
);
3437 void SwTextNode::ClearSwpHintsArr( bool bDelFields
)
3443 while ( nPos
< m_pSwpHints
->Count() )
3445 SwTextAttr
* pDel
= m_pSwpHints
->Get( nPos
);
3448 switch( pDel
->Which() )
3450 case RES_TXTATR_FLYCNT
:
3451 case RES_TXTATR_FTN
:
3454 case RES_TXTATR_FIELD
:
3455 case RES_TXTATR_ANNOTATION
:
3456 case RES_TXTATR_INPUTFIELD
:
3466 m_pSwpHints
->DeleteAtPos( nPos
);
3467 DestroyAttr( pDel
);
3474 LanguageType
SwTextNode::GetLang( const sal_Int32 nBegin
, const sal_Int32 nLen
,
3475 sal_uInt16 nScript
) const
3477 LanguageType nRet
= LANGUAGE_DONTKNOW
;
3481 nScript
= g_pBreakIt
->GetRealScriptOfText( m_Text
, nBegin
);
3484 // #i91465# Consider nScript if pSwpHints == 0
3485 const sal_uInt16 nWhichId
= GetWhichOfScript( RES_CHRATR_LANGUAGE
, nScript
);
3489 const sal_Int32 nEnd
= nBegin
+ nLen
;
3490 const size_t nSize
= m_pSwpHints
->Count();
3491 for ( size_t i
= 0; i
< nSize
; ++i
)
3493 const SwTextAttr
*pHt
= m_pSwpHints
->Get(i
);
3494 const sal_Int32 nAttrStart
= pHt
->GetStart();
3495 if( nEnd
< nAttrStart
)
3498 const sal_uInt16 nWhich
= pHt
->Which();
3500 if( nWhichId
== nWhich
||
3501 ( ( pHt
->IsCharFormatAttr() || RES_TXTATR_AUTOFMT
== nWhich
) && CharFormat::IsItemIncluded( nWhichId
, pHt
) ) )
3503 const sal_Int32
*pEndIdx
= pHt
->End();
3504 // do the attribute and the range overlap?
3509 if( nAttrStart
>= nEnd
|| nBegin
>= *pEndIdx
)
3512 else if( nBegin
!= nAttrStart
|| ( nAttrStart
!= *pEndIdx
&& nBegin
))
3514 if( nAttrStart
>= nBegin
)
3516 if( pHt
->DontExpand() ? nBegin
>= *pEndIdx
: nBegin
> *pEndIdx
)
3519 const SfxPoolItem
* pItem
= CharFormat::GetItem( *pHt
, nWhichId
);
3520 const LanguageType nLng
= static_cast<const SvxLanguageItem
*>(pItem
)->GetLanguage();
3522 // does the attribute completely cover the range?
3523 if( nAttrStart
<= nBegin
&& nEnd
<= *pEndIdx
)
3525 else if( LANGUAGE_DONTKNOW
== nRet
)
3526 nRet
= nLng
; // partial overlap, the first one wins
3530 if( LANGUAGE_DONTKNOW
== nRet
)
3532 nRet
= static_cast<const SvxLanguageItem
&>(GetSwAttrSet().Get( nWhichId
)).GetLanguage();
3533 if( LANGUAGE_DONTKNOW
== nRet
)
3534 nRet
= GetAppLanguage();
3539 sal_Unicode
GetCharOfTextAttr( const SwTextAttr
& rAttr
)
3541 sal_Unicode cRet
= CH_TXTATR_BREAKWORD
;
3542 switch ( rAttr
.Which() )
3544 case RES_TXTATR_REFMARK
:
3545 case RES_TXTATR_TOXMARK
:
3546 case RES_TXTATR_ANNOTATION
:
3547 cRet
= CH_TXTATR_INWORD
;
3550 case RES_TXTATR_FIELD
:
3551 case RES_TXTATR_FLYCNT
:
3552 case RES_TXTATR_FTN
:
3553 case RES_TXTATR_META
:
3554 case RES_TXTATR_METAFIELD
:
3555 case RES_TXTATR_CONTENTCONTROL
:
3557 cRet
= CH_TXTATR_BREAKWORD
;
3560 case RES_TXTATR_LINEBREAK
:
3562 cRet
= CH_TXTATR_NEWLINE
;
3567 assert(!"GetCharOfTextAttr: unknown attr");
3573 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */