Line data Source code
1 : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 : /*
3 : * This file is part of the LibreOffice project.
4 : *
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/.
8 : *
9 : * This file incorporates work covered by the following license notice:
10 : *
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 .
18 : */
19 :
20 : #include <textdoc.hxx>
21 :
22 : #include <stdlib.h>
23 :
24 :
25 :
26 : // Vergleichmethode wird von QuickSort gerufen...
27 :
28 0 : static bool CompareStart( const TextCharAttrib* pFirst, const TextCharAttrib* pSecond )
29 : {
30 0 : return pFirst->GetStart() < pSecond->GetStart();
31 : }
32 :
33 : // -------------------------------------------------------------------------
34 : // (+) class TextCharAttrib
35 : // -------------------------------------------------------------------------
36 0 : TextCharAttrib::TextCharAttrib( const TextAttrib& rAttr, sal_uInt16 nStart, sal_uInt16 nEnd )
37 : {
38 0 : mpAttr = rAttr.Clone();
39 : mnStart = nStart,
40 0 : mnEnd = nEnd;
41 0 : }
42 :
43 0 : TextCharAttrib::TextCharAttrib( const TextCharAttrib& rTextCharAttrib )
44 : {
45 0 : mpAttr = rTextCharAttrib.GetAttr().Clone();
46 0 : mnStart = rTextCharAttrib.mnStart;
47 0 : mnEnd = rTextCharAttrib.mnEnd;
48 0 : }
49 :
50 0 : TextCharAttrib::~TextCharAttrib()
51 : {
52 0 : delete mpAttr;
53 0 : }
54 :
55 : // -------------------------------------------------------------------------
56 : // (+) class TextCharAttribList
57 : // -------------------------------------------------------------------------
58 :
59 0 : TextCharAttribList::TextCharAttribList()
60 : {
61 0 : mbHasEmptyAttribs = sal_False;
62 0 : }
63 :
64 0 : TextCharAttribList::~TextCharAttribList()
65 : {
66 : // PTRARR_DEL
67 0 : }
68 :
69 0 : void TextCharAttribList::Clear( sal_Bool bDestroyAttribs )
70 : {
71 0 : if ( bDestroyAttribs )
72 0 : for(iterator it = begin(); it != end(); ++it)
73 0 : delete *it;
74 0 : TextCharAttribs::clear();
75 0 : }
76 :
77 :
78 0 : void TextCharAttribList::InsertAttrib( TextCharAttrib* pAttrib )
79 : {
80 0 : if ( pAttrib->IsEmpty() )
81 0 : mbHasEmptyAttribs = sal_True;
82 :
83 0 : const sal_uInt16 nCount = size();
84 0 : const sal_uInt16 nStart = pAttrib->GetStart(); // vielleicht besser fuer Comp.Opt.
85 0 : sal_Bool bInserted = sal_False;
86 0 : for ( sal_uInt16 x = 0; x < nCount; x++ )
87 : {
88 0 : TextCharAttrib* pCurAttrib = GetAttrib( x );
89 0 : if ( pCurAttrib->GetStart() > nStart )
90 : {
91 0 : insert( begin() + x, pAttrib );
92 0 : bInserted = sal_True;
93 0 : break;
94 : }
95 : }
96 0 : if ( !bInserted )
97 0 : push_back( pAttrib );
98 0 : }
99 :
100 0 : void TextCharAttribList::ResortAttribs()
101 : {
102 0 : if ( !empty() )
103 0 : std::sort( begin(), end(), CompareStart );
104 0 : }
105 :
106 0 : TextCharAttrib* TextCharAttribList::FindAttrib( sal_uInt16 nWhich, sal_uInt16 nPos )
107 : {
108 : // Rueckwaerts, falls eins dort endet, das naechste startet.
109 : // => Das startende gilt...
110 :
111 0 : for ( sal_uInt16 nAttr = size(); nAttr; )
112 : {
113 0 : TextCharAttrib* pAttr = GetAttrib( --nAttr );
114 :
115 0 : if ( pAttr->GetEnd() < nPos )
116 0 : return 0;
117 :
118 0 : if ( ( pAttr->Which() == nWhich ) && pAttr->IsIn(nPos) )
119 0 : return pAttr;
120 : }
121 0 : return NULL;
122 : }
123 :
124 0 : TextCharAttrib* TextCharAttribList::FindNextAttrib( sal_uInt16 nWhich, sal_uInt16 nFromPos, sal_uInt16 nMaxPos ) const
125 : {
126 : DBG_ASSERT( nWhich, "FindNextAttrib: Which?" );
127 0 : const sal_uInt16 nAttribs = size();
128 0 : for ( sal_uInt16 nAttr = 0; nAttr < nAttribs; nAttr++ )
129 : {
130 0 : TextCharAttrib* pAttr = GetAttrib( nAttr );
131 0 : if ( ( pAttr->GetStart() >= nFromPos ) &&
132 0 : ( pAttr->GetEnd() <= nMaxPos ) &&
133 0 : ( pAttr->Which() == nWhich ) )
134 0 : return pAttr;
135 : }
136 0 : return NULL;
137 : }
138 :
139 0 : sal_Bool TextCharAttribList::HasAttrib( sal_uInt16 nWhich ) const
140 : {
141 0 : for ( sal_uInt16 nAttr = size(); nAttr; )
142 : {
143 0 : const TextCharAttrib* pAttr = GetAttrib( --nAttr );
144 0 : if ( pAttr->Which() == nWhich )
145 0 : return sal_True;
146 : }
147 0 : return sal_False;
148 : }
149 :
150 0 : sal_Bool TextCharAttribList::HasBoundingAttrib( sal_uInt16 nBound )
151 : {
152 : // Rueckwaerts, falls eins dort endet, das naechste startet.
153 : // => Das startende gilt...
154 0 : for ( sal_uInt16 nAttr = size(); nAttr; )
155 : {
156 0 : TextCharAttrib* pAttr = GetAttrib( --nAttr );
157 :
158 0 : if ( pAttr->GetEnd() < nBound )
159 0 : return sal_False;
160 :
161 0 : if ( ( pAttr->GetStart() == nBound ) || ( pAttr->GetEnd() == nBound ) )
162 0 : return sal_True;
163 : }
164 0 : return sal_False;
165 : }
166 :
167 0 : TextCharAttrib* TextCharAttribList::FindEmptyAttrib( sal_uInt16 nWhich, sal_uInt16 nPos )
168 : {
169 0 : if ( !mbHasEmptyAttribs )
170 0 : return 0;
171 :
172 0 : const sal_uInt16 nAttribs = size();
173 0 : for ( sal_uInt16 nAttr = 0; nAttr < nAttribs; nAttr++ )
174 : {
175 0 : TextCharAttrib* pAttr = GetAttrib( nAttr );
176 0 : if ( pAttr->GetStart() > nPos )
177 0 : return 0;
178 :
179 0 : if ( ( pAttr->GetStart() == nPos ) && ( pAttr->GetEnd() == nPos ) && ( pAttr->Which() == nWhich ) )
180 0 : return pAttr;
181 : }
182 0 : return 0;
183 : }
184 :
185 0 : void TextCharAttribList::DeleteEmptyAttribs()
186 : {
187 0 : for ( sal_uInt16 nAttr = 0; nAttr < size(); nAttr++ )
188 : {
189 0 : TextCharAttrib* pAttr = GetAttrib( nAttr );
190 0 : if ( pAttr->IsEmpty() )
191 : {
192 0 : erase( begin() + nAttr );
193 0 : delete pAttr;
194 0 : nAttr--;
195 : }
196 : }
197 0 : mbHasEmptyAttribs = sal_False;
198 0 : }
199 :
200 : // -------------------------------------------------------------------------
201 : // (+) class TextNode
202 : // -------------------------------------------------------------------------
203 :
204 0 : TextNode::TextNode( const String& rText ) :
205 0 : maText( rText )
206 : {
207 0 : }
208 :
209 0 : void TextNode::ExpandAttribs( sal_uInt16 nIndex, sal_uInt16 nNew )
210 : {
211 0 : if ( !nNew )
212 0 : return;
213 :
214 0 : sal_Bool bResort = sal_False;
215 0 : sal_uInt16 nAttribs = maCharAttribs.Count();
216 0 : for ( sal_uInt16 nAttr = 0; nAttr < nAttribs; nAttr++ )
217 : {
218 0 : TextCharAttrib* pAttrib = maCharAttribs.GetAttrib( nAttr );
219 0 : if ( pAttrib->GetEnd() >= nIndex )
220 : {
221 : // Alle Attribute hinter der Einfuegeposition verschieben...
222 0 : if ( pAttrib->GetStart() > nIndex )
223 : {
224 0 : pAttrib->MoveForward( nNew );
225 : }
226 : // 0: Leeres Attribut expandieren, wenn an Einfuegestelle
227 0 : else if ( pAttrib->IsEmpty() )
228 : {
229 : // Index nicht pruefen, leeres durfte nur dort liegen.
230 : // Wenn spaeter doch Ueberpruefung:
231 : // Spezialfall: Start == 0; AbsLen == 1, nNew = 1 => Expand, weil durch Absatzumbruch!
232 : // Start <= nIndex, End >= nIndex => Start=End=nIndex!
233 : // if ( pAttrib->GetStart() == nIndex )
234 0 : pAttrib->Expand( nNew );
235 : }
236 : // 1: Attribut startet davor, geht bis Index...
237 0 : else if ( pAttrib->GetEnd() == nIndex ) // Start muss davor liegen
238 : {
239 : // Nur expandieren, wenn kein Feature,
240 : // und wenn nicht in ExcludeListe!
241 : // Sonst geht z.B. ein UL bis zum neuen ULDB, beide expandieren
242 0 : if ( !maCharAttribs.FindEmptyAttrib( pAttrib->Which(), nIndex ) )
243 : {
244 0 : pAttrib->Expand( nNew );
245 : }
246 : else
247 0 : bResort = sal_True;
248 : }
249 : // 2: Attribut startet davor, geht hinter Index...
250 0 : else if ( ( pAttrib->GetStart() < nIndex ) && ( pAttrib->GetEnd() > nIndex ) )
251 : {
252 0 : pAttrib->Expand( nNew );
253 : }
254 : // 3: Attribut startet auf Index...
255 0 : else if ( pAttrib->GetStart() == nIndex )
256 : {
257 0 : if ( nIndex == 0 )
258 : {
259 0 : pAttrib->Expand( nNew );
260 : // bResort = sal_True; // es gibt ja keine Features mehr...
261 : }
262 : else
263 0 : pAttrib->MoveForward( nNew );
264 : }
265 : }
266 :
267 : DBG_ASSERT( pAttrib->GetStart() <= pAttrib->GetEnd(), "Expand: Attribut verdreht!" );
268 : DBG_ASSERT( ( pAttrib->GetEnd() <= maText.Len() ), "Expand: Attrib groesser als Absatz!" );
269 : DBG_ASSERT( !pAttrib->IsEmpty(), "Leeres Attribut nach ExpandAttribs?" );
270 : }
271 :
272 0 : if ( bResort )
273 0 : maCharAttribs.ResortAttribs();
274 : }
275 :
276 0 : void TextNode::CollapsAttribs( sal_uInt16 nIndex, sal_uInt16 nDeleted )
277 : {
278 0 : if ( !nDeleted )
279 0 : return;
280 :
281 0 : sal_Bool bResort = sal_False;
282 0 : sal_uInt16 nEndChanges = nIndex+nDeleted;
283 :
284 0 : for ( sal_uInt16 nAttr = 0; nAttr < maCharAttribs.Count(); nAttr++ )
285 : {
286 0 : TextCharAttrib* pAttrib = maCharAttribs.GetAttrib( nAttr );
287 0 : sal_Bool bDelAttr = sal_False;
288 0 : if ( pAttrib->GetEnd() >= nIndex )
289 : {
290 : // Alles Attribute hinter der Einfuegeposition verschieben...
291 0 : if ( pAttrib->GetStart() >= nEndChanges )
292 : {
293 0 : pAttrib->MoveBackward( nDeleted );
294 : }
295 : // 1. Innenliegende Attribute loeschen...
296 0 : else if ( ( pAttrib->GetStart() >= nIndex ) && ( pAttrib->GetEnd() <= nEndChanges ) )
297 : {
298 : // Spezialfall: Attrubt deckt genau den Bereich ab
299 : // => als leeres Attribut behalten.
300 0 : if ( ( pAttrib->GetStart() == nIndex ) && ( pAttrib->GetEnd() == nEndChanges ) )
301 0 : pAttrib->GetEnd() = nIndex; // leer
302 : else
303 0 : bDelAttr = sal_True;
304 : }
305 : // 2. Attribut beginnt davor, endet drinnen oder dahinter...
306 0 : else if ( ( pAttrib->GetStart() <= nIndex ) && ( pAttrib->GetEnd() > nIndex ) )
307 : {
308 0 : if ( pAttrib->GetEnd() <= nEndChanges ) // endet drinnen
309 0 : pAttrib->GetEnd() = nIndex;
310 : else
311 0 : pAttrib->Collaps( nDeleted ); // endet dahinter
312 : }
313 : // 3. Attribut beginnt drinnen, endet dahinter...
314 0 : else if ( ( pAttrib->GetStart() >= nIndex ) && ( pAttrib->GetEnd() > nEndChanges ) )
315 : {
316 : // Features duerfen nicht expandieren!
317 0 : pAttrib->GetStart() = nEndChanges;
318 0 : pAttrib->MoveBackward( nDeleted );
319 : }
320 : }
321 :
322 : DBG_ASSERT( pAttrib->GetStart() <= pAttrib->GetEnd(), "Collaps: Attribut verdreht!" );
323 : DBG_ASSERT( ( pAttrib->GetEnd() <= maText.Len()) || bDelAttr, "Collaps: Attrib groesser als Absatz!" );
324 0 : if ( bDelAttr /* || pAttrib->IsEmpty() */ )
325 : {
326 0 : bResort = sal_True;
327 0 : maCharAttribs.RemoveAttrib( nAttr );
328 0 : delete pAttrib;
329 0 : nAttr--;
330 : }
331 0 : else if ( pAttrib->IsEmpty() )
332 0 : maCharAttribs.HasEmptyAttribs() = sal_True;
333 : }
334 :
335 0 : if ( bResort )
336 0 : maCharAttribs.ResortAttribs();
337 : }
338 :
339 0 : void TextNode::InsertText( sal_uInt16 nPos, const String& rText )
340 : {
341 0 : maText.Insert( rText, nPos );
342 0 : ExpandAttribs( nPos, rText.Len() );
343 0 : }
344 :
345 0 : void TextNode::InsertText( sal_uInt16 nPos, sal_Unicode c )
346 : {
347 0 : maText.Insert( c, nPos );
348 0 : ExpandAttribs( nPos, 1 );
349 0 : }
350 :
351 0 : void TextNode::RemoveText( sal_uInt16 nPos, sal_uInt16 nChars )
352 : {
353 0 : maText.Erase( nPos, nChars );
354 0 : CollapsAttribs( nPos, nChars );
355 0 : }
356 :
357 0 : TextNode* TextNode::Split( sal_uInt16 nPos, sal_Bool bKeepEndingAttribs )
358 : {
359 0 : String aNewText;
360 0 : if ( nPos < maText.Len() )
361 : {
362 0 : aNewText = maText.Copy( nPos );
363 0 : maText.Erase( nPos );
364 : }
365 0 : TextNode* pNew = new TextNode( aNewText );
366 :
367 0 : for ( sal_uInt16 nAttr = 0; nAttr < maCharAttribs.Count(); nAttr++ )
368 : {
369 0 : TextCharAttrib* pAttrib = maCharAttribs.GetAttrib( nAttr );
370 0 : if ( pAttrib->GetEnd() < nPos )
371 : {
372 : // bleiben unveraendert....
373 : ;
374 : }
375 0 : else if ( pAttrib->GetEnd() == nPos )
376 : {
377 : // muessen als leeres Attribut kopiert werden.
378 : // !FindAttrib nur sinnvoll, wenn Rueckwaerts durch Liste!
379 0 : if ( bKeepEndingAttribs && !pNew->maCharAttribs.FindAttrib( pAttrib->Which(), 0 ) )
380 : {
381 0 : TextCharAttrib* pNewAttrib = new TextCharAttrib( *pAttrib );
382 0 : pNewAttrib->GetStart() = 0;
383 0 : pNewAttrib->GetEnd() = 0;
384 0 : pNew->maCharAttribs.InsertAttrib( pNewAttrib );
385 : }
386 : }
387 0 : else if ( pAttrib->IsInside( nPos ) || ( !nPos && !pAttrib->GetStart() ) )
388 : {
389 : // Wenn ganz vorne gecuttet wird, muss das Attribut erhalten bleiben!
390 : // muessen kopiert und geaendert werden
391 0 : TextCharAttrib* pNewAttrib = new TextCharAttrib( *pAttrib );
392 0 : pNewAttrib->GetStart() = 0;
393 0 : pNewAttrib->GetEnd() = pAttrib->GetEnd()-nPos;
394 0 : pNew->maCharAttribs.InsertAttrib( pNewAttrib );
395 : // stutzen:
396 0 : pAttrib->GetEnd() = nPos;
397 : }
398 : else
399 : {
400 : DBG_ASSERT( pAttrib->GetStart() >= nPos, "Start < nPos!" );
401 : DBG_ASSERT( pAttrib->GetEnd() >= nPos, "End < nPos!" );
402 : // alle dahinter verschieben in den neuen Node (this)
403 0 : maCharAttribs.RemoveAttrib( nAttr );
404 0 : pNew->maCharAttribs.InsertAttrib( pAttrib );
405 0 : pAttrib->GetStart() = pAttrib->GetStart() - nPos;
406 0 : pAttrib->GetEnd() = pAttrib->GetEnd() - nPos;
407 0 : nAttr--;
408 : }
409 : }
410 0 : return pNew;
411 : }
412 :
413 0 : void TextNode::Append( const TextNode& rNode )
414 : {
415 0 : sal_uInt16 nOldLen = maText.Len();
416 :
417 0 : maText += rNode.GetText();
418 :
419 0 : const sal_uInt16 nAttribs = rNode.GetCharAttribs().Count();
420 0 : for ( sal_uInt16 nAttr = 0; nAttr < nAttribs; nAttr++ )
421 : {
422 0 : TextCharAttrib* pAttrib = rNode.GetCharAttribs().GetAttrib( nAttr );
423 0 : sal_Bool bMelted = sal_False;
424 0 : if ( pAttrib->GetStart() == 0 )
425 : {
426 : // Evtl koennen Attribute zusammengefasst werden:
427 0 : sal_uInt16 nTmpAttribs = maCharAttribs.Count();
428 0 : for ( sal_uInt16 nTmpAttr = 0; nTmpAttr < nTmpAttribs; nTmpAttr++ )
429 : {
430 0 : TextCharAttrib* pTmpAttrib = maCharAttribs.GetAttrib( nTmpAttr );
431 :
432 0 : if ( pTmpAttrib->GetEnd() == nOldLen )
433 : {
434 0 : if ( ( pTmpAttrib->Which() == pAttrib->Which() ) &&
435 0 : ( pTmpAttrib->GetAttr() == pAttrib->GetAttr() ) )
436 : {
437 0 : pTmpAttrib->GetEnd() =
438 0 : pTmpAttrib->GetEnd() + pAttrib->GetLen();
439 0 : bMelted = sal_True;
440 0 : break; // es kann nur eins von der Sorte an der Stelle geben
441 : }
442 : }
443 : }
444 : }
445 :
446 0 : if ( !bMelted )
447 : {
448 0 : TextCharAttrib* pNewAttrib = new TextCharAttrib( *pAttrib );
449 0 : pNewAttrib->GetStart() = pNewAttrib->GetStart() + nOldLen;
450 0 : pNewAttrib->GetEnd() = pNewAttrib->GetEnd() + nOldLen;
451 0 : maCharAttribs.InsertAttrib( pNewAttrib );
452 : }
453 : }
454 0 : }
455 :
456 : // -------------------------------------------------------------------------
457 : // (+) class TextDoc
458 : // -------------------------------------------------------------------------
459 :
460 0 : TextDoc::TextDoc()
461 : {
462 0 : mnLeftMargin = 0;
463 0 : };
464 :
465 0 : TextDoc::~TextDoc()
466 : {
467 0 : DestroyTextNodes();
468 0 : }
469 :
470 0 : void TextDoc::Clear()
471 : {
472 0 : DestroyTextNodes();
473 0 : }
474 :
475 0 : void TextDoc::DestroyTextNodes()
476 : {
477 0 : for ( sal_uLong nNode = 0; nNode < maTextNodes.Count(); nNode++ )
478 0 : delete maTextNodes.GetObject( nNode );
479 0 : maTextNodes.clear();
480 0 : }
481 :
482 0 : String TextDoc::GetText( const sal_Unicode* pSep ) const
483 : {
484 0 : sal_uLong nLen = GetTextLen( pSep );
485 0 : sal_uLong nNodes = maTextNodes.Count();
486 :
487 0 : if ( nLen > STRING_MAXLEN )
488 : {
489 : OSL_FAIL( "Text zu gross fuer String" );
490 0 : return String();
491 : }
492 :
493 0 : String aASCIIText;
494 0 : sal_uLong nLastNode = nNodes-1;
495 0 : for ( sal_uLong nNode = 0; nNode < nNodes; nNode++ )
496 : {
497 0 : TextNode* pNode = maTextNodes.GetObject( nNode );
498 0 : String aTmp( pNode->GetText() );
499 0 : aASCIIText += aTmp;
500 0 : if ( pSep && ( nNode != nLastNode ) )
501 0 : aASCIIText += pSep;
502 0 : }
503 :
504 0 : return aASCIIText;
505 : }
506 :
507 0 : XubString TextDoc::GetText( sal_uLong nPara ) const
508 : {
509 0 : XubString aText;
510 0 : TextNode* pNode = ( nPara < maTextNodes.Count() ) ? maTextNodes.GetObject( nPara ) : 0;
511 0 : if ( pNode )
512 0 : aText = pNode->GetText();
513 :
514 0 : return aText;
515 : }
516 :
517 :
518 0 : sal_uLong TextDoc::GetTextLen( const sal_Unicode* pSep, const TextSelection* pSel ) const
519 : {
520 0 : sal_uLong nLen = 0;
521 0 : sal_uLong nNodes = maTextNodes.Count();
522 0 : if ( nNodes )
523 : {
524 0 : sal_uLong nStartNode = 0;
525 0 : sal_uLong nEndNode = nNodes-1;
526 0 : if ( pSel )
527 : {
528 0 : nStartNode = pSel->GetStart().GetPara();
529 0 : nEndNode = pSel->GetEnd().GetPara();
530 : }
531 :
532 0 : for ( sal_uLong nNode = nStartNode; nNode <= nEndNode; nNode++ )
533 : {
534 0 : TextNode* pNode = maTextNodes.GetObject( nNode );
535 :
536 0 : sal_uInt16 nS = 0;
537 0 : sal_uLong nE = pNode->GetText().Len();
538 0 : if ( pSel && ( nNode == pSel->GetStart().GetPara() ) )
539 0 : nS = pSel->GetStart().GetIndex();
540 0 : if ( pSel && ( nNode == pSel->GetEnd().GetPara() ) )
541 0 : nE = pSel->GetEnd().GetIndex();
542 :
543 0 : nLen += ( nE - nS );
544 : }
545 :
546 0 : if ( pSep )
547 0 : nLen += (nEndNode-nStartNode) * rtl_ustr_getLength(pSep);
548 : }
549 :
550 0 : return nLen;
551 : }
552 :
553 0 : TextPaM TextDoc::InsertText( const TextPaM& rPaM, sal_Unicode c )
554 : {
555 : DBG_ASSERT( c != 0x0A, "TextDoc::InsertText: Zeilentrenner in Absatz nicht erlaubt!" );
556 : DBG_ASSERT( c != 0x0D, "TextDoc::InsertText: Zeilentrenner in Absatz nicht erlaubt!" );
557 :
558 0 : TextNode* pNode = maTextNodes.GetObject( rPaM.GetPara() );
559 0 : pNode->InsertText( rPaM.GetIndex(), c );
560 :
561 0 : TextPaM aPaM( rPaM.GetPara(), rPaM.GetIndex()+1 );
562 0 : return aPaM;
563 : }
564 :
565 0 : TextPaM TextDoc::InsertText( const TextPaM& rPaM, const XubString& rStr )
566 : {
567 : DBG_ASSERT( rStr.Search( 0x0A ) == STRING_NOTFOUND, "TextDoc::InsertText: Zeilentrenner in Absatz nicht erlaubt!" );
568 : DBG_ASSERT( rStr.Search( 0x0D ) == STRING_NOTFOUND, "TextDoc::InsertText: Zeilentrenner in Absatz nicht erlaubt!" );
569 :
570 0 : TextNode* pNode = maTextNodes.GetObject( rPaM.GetPara() );
571 0 : pNode->InsertText( rPaM.GetIndex(), rStr );
572 :
573 0 : TextPaM aPaM( rPaM.GetPara(), rPaM.GetIndex()+rStr.Len() );
574 0 : return aPaM;
575 : }
576 :
577 0 : TextPaM TextDoc::InsertParaBreak( const TextPaM& rPaM, sal_Bool bKeepEndingAttribs )
578 : {
579 0 : TextNode* pNode = maTextNodes.GetObject( rPaM.GetPara() );
580 0 : TextNode* pNew = pNode->Split( rPaM.GetIndex(), bKeepEndingAttribs );
581 :
582 0 : maTextNodes.Insert( pNew, rPaM.GetPara()+1 );
583 :
584 0 : TextPaM aPaM( rPaM.GetPara()+1, 0 );
585 0 : return aPaM;
586 : }
587 :
588 0 : TextPaM TextDoc::ConnectParagraphs( TextNode* pLeft, TextNode* pRight )
589 : {
590 0 : sal_uInt16 nPrevLen = pLeft->GetText().Len();
591 0 : pLeft->Append( *pRight );
592 :
593 : // der rechte verschwindet.
594 0 : sal_uLong nRight = maTextNodes.GetPos( pRight );
595 0 : maTextNodes.Remove( nRight );
596 0 : delete pRight;
597 :
598 0 : sal_uLong nLeft = maTextNodes.GetPos( pLeft );
599 0 : TextPaM aPaM( nLeft, nPrevLen );
600 0 : return aPaM;
601 : }
602 :
603 0 : TextPaM TextDoc::RemoveChars( const TextPaM& rPaM, sal_uInt16 nChars )
604 : {
605 0 : TextNode* pNode = maTextNodes.GetObject( rPaM.GetPara() );
606 0 : pNode->RemoveText( rPaM.GetIndex(), nChars );
607 :
608 0 : return rPaM;
609 : }
610 :
611 0 : sal_Bool TextDoc::IsValidPaM( const TextPaM& rPaM )
612 : {
613 0 : if ( rPaM.GetPara() >= maTextNodes.Count() )
614 : {
615 : OSL_FAIL( "PaM: Para out of range" );
616 0 : return sal_False;
617 : }
618 0 : TextNode * pNode = maTextNodes.GetObject( rPaM.GetPara() );
619 0 : if ( rPaM.GetIndex() > pNode->GetText().Len() )
620 : {
621 : OSL_FAIL( "PaM: Index out of range" );
622 0 : return sal_False;
623 : }
624 0 : return sal_True;
625 : }
626 :
627 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|