/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include // compare function called by QuickSort static bool CompareStart( const TextCharAttrib& pFirst, const TextCharAttrib& pSecond ) { return pFirst.GetStart() < pSecond.GetStart(); } TextCharAttrib::TextCharAttrib( const TextAttrib& rAttr, sal_Int32 nStart, sal_Int32 nEnd ) { mpAttr = rAttr.Clone(); mnStart = nStart, mnEnd = nEnd; } TextCharAttrib::TextCharAttrib( const TextCharAttrib& rTextCharAttrib ) { mpAttr = rTextCharAttrib.GetAttr().Clone(); mnStart = rTextCharAttrib.mnStart; mnEnd = rTextCharAttrib.mnEnd; } TextCharAttrib::~TextCharAttrib() { delete mpAttr; } TextCharAttribList::TextCharAttribList() { mbHasEmptyAttribs = false; } TextCharAttribList::~TextCharAttribList() { // PTRARR_DEL } void TextCharAttribList::Clear() { maAttribs.clear(); } void TextCharAttribList::InsertAttrib( TextCharAttrib* pAttrib ) { if ( pAttrib->IsEmpty() ) mbHasEmptyAttribs = true; const sal_Int32 nStart = pAttrib->GetStart(); // maybe better for Comp.Opt. bool bInserted = false; for (TextCharAttribs::iterator it = maAttribs.begin(); it != maAttribs.end(); ++it) { if ( it->GetStart() > nStart ) { maAttribs.insert( it, pAttrib ); bInserted = true; break; } } if ( !bInserted ) maAttribs.push_back( pAttrib ); } void TextCharAttribList::ResortAttribs() { maAttribs.sort(CompareStart); } TextCharAttrib* TextCharAttribList::FindAttrib( sal_uInt16 nWhich, sal_Int32 nPos ) { for (TextCharAttribs::reverse_iterator it = maAttribs.rbegin(); it != maAttribs.rend(); ++it) { if ( it->GetEnd() < nPos ) return 0; if ( ( it->Which() == nWhich ) && it->IsIn(nPos) ) return &*it; } return NULL; } const TextCharAttrib* TextCharAttribList::FindNextAttrib( sal_uInt16 nWhich, sal_Int32 nFromPos, sal_Int32 nMaxPos ) const { DBG_ASSERT( nWhich, "FindNextAttrib: Which?" ); for (TextCharAttribs::const_iterator it = maAttribs.begin(); it != maAttribs.end(); ++it) { if ( ( it->GetStart() >= nFromPos ) && ( it->GetEnd() <= nMaxPos ) && ( it->Which() == nWhich ) ) return &*it; } return NULL; } bool TextCharAttribList::HasAttrib( sal_uInt16 nWhich ) const { for (TextCharAttribs::const_reverse_iterator it = maAttribs.rbegin(); it != maAttribs.rend(); ++it) { if ( it->Which() == nWhich ) return true; } return false; } bool TextCharAttribList::HasBoundingAttrib( sal_Int32 nBound ) { for (TextCharAttribs::reverse_iterator it = maAttribs.rbegin(); it != maAttribs.rend(); ++it) { if ( it->GetEnd() < nBound ) return false; if ( ( it->GetStart() == nBound ) || ( it->GetEnd() == nBound ) ) return true; } return false; } TextCharAttrib* TextCharAttribList::FindEmptyAttrib( sal_uInt16 nWhich, sal_Int32 nPos ) { if ( !mbHasEmptyAttribs ) return 0; for (TextCharAttribs::iterator it = maAttribs.begin(); it != maAttribs.end(); ++it) { if ( it->GetStart() > nPos ) return 0; if ( ( it->GetStart() == nPos ) && ( it->GetEnd() == nPos ) && ( it->Which() == nWhich ) ) return &*it; } return 0; } void TextCharAttribList::DeleteEmptyAttribs() { maAttribs.erase_if(boost::mem_fn(&TextCharAttrib::IsEmpty)); mbHasEmptyAttribs = false; } TextNode::TextNode( const OUString& rText ) : maText( rText ) { } void TextNode::ExpandAttribs( sal_Int32 nIndex, sal_Int32 nNew ) { if ( !nNew ) return; bool bResort = false; sal_uInt16 nAttribs = maCharAttribs.Count(); for ( sal_uInt16 nAttr = 0; nAttr < nAttribs; nAttr++ ) { TextCharAttrib& rAttrib = maCharAttribs.GetAttrib( nAttr ); if ( rAttrib.GetEnd() >= nIndex ) { // move all attributes that are behind the cursor if ( rAttrib.GetStart() > nIndex ) { rAttrib.MoveForward( nNew ); } // 0: expand empty attribute, if at cursor else if ( rAttrib.IsEmpty() ) { // Do not check the index; empty one may only be here. // If checking later anyway, special case: // Start == 0; AbsLen == 1, nNew = 1 => Expand due to new paragraph! // Start <= nIndex, End >= nIndex => Start=End=nIndex! rAttrib.Expand( nNew ); } // 1: attribute starts before and reaches up to index else if ( rAttrib.GetEnd() == nIndex ) // start must be before { // Only expand if no feature and not in Exclude list! // Otherwise e.g. an UL would go until the new ULDB, thus expand both. if ( !maCharAttribs.FindEmptyAttrib( rAttrib.Which(), nIndex ) ) { rAttrib.Expand( nNew ); } else bResort = true; } // 2: attribute starts before and reaches past the index else if ( ( rAttrib.GetStart() < nIndex ) && ( rAttrib.GetEnd() > nIndex ) ) { rAttrib.Expand( nNew ); } // 3: attribute starts at Index else if ( rAttrib.GetStart() == nIndex ) { if ( nIndex == 0 ) { rAttrib.Expand( nNew ); } else rAttrib.MoveForward( nNew ); } } DBG_ASSERT( rAttrib.GetStart() <= rAttrib.GetEnd(), "Expand: Attribut verdreht!" ); DBG_ASSERT( ( rAttrib.GetEnd() <= maText.getLength() ), "Expand: Attrib groesser als Absatz!" ); DBG_ASSERT( !rAttrib.IsEmpty(), "Leeres Attribut nach ExpandAttribs?" ); } if ( bResort ) maCharAttribs.ResortAttribs(); } void TextNode::CollapsAttribs( sal_Int32 nIndex, sal_Int32 nDeleted ) { if ( !nDeleted ) return; bool bResort = false; const sal_Int32 nEndChanges = nIndex+nDeleted; for ( sal_uInt16 nAttr = 0; nAttr < maCharAttribs.Count(); nAttr++ ) { TextCharAttrib& rAttrib = maCharAttribs.GetAttrib( nAttr ); bool bDelAttr = false; if ( rAttrib.GetEnd() >= nIndex ) { // move all attributes that are behind the cursor if ( rAttrib.GetStart() >= nEndChanges ) { rAttrib.MoveBackward( nDeleted ); } // 1. delete inner attributes else if ( ( rAttrib.GetStart() >= nIndex ) && ( rAttrib.GetEnd() <= nEndChanges ) ) { // special case: attribute covers the region exactly // => keep as an empty attribute if ( ( rAttrib.GetStart() == nIndex ) && ( rAttrib.GetEnd() == nEndChanges ) ) rAttrib.GetEnd() = nIndex; // empty else bDelAttr = true; } // 2. attribute starts before, ends inside or after else if ( ( rAttrib.GetStart() <= nIndex ) && ( rAttrib.GetEnd() > nIndex ) ) { if ( rAttrib.GetEnd() <= nEndChanges ) // ends inside rAttrib.GetEnd() = nIndex; else rAttrib.Collaps( nDeleted ); // ends after } // 3. attribute starts inside, ends after else if ( ( rAttrib.GetStart() >= nIndex ) && ( rAttrib.GetEnd() > nEndChanges ) ) { // features are not allowed to expand! rAttrib.GetStart() = nEndChanges; rAttrib.MoveBackward( nDeleted ); } } DBG_ASSERT( rAttrib.GetStart() <= rAttrib.GetEnd(), "Collaps: Attribut verdreht!" ); DBG_ASSERT( ( rAttrib.GetEnd() <= maText.getLength()) || bDelAttr, "Collaps: Attrib groesser als Absatz!" ); if ( bDelAttr /* || rAttrib.IsEmpty() */ ) { bResort = true; maCharAttribs.RemoveAttrib( nAttr ); nAttr--; } else if ( rAttrib.IsEmpty() ) maCharAttribs.HasEmptyAttribs() = true; } if ( bResort ) maCharAttribs.ResortAttribs(); } void TextNode::InsertText( sal_Int32 nPos, const OUString& rText ) { maText = maText.replaceAt( nPos, 0, rText ); ExpandAttribs( nPos, rText.getLength() ); } void TextNode::InsertText( sal_Int32 nPos, sal_Unicode c ) { maText = maText.replaceAt( nPos, 0, OUString(c) ); ExpandAttribs( nPos, 1 ); } void TextNode::RemoveText( sal_Int32 nPos, sal_Int32 nChars ) { maText = maText.replaceAt( nPos, nChars, "" ); CollapsAttribs( nPos, nChars ); } TextNode* TextNode::Split( sal_Int32 nPos, bool bKeepEndingAttribs ) { OUString aNewText; if ( nPos < maText.getLength() ) { aNewText = maText.copy( nPos ); maText = maText.copy(0, nPos); } TextNode* pNew = new TextNode( aNewText ); for ( sal_uInt16 nAttr = 0; nAttr < maCharAttribs.Count(); nAttr++ ) { TextCharAttrib& rAttrib = maCharAttribs.GetAttrib( nAttr ); if ( rAttrib.GetEnd() < nPos ) { // no change ; } else if ( rAttrib.GetEnd() == nPos ) { // must be copied as an empty attribute // !FindAttrib only sensible if traversing backwards through the list! if ( bKeepEndingAttribs && !pNew->maCharAttribs.FindAttrib( rAttrib.Which(), 0 ) ) { TextCharAttrib* pNewAttrib = new TextCharAttrib( rAttrib ); pNewAttrib->GetStart() = 0; pNewAttrib->GetEnd() = 0; pNew->maCharAttribs.InsertAttrib( pNewAttrib ); } } else if ( rAttrib.IsInside( nPos ) || ( !nPos && !rAttrib.GetStart() ) ) { // If cutting at the very beginning, the attribute has to be // copied and changed TextCharAttrib* pNewAttrib = new TextCharAttrib( rAttrib ); pNewAttrib->GetStart() = 0; pNewAttrib->GetEnd() = rAttrib.GetEnd()-nPos; pNew->maCharAttribs.InsertAttrib( pNewAttrib ); // trim rAttrib.GetEnd() = nPos; } else { DBG_ASSERT( rAttrib.GetStart() >= nPos, "Start < nPos!" ); DBG_ASSERT( rAttrib.GetEnd() >= nPos, "End < nPos!" ); // move all into the new node (this) maCharAttribs.RemoveAttrib( nAttr ); pNew->maCharAttribs.InsertAttrib( &rAttrib ); rAttrib.GetStart() = rAttrib.GetStart() - nPos; rAttrib.GetEnd() = rAttrib.GetEnd() - nPos; nAttr--; } } return pNew; } void TextNode::Append( const TextNode& rNode ) { sal_Int32 nOldLen = maText.getLength(); maText += rNode.GetText(); const sal_uInt16 nAttribs = rNode.GetCharAttribs().Count(); for ( sal_uInt16 nAttr = 0; nAttr < nAttribs; nAttr++ ) { const TextCharAttrib& rAttrib = rNode.GetCharAttrib( nAttr ); bool bMelted = false; if ( rAttrib.GetStart() == 0 ) { // potentially merge attributes sal_uInt16 nTmpAttribs = maCharAttribs.Count(); for ( sal_uInt16 nTmpAttr = 0; nTmpAttr < nTmpAttribs; nTmpAttr++ ) { TextCharAttrib& rTmpAttrib = maCharAttribs.GetAttrib( nTmpAttr ); if ( rTmpAttrib.GetEnd() == nOldLen ) { if ( ( rTmpAttrib.Which() == rAttrib.Which() ) && ( rTmpAttrib.GetAttr() == rAttrib.GetAttr() ) ) { rTmpAttrib.GetEnd() = rTmpAttrib.GetEnd() + rAttrib.GetLen(); bMelted = true; break; // there can be only one of this type at this position } } } } if ( !bMelted ) { TextCharAttrib* pNewAttrib = new TextCharAttrib( rAttrib ); pNewAttrib->GetStart() = pNewAttrib->GetStart() + nOldLen; pNewAttrib->GetEnd() = pNewAttrib->GetEnd() + nOldLen; maCharAttribs.InsertAttrib( pNewAttrib ); } } } TextDoc::TextDoc() { mnLeftMargin = 0; }; TextDoc::~TextDoc() { DestroyTextNodes(); } void TextDoc::Clear() { DestroyTextNodes(); } void TextDoc::DestroyTextNodes() { for ( auto pNode : maTextNodes ) delete pNode; maTextNodes.clear(); } OUString TextDoc::GetText( const sal_Unicode* pSep ) const { sal_uInt32 nNodes = static_cast(maTextNodes.size()); OUString aASCIIText; const sal_uInt32 nLastNode = nNodes-1; for ( sal_uInt32 nNode = 0; nNode < nNodes; ++nNode ) { TextNode* pNode = maTextNodes[ nNode ]; OUString aTmp( pNode->GetText() ); aASCIIText += aTmp; if ( pSep && ( nNode != nLastNode ) ) aASCIIText += pSep; } return aASCIIText; } OUString TextDoc::GetText( sal_uInt32 nPara ) const { OUString aText; TextNode* pNode = ( nPara < maTextNodes.size() ) ? maTextNodes[ nPara ] : 0; if ( pNode ) aText = pNode->GetText(); return aText; } sal_uLong TextDoc::GetTextLen( const sal_Unicode* pSep, const TextSelection* pSel ) const { sal_uLong nLen = 0; sal_uInt32 nNodes = static_cast(maTextNodes.size()); if ( nNodes ) { sal_uInt32 nStartNode = 0; sal_uInt32 nEndNode = nNodes-1; if ( pSel ) { nStartNode = pSel->GetStart().GetPara(); nEndNode = pSel->GetEnd().GetPara(); } for ( sal_uInt32 nNode = nStartNode; nNode <= nEndNode; ++nNode ) { TextNode* pNode = maTextNodes[ nNode ]; sal_Int32 nS = 0; sal_Int32 nE = pNode->GetText().getLength(); if ( pSel && ( nNode == pSel->GetStart().GetPara() ) ) nS = pSel->GetStart().GetIndex(); if ( pSel && ( nNode == pSel->GetEnd().GetPara() ) ) nE = pSel->GetEnd().GetIndex(); nLen += ( nE - nS ); } if ( pSep ) nLen += (nEndNode-nStartNode) * rtl_ustr_getLength(pSep); } return nLen; } TextPaM TextDoc::InsertText( const TextPaM& rPaM, sal_Unicode c ) { DBG_ASSERT( c != 0x0A, "TextDoc::InsertText: Zeilentrenner in Absatz nicht erlaubt!" ); DBG_ASSERT( c != 0x0D, "TextDoc::InsertText: Zeilentrenner in Absatz nicht erlaubt!" ); TextNode* pNode = maTextNodes[ rPaM.GetPara() ]; pNode->InsertText( rPaM.GetIndex(), c ); TextPaM aPaM( rPaM.GetPara(), rPaM.GetIndex()+1 ); return aPaM; } TextPaM TextDoc::InsertText( const TextPaM& rPaM, const OUString& rStr ) { DBG_ASSERT( rStr.indexOf( 0x0A ) == -1, "TextDoc::InsertText: Zeilentrenner in Absatz nicht erlaubt!" ); DBG_ASSERT( rStr.indexOf( 0x0D ) == -1, "TextDoc::InsertText: Zeilentrenner in Absatz nicht erlaubt!" ); TextNode* pNode = maTextNodes[ rPaM.GetPara() ]; pNode->InsertText( rPaM.GetIndex(), rStr ); TextPaM aPaM( rPaM.GetPara(), rPaM.GetIndex()+rStr.getLength() ); return aPaM; } TextPaM TextDoc::InsertParaBreak( const TextPaM& rPaM, bool bKeepEndingAttribs ) { TextNode* pNode = maTextNodes[ rPaM.GetPara() ]; TextNode* pNew = pNode->Split( rPaM.GetIndex(), bKeepEndingAttribs ); DBG_ASSERT( maTextNodes.size()GetText().getLength(); pLeft->Append( *pRight ); // the paragraph on the right vanishes maTextNodes.erase( std::find( maTextNodes.begin(), maTextNodes.end(), pRight ) ); delete pRight; sal_uLong nLeft = ::std::find( maTextNodes.begin(), maTextNodes.end(), pLeft ) - maTextNodes.begin(); TextPaM aPaM( nLeft, nPrevLen ); return aPaM; } TextPaM TextDoc::RemoveChars( const TextPaM& rPaM, sal_Int32 nChars ) { TextNode* pNode = maTextNodes[ rPaM.GetPara() ]; pNode->RemoveText( rPaM.GetIndex(), nChars ); return rPaM; } bool TextDoc::IsValidPaM( const TextPaM& rPaM ) { if ( rPaM.GetPara() >= maTextNodes.size() ) { OSL_FAIL( "PaM: Para out of range" ); return false; } TextNode * pNode = maTextNodes[ rPaM.GetPara() ]; if ( rPaM.GetIndex() > pNode->GetText().getLength() ) { OSL_FAIL( "PaM: Index out of range" ); return false; } return true; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */