diff options
Diffstat (limited to 'sw/source/core/text/redlnitr.cxx')
-rw-r--r-- | sw/source/core/text/redlnitr.cxx | 504 |
1 files changed, 504 insertions, 0 deletions
diff --git a/sw/source/core/text/redlnitr.cxx b/sw/source/core/text/redlnitr.cxx new file mode 100644 index 000000000000..441077728afe --- /dev/null +++ b/sw/source/core/text/redlnitr.cxx @@ -0,0 +1,504 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ +/************************************************************************* + * + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * Copyright 2000, 2010 Oracle and/or its affiliates. + * + * OpenOffice.org - a multi-platform office productivity suite + * + * This file is part of OpenOffice.org. + * + * OpenOffice.org is free software: you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License version 3 + * only, as published by the Free Software Foundation. + * + * OpenOffice.org is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Lesser General Public License version 3 for more details + * (a copy is included in the LICENSE file that accompanied this code). + * + * You should have received a copy of the GNU Lesser General Public License + * version 3 along with OpenOffice.org. If not, see + * <http://www.openoffice.org/license.html> + * for a copy of the LGPLv3 License. + * + ************************************************************************/ + +// MARKER(update_precomp.py): autogen include statement, do not remove +#include "precompiled_sw.hxx" + + +#include "hintids.hxx" +#include <svl/whiter.hxx> +#include <tools/shl.hxx> +#include <com/sun/star/i18n/ScriptType.hdl> +#include <swmodule.hxx> +#include <redline.hxx> // SwRedline +#include <txtatr.hxx> // SwTxt ... +#include <docary.hxx> // SwRedlineTbl +#include <itratr.hxx> // SwAttrIter +#include <ndtxt.hxx> // SwTxtNode +#include <doc.hxx> // SwDoc +#include <rootfrm.hxx> +#include <breakit.hxx> +#include <vcl/keycodes.hxx> +#include <vcl/cmdevt.hxx> +#include <vcl/settings.hxx> +#include <txtfrm.hxx> // SwTxtFrm +#include <vcl/svapp.hxx> +#include <redlnitr.hxx> +#include <extinput.hxx> +#include <sfx2/printer.hxx> +#include <vcl/window.hxx> + +using namespace ::com::sun::star; + +/************************************************************************* + * SwAttrIter::CtorInitAttrIter() + *************************************************************************/ +void SwAttrIter::CtorInitAttrIter( SwTxtNode& rTxtNode, SwScriptInfo& rScrInf, SwTxtFrm* pFrm ) +{ + // Beim HTML-Import kann es vorkommen, dass kein Layout existiert. + SwRootFrm* pRootFrm = rTxtNode.getIDocumentLayoutAccess()->GetCurrentLayout(); + pShell = pRootFrm ? pRootFrm->GetCurrShell() : 0; //swmod 080218 + + pScriptInfo = &rScrInf; + + // attributes set at the whole paragraph + pAttrSet = rTxtNode.GetpSwAttrSet(); + // attribute array + pHints = rTxtNode.GetpSwpHints(); + + // Build a font matching the default paragraph style: + SwFontAccess aFontAccess( &rTxtNode.GetAnyFmtColl(), pShell ); + delete pFnt; + pFnt = new SwFont( *aFontAccess.Get()->GetFont() ); + + // set font to vertical if frame layout is vertical + sal_Bool bVertLayout = sal_False; + sal_Bool bRTL = sal_False; + if ( pFrm ) + { + if ( pFrm->IsVertical() ) + { + bVertLayout = sal_True; + pFnt->SetVertical( pFnt->GetOrientation(), sal_True ); + } + bRTL = pFrm->IsRightToLeft(); + } + + // Initialize the default attribute of the attribute handler + // based on the attribute array cached together with the font. + // If any further attributes for the paragraph are given in pAttrSet + // consider them during construction of the default array, and apply + // them to the font + aAttrHandler.Init( aFontAccess.Get()->GetDefault(), pAttrSet, + *rTxtNode.getIDocumentSettingAccess(), pShell, *pFnt, bVertLayout ); + + aMagicNo[SW_LATIN] = aMagicNo[SW_CJK] = aMagicNo[SW_CTL] = NULL; + + // determine script changes if not already done for current paragraph + OSL_ENSURE( pScriptInfo, "No script info available"); + if ( pScriptInfo->GetInvalidity() != STRING_LEN ) + pScriptInfo->InitScriptInfo( rTxtNode, bRTL ); + + if ( pBreakIt->GetBreakIter().is() ) + { + pFnt->SetActual( SwScriptInfo::WhichFont( 0, 0, pScriptInfo ) ); + + xub_StrLen nChg = 0; + sal_uInt16 nCnt = 0; + + do + { + nChg = pScriptInfo->GetScriptChg( nCnt ); + sal_uInt16 nScript = pScriptInfo->GetScriptType( nCnt++ ); + sal_uInt8 nTmp = 4; + switch ( nScript ) { + case i18n::ScriptType::ASIAN : + if( !aMagicNo[SW_CJK] ) nTmp = SW_CJK; break; + case i18n::ScriptType::COMPLEX : + if( !aMagicNo[SW_CTL] ) nTmp = SW_CTL; break; + default: + if( !aMagicNo[SW_LATIN ] ) nTmp = SW_LATIN; + } + if( nTmp < 4 ) + { + pFnt->ChkMagic( pShell, nTmp ); + pFnt->GetMagic( aMagicNo[ nTmp ], aFntIdx[ nTmp ], nTmp ); + } + } while( nChg < rTxtNode.GetTxt().Len() ); + } + else + { + pFnt->ChkMagic( pShell, SW_LATIN ); + pFnt->GetMagic( aMagicNo[ SW_LATIN ], aFntIdx[ SW_LATIN ], SW_LATIN ); + } + + nStartIndex = nEndIndex = nPos = nChgCnt = 0; + nPropFont = 0; + SwDoc* pDoc = rTxtNode.GetDoc(); + const IDocumentRedlineAccess* pIDRA = rTxtNode.getIDocumentRedlineAccess(); + + const SwExtTextInput* pExtInp = pDoc->GetExtTextInput( rTxtNode ); + const bool bShow = IDocumentRedlineAccess::IsShowChanges( pIDRA->GetRedlineMode() ); + if( pExtInp || bShow ) + { + MSHORT nRedlPos = pIDRA->GetRedlinePos( rTxtNode, USHRT_MAX ); + if( pExtInp || MSHRT_MAX != nRedlPos ) + { + const SvUShorts* pArr = 0; + xub_StrLen nInputStt = 0; + if( pExtInp ) + { + pArr = &pExtInp->GetAttrs(); + nInputStt = pExtInp->Start()->nContent.GetIndex(); + Seek( 0 ); + } + + pRedln = new SwRedlineItr( rTxtNode, *pFnt, aAttrHandler, nRedlPos, + bShow, pArr, nInputStt ); + + if( pRedln->IsOn() ) + ++nChgCnt; + } + } +} + +/************************************************************************* + * SwRedlineItr - Der Redline-Iterator + * + * Folgende Informationen/Zustaende gibt es im RedlineIterator: + * + * nFirst ist der erste Index der RedlineTbl, der mit dem Absatz ueberlappt. + * + * nAct ist der zur Zeit aktive ( wenn bOn gesetzt ist ) oder der naechste + * in Frage kommende Index. + * nStart und nEnd geben die Grenzen des Objekts innerhalb des Absatzes an. + * + * Wenn bOn gesetzt ist, ist der Font entsprechend manipuliert worden. + * + * Wenn nAct auf MSHRT_MAX gesetzt wurde ( durch Reset() ), so ist zur Zeit + * kein Redline aktiv, nStart und nEnd sind invalid. + *************************************************************************/ + +SwRedlineItr::SwRedlineItr( const SwTxtNode& rTxtNd, SwFont& rFnt, + SwAttrHandler& rAH, MSHORT nRed, sal_Bool bShw, const SvUShorts *pArr, + xub_StrLen nExtStart ) + : rDoc( *rTxtNd.GetDoc() ), rNd( rTxtNd ), rAttrHandler( rAH ), pSet( 0 ), + nNdIdx( rTxtNd.GetIndex() ), nFirst( nRed ), + nAct( MSHRT_MAX ), bOn( sal_False ), bShow( bShw ) +{ + if( pArr ) + pExt = new SwExtend( *pArr, nExtStart ); + else + pExt = NULL; + Seek( rFnt, 0, STRING_LEN ); +} + +SwRedlineItr::~SwRedlineItr() +{ + Clear( NULL ); + delete pSet; + delete pExt; +} + +// Der Return-Wert von SwRedlineItr::Seek gibt an, ob der aktuelle Font +// veraendert wurde durch Verlassen (-1) oder Betreten eines Bereichs (+1) + +short SwRedlineItr::_Seek( SwFont& rFnt, xub_StrLen nNew, xub_StrLen nOld ) +{ + short nRet = 0; + if( ExtOn() ) + return 0; // Abkuerzung: wenn wir innerhalb eines ExtendTextInputs sind + // kann es keine anderen Attributwechsel (auch nicht durch Redlining) geben + if( bShow ) + { + if( bOn ) + { + if( nNew >= nEnd ) + { + --nRet; + _Clear( &rFnt ); // Wir gehen hinter den aktuellen Bereich + ++nAct; // und pruefen gleich den naechsten + } + else if( nNew < nStart ) + { + --nRet; + _Clear( &rFnt ); // Wir gehen vor den aktuellen Bereich + if( nAct > nFirst ) + nAct = nFirst; // Die Pruefung muss von vorne beginnen + else + return nRet + EnterExtend( rFnt, nNew ); // Es gibt keinen vor uns. + } + else + return nRet + EnterExtend( rFnt, nNew ); // Wir sind im gleichen Bereich geblieben. + } + if( MSHRT_MAX == nAct || nOld > nNew ) + nAct = nFirst; + + nStart = STRING_LEN; + nEnd = STRING_LEN; + + for( ; nAct < rDoc.GetRedlineTbl().Count() ; ++nAct ) + { + rDoc.GetRedlineTbl()[ nAct ]->CalcStartEnd( nNdIdx, nStart, nEnd ); + + if( nNew < nEnd ) + { + if( nNew >= nStart ) // der einzig moegliche Kandidat + { + bOn = sal_True; + const SwRedline *pRed = rDoc.GetRedlineTbl()[ nAct ]; + + if (pSet) + pSet->ClearItem(); + else + { + SwAttrPool& rPool = + const_cast<SwDoc&>(rDoc).GetAttrPool(); + pSet = new SfxItemSet(rPool, RES_CHRATR_BEGIN, RES_CHRATR_END-1); + } + + if( 1 < pRed->GetStackCount() ) + FillHints( pRed->GetAuthor( 1 ), pRed->GetType( 1 ) ); + FillHints( pRed->GetAuthor(), pRed->GetType() ); + + SfxWhichIter aIter( *pSet ); + MSHORT nWhich = aIter.FirstWhich(); + while( nWhich ) + { + const SfxPoolItem* pItem; + if( ( nWhich < RES_CHRATR_END ) && + ( SFX_ITEM_SET == pSet->GetItemState( nWhich, sal_True, &pItem ) ) ) + { + SwTxtAttr* pAttr = MakeRedlineTxtAttr( + const_cast<SwDoc&>(rDoc), + *const_cast<SfxPoolItem*>(pItem) ); + pAttr->SetPriorityAttr( sal_True ); + aHints.C40_INSERT( SwTxtAttr, pAttr, aHints.Count()); + rAttrHandler.PushAndChg( *pAttr, rFnt ); + if( RES_CHRATR_COLOR == nWhich ) + rFnt.SetNoCol( sal_True ); + } + nWhich = aIter.NextWhich(); + } + + ++nRet; + } + break; + } + nStart = STRING_LEN; + nEnd = STRING_LEN; + } + } + return nRet + EnterExtend( rFnt, nNew ); +} + +void SwRedlineItr::FillHints( MSHORT nAuthor, RedlineType_t eType ) +{ + switch ( eType ) + { + case nsRedlineType_t::REDLINE_INSERT: + SW_MOD()->GetInsertAuthorAttr(nAuthor, *pSet); + break; + case nsRedlineType_t::REDLINE_DELETE: + SW_MOD()->GetDeletedAuthorAttr(nAuthor, *pSet); + break; + case nsRedlineType_t::REDLINE_FORMAT: + case nsRedlineType_t::REDLINE_FMTCOLL: + SW_MOD()->GetFormatAuthorAttr(nAuthor, *pSet); + break; + default: + break; + } +} + +void SwRedlineItr::ChangeTxtAttr( SwFont* pFnt, SwTxtAttr &rHt, sal_Bool bChg ) +{ + OSL_ENSURE( IsOn(), "SwRedlineItr::ChangeTxtAttr: Off?" ); + + if( !bShow && !pExt ) + return; + + if( bChg ) + { + if ( pExt && pExt->IsOn() ) + rAttrHandler.PushAndChg( rHt, *pExt->GetFont() ); + else + rAttrHandler.PushAndChg( rHt, *pFnt ); + } + else + { + OSL_ENSURE( ! pExt || ! pExt->IsOn(), "Pop of attribute during opened extension" ); + rAttrHandler.PopAndChg( rHt, *pFnt ); + } +} + +void SwRedlineItr::_Clear( SwFont* pFnt ) +{ + OSL_ENSURE( bOn, "SwRedlineItr::Clear: Off?" ); + bOn = sal_False; + while( aHints.Count() ) + { + SwTxtAttr *pPos = aHints[ 0 ]; + aHints.Remove(0); + if( pFnt ) + rAttrHandler.PopAndChg( *pPos, *pFnt ); + else + rAttrHandler.Pop( *pPos ); + SwTxtAttr::Destroy(pPos, const_cast<SwDoc&>(rDoc).GetAttrPool() ); + } + if( pFnt ) + pFnt->SetNoCol( sal_False ); +} + +xub_StrLen SwRedlineItr::_GetNextRedln( xub_StrLen nNext ) +{ + nNext = NextExtend( nNext ); + if( !bShow || MSHRT_MAX == nFirst ) + return nNext; + if( MSHRT_MAX == nAct ) + { + nAct = nFirst; + rDoc.GetRedlineTbl()[ nAct ]->CalcStartEnd( nNdIdx, nStart, nEnd ); + } + if( bOn || !nStart ) + { + if( nEnd < nNext ) + nNext = nEnd; + } + else if( nStart < nNext ) + nNext = nStart; + return nNext; +} + +sal_Bool SwRedlineItr::_ChkSpecialUnderline() const +{ + // Wenn die Unterstreichung oder das Escapement vom Redling kommt, + // wenden wir immer das SpecialUnderlining, d.h. die Unterstreichung + // unter der Grundlinie an. + for( MSHORT i = 0; i < aHints.Count(); ++i ) + { + MSHORT nWhich = aHints[i]->Which(); + if( RES_CHRATR_UNDERLINE == nWhich || + RES_CHRATR_ESCAPEMENT == nWhich ) + return sal_True; + } + return sal_False; +} + +sal_Bool SwRedlineItr::CheckLine( xub_StrLen nChkStart, xub_StrLen nChkEnd ) +{ + if( nFirst == MSHRT_MAX ) + return sal_False; + if( nChkEnd == nChkStart ) // Leerzeilen gucken ein Zeichen weiter. + ++nChkEnd; + xub_StrLen nOldStart = nStart; + xub_StrLen nOldEnd = nEnd; + xub_StrLen nOldAct = nAct; + sal_Bool bRet = sal_False; + + for( nAct = nFirst; nAct < rDoc.GetRedlineTbl().Count() ; ++nAct ) + { + rDoc.GetRedlineTbl()[ nAct ]->CalcStartEnd( nNdIdx, nStart, nEnd ); + if( nChkEnd < nStart ) + break; + if( nChkStart <= nEnd && ( nChkEnd > nStart || STRING_LEN == nEnd ) ) + { + bRet = sal_True; + break; + } + } + + nStart = nOldStart; + nEnd = nOldEnd; + nAct = nOldAct; + return bRet; +} + +void SwExtend::ActualizeFont( SwFont &rFnt, MSHORT nAttr ) +{ + if ( nAttr & EXTTEXTINPUT_ATTR_UNDERLINE ) + rFnt.SetUnderline( UNDERLINE_SINGLE ); + else if ( nAttr & EXTTEXTINPUT_ATTR_BOLDUNDERLINE ) + rFnt.SetUnderline( UNDERLINE_BOLD ); + else if ( nAttr & EXTTEXTINPUT_ATTR_DOTTEDUNDERLINE ) + rFnt.SetUnderline( UNDERLINE_DOTTED ); + else if ( nAttr & EXTTEXTINPUT_ATTR_DASHDOTUNDERLINE ) + rFnt.SetUnderline( UNDERLINE_DOTTED ); + + if ( nAttr & EXTTEXTINPUT_ATTR_REDTEXT ) + rFnt.SetColor( Color( COL_RED ) ); + + if ( nAttr & EXTTEXTINPUT_ATTR_HIGHLIGHT ) + { + const StyleSettings& rStyleSettings = GetpApp()->GetSettings().GetStyleSettings(); + rFnt.SetColor( rStyleSettings.GetHighlightTextColor() ); + rFnt.SetBackColor( new Color( rStyleSettings.GetHighlightColor() ) ); + } + if ( nAttr & EXTTEXTINPUT_ATTR_GRAYWAVELINE ) + rFnt.SetGreyWave( sal_True ); +} + +short SwExtend::Enter( SwFont& rFnt, xub_StrLen nNew ) +{ + OSL_ENSURE( !Inside(), "SwExtend: Enter without Leave" ); + OSL_ENSURE( !pFnt, "SwExtend: Enter with Font" ); + nPos = nNew; + if( Inside() ) + { + pFnt = new SwFont( rFnt ); + ActualizeFont( rFnt, rArr[ nPos - nStart ] ); + return 1; + } + return 0; +} + +sal_Bool SwExtend::_Leave( SwFont& rFnt, xub_StrLen nNew ) +{ + OSL_ENSURE( Inside(), "SwExtend: Leave without Enter" ); + MSHORT nOldAttr = rArr[ nPos - nStart ]; + nPos = nNew; + if( Inside() ) + { // Wir sind innerhalb des ExtendText-Bereichs geblieben + MSHORT nAttr = rArr[ nPos - nStart ]; + if( nOldAttr != nAttr ) // Gibt es einen (inneren) Attributwechsel? + { + rFnt = *pFnt; + ActualizeFont( rFnt, nAttr ); + } + } + else + { + rFnt = *pFnt; + delete pFnt; + pFnt = NULL; + return sal_True; + } + return sal_False; +} + +xub_StrLen SwExtend::Next( xub_StrLen nNext ) +{ + if( nPos < nStart ) + { + if( nNext > nStart ) + nNext = nStart; + } + else if( nPos < nEnd ) + { + MSHORT nIdx = nPos - nStart; + MSHORT nAttr = rArr[ nIdx ]; + while( ++nIdx < rArr.Count() && nAttr == rArr[ nIdx ] ) + ; //nothing + nIdx = nIdx + nStart; + if( nNext > nIdx ) + nNext = nIdx; + } + return nNext; +} + +/* vim:set shiftwidth=4 softtabstop=4 expandtab: */ |