/************************************************************************* * * $RCSfile: redlnitr.cxx,v $ * * $Revision: 1.8 $ * * last change: $Author: ama $ $Date: 2001-02-20 10:22:10 $ * * The Contents of this file are made available subject to the terms of * either of the following licenses * * - GNU Lesser General Public License Version 2.1 * - Sun Industry Standards Source License Version 1.1 * * Sun Microsystems Inc., October, 2000 * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2000 by Sun Microsystems, Inc. * 901 San Antonio Road, Palo Alto, CA 94303, USA * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software Foundation. * * This library 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 for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA * * * Sun Industry Standards Source License Version 1.1 * ================================================= * The contents of this file are subject to the Sun Industry Standards * Source License Version 1.1 (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.openoffice.org/license.html. * * Software provided under this License is provided on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, * WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS, * MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING. * See the License for the specific provisions governing your rights and * obligations concerning the Software. * * The Initial Developer of the Original Code is: Sun Microsystems, Inc. * * Copyright: 2000 by Sun Microsystems, Inc. * * All Rights Reserved. * * Contributor(s): _______________________________________ * * ************************************************************************/ #ifdef PRECOMPILED #include "core_pch.hxx" #endif #pragma hdrstop #include "hintids.hxx" #ifndef _SFX_WHITER_HXX //autogen #include #endif #ifndef _SHL_HXX #include #endif #ifndef _COM_SUN_STAR_I18N_SCRIPTTYPE_HDL_ #include #endif #ifndef _SWMODULE_HXX #include #endif #ifndef _REDLINE_HXX #include // SwRedline #endif #ifndef _TXTATR_HXX #include // SwTxt ... #endif #ifndef _DOCARY_HXX #include // SwRedlineTbl #endif #ifndef _ITRATR_HXX #include // SwAttrIter #endif #ifndef _NDTXT_HXX #include // SwTxtNode #endif #ifndef _SWFNTCCH_HXX #include // SwFontAccess #endif #ifndef _DOC_HXX #include // SwDoc #endif #ifndef _TXATBASE_HXX #include // SwTxtAttr #endif #ifndef _ROOTFRM_HXX #include #endif #ifndef _FRMSH_HXX #include #endif #ifndef _BREAKIT_HXX #include #endif ////////////////////////// #ifndef _SV_KEYCODES_HXX //autogen #include #endif #ifndef _VCL_CMDEVT_HXX //autogen #include #endif #ifndef _SV_SETTINGS_HXX //autogen #include #endif #ifndef _APP_HXX //autogen #include #endif #ifndef _REDLNITR_HXX #include #endif #ifndef _EXTINPUT_HXX #include #endif using namespace ::com::sun::star; /************************************************************************* * SwAttrIter::CtorInit() *************************************************************************/ void SwAttrIter::CtorInit( SwTxtNode& rTxtNode, SwScriptInfo* pScrInf ) { // Beim HTML-Import kann es vorkommen, dass kein Layout existiert. SwRootFrm *pRootFrm = rTxtNode.GetDoc()->GetRootFrm(); pShell = pRootFrm ? pRootFrm->GetShell() : 0; pScriptInfo = pScrInf; pAttrSet = &rTxtNode.GetSwAttrSet(); pHints = rTxtNode.GetpSwpHints(); delete pFnt; if ( rTxtNode.HasSwAttrSet() ) { // Hier wird noch ein weiterer Cache eingebaut werden, // der ueber ein paar SfxItemSets sucht. pFnt = new SwFont( pAttrSet ); } else { //FEATURE::CONDCOLL // SwFontAccess aFontAccess( rTxtNode.GetFmtColl() ); SwFontAccess aFontAccess( &rTxtNode.GetAnyFmtColl(), pShell ); //FEATURE::CONDCOLL pFnt = new SwFont( *aFontAccess.Get()->GetFont() ); } aMagicNo[SW_LATIN] = aMagicNo[SW_CJK] = aMagicNo[SW_CTL] = NULL; // determine script changes if not already done for current paragraph ASSERT( pScriptInfo, "No script info available"); if ( pScriptInfo->GetInvalidity() != STRING_LEN ) pScriptInfo->InitScriptInfo( rTxtNode.GetTxt() ); if ( pBreakIt->xBreak.is() ) { USHORT nScript = pScriptInfo->GetScriptType( 0 ); xub_StrLen nChg = 0; USHORT nCnt = 0; if( i18n::ScriptType::WEAK == nScript ) { nChg = pScriptInfo->NextScriptChg( 0 ); if( nChg < rTxtNode.GetTxt().Len() ) nScript = pScriptInfo->GetScriptType( 0 ); } do { nChg = pScriptInfo->GetScriptChg( nCnt ); nScript = pScriptInfo->GetScriptType( nCnt++ ); BYTE 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 SwExtTextInput* pExtInp = pDoc->GetExtTextInput( rTxtNode ); sal_Bool bShow = ::IsShowChanges( pDoc->GetRedlineMode() ); if( pExtInp || bShow || (SwHoriOrient)SW_MOD()->GetRedlineMarkPos() != HORI_NONE ) { MSHORT nRedlPos = pDoc->GetRedlinePos( rTxtNode ); if( pExtInp || MSHRT_MAX != nRedlPos ) { const SvUShorts* pArr = 0; xub_StrLen nInputStt = 0; if( pExtInp ) { pArr = &pExtInp->GetAttrs(); nInputStt = pExtInp->Start()->nContent.GetIndex(); } pRedln = new SwRedlineItr( rTxtNode, *pFnt, 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, MSHORT nRed, sal_Bool bShw, const SvUShorts *pArr, xub_StrLen nExtStart ) : rDoc( *rTxtNd.GetDoc() ), rNd( rTxtNd ), nNdIdx( rTxtNd.GetIndex() ), nFirst( nRed ), nAct( MSHRT_MAX ), bOn( sal_False ), pSet(0), 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; } void SwRedlineItr::CalcStartEnd() { const SwRedline* pTmp = rDoc.GetRedlineTbl()[ nAct ]; const SwPosition *pRStt = pTmp->Start(), *pREnd = pTmp->End(); if( pRStt->nNode < nNdIdx ) { if( pREnd->nNode > nNdIdx ) { nStart = 0; // Absatz ist komplett enthalten nEnd = STRING_LEN; } else { ASSERT( pREnd->nNode == nNdIdx, "SwRedlineItr::Seek: GetRedlinePos Error" ); nStart = 0; // Absatz wird vorne ueberlappt nEnd = pREnd->nContent.GetIndex(); } } else if( pRStt->nNode == nNdIdx ) { nStart = pRStt->nContent.GetIndex(); if( pREnd->nNode == nNdIdx ) nEnd = pREnd->nContent.GetIndex(); // Innerhalb des Absatzes else nEnd = STRING_LEN; // Absatz wird hinten ueberlappt } else { nStart = STRING_LEN; nEnd = STRING_LEN; } } // 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() ) { if( !LeaveExtend( rFnt, nNew ) ) return 0; // Abkuerzung: wenn wir innerhalb eines ExtendTextInputs sind // kann es keine anderen Attributwechsel (auch nicht durch Redlining) geben ++nRet; } 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 ) { CalcStartEnd(); 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 = (SwAttrPool& )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 ); register 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 = ((SwTxtNode&)rNd).MakeTmpTxtAttr(*pItem); aHints.C40_INSERT( SwTxtAttr, pAttr, aHints.Count()); pAttr->ChgFnt( &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, SwRedlineType eType ) { switch ( eType ) { case REDLINE_INSERT: SW_MOD()->GetInsertAuthorAttr(nAuthor, *pSet); break; case REDLINE_DELETE: SW_MOD()->GetDeletedAuthorAttr(nAuthor, *pSet); break; case REDLINE_FORMAT: case REDLINE_FMTCOLL: SW_MOD()->GetFormatAuthorAttr(nAuthor, *pSet); break; } } void SwRedlineItr::_ChangeTxtAttr( SwFont* pFnt, SwTxtAttr &rHt, sal_Bool bChg ) { ASSERT( IsOn(), "SwRedlineItr::ChangeTxtAttr: Off?" ); if( ExtOn() ) pExt->ChangeTxtAttr( rHt, bChg ); MSHORT nWhich = rHt.Which(); MSHORT i; if( RES_TXTATR_CHARFMT == nWhich || RES_TXTATR_INETFMT == nWhich ) { for( i = 0; i < aHints.Count(); ++i ) aHints[ i ]->RstFnt( pFnt ); if( bChg ) rHt.ChgFnt( pFnt ); else rHt.RstFnt( pFnt ); for( i = 0; i < aHints.Count(); ++i ) aHints[ i ]->ChgFnt( pFnt ); } else { for( i = 0; i < aHints.Count(); ++i ) { SwTxtAttr *pPos = aHints[ i ]; if( pPos->Which() == nWhich ) { if( bChg ) rHt.ChgTxtAttr( *pPos ); else rHt.RstTxtAttr( *pPos ); return; } } if( bChg ) rHt.ChgFnt( pFnt ); else rHt.RstFnt( pFnt ); } } void SwRedlineItr::_Clear( SwFont* pFnt ) { ASSERT( bOn, "SwRedlineItr::Clear: Off?" ); bOn = sal_False; while( aHints.Count() ) { SwTxtAttr *pPos = aHints[ 0 ]; aHints.Remove(0); if( pFnt ) pPos->RstFnt( pFnt ); delete pPos; } 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; CalcStartEnd(); } 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 ) { SwTxtAttr *pPos = aHints[ 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 ) { CalcStartEnd(); 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.SetFillColor( rStyleSettings.GetHighlightColor() ); rFnt.SetTransparent( sal_False ); } if ( nAttr & EXTTEXTINPUT_ATTR_GRAYWAVELINE ) rFnt.SetGreyWave( sal_True ); } short SwExtend::Enter( SwFont& rFnt, xub_StrLen nNew ) { ASSERT( !Inside(), "SwExtend: Enter without Leave" ); ASSERT( !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 ) { ASSERT( 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 += nStart; if( nNext > nIdx ) nNext = nIdx; } return nNext; } void SwExtend::ChangeTxtAttr( SwTxtAttr &rHt, sal_Bool bChg ) { ASSERT( pFnt, "SwExtend: No font, no fun." ); if( bChg ) rHt.ChgFnt( pFnt ); else rHt.RstFnt( pFnt ); }