/* -*- 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 "hintids.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; void SwAttrIter::CtorInitAttrIter( SwTextNode& rTextNode, SwScriptInfo& rScrInf, SwTextFrame* pFrame ) { // during HTML-Import it can happen, that no layout exists SwRootFrame* pRootFrame = rTextNode.getIDocumentLayoutAccess().GetCurrentLayout(); pShell = pRootFrame ? pRootFrame->GetCurrShell() : nullptr; pScriptInfo = &rScrInf; // attributes set at the whole paragraph pAttrSet = rTextNode.GetpSwAttrSet(); // attribute array pHints = rTextNode.GetpSwpHints(); // Build a font matching the default paragraph style: SwFontAccess aFontAccess( &rTextNode.GetAnyFormatColl(), pShell ); delete pFnt; pFnt = new SwFont( aFontAccess.Get()->GetFont() ); // set font to vertical if frame layout is vertical bool bVertLayout = false; bool bRTL = false; if ( pFrame ) { if ( pFrame->IsVertical() ) { bVertLayout = true; pFnt->SetVertical( pFnt->GetOrientation(), true ); } bRTL = pFrame->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, *rTextNode.getIDocumentSettingAccess(), pShell, *pFnt, bVertLayout ); aMagicNo[SW_LATIN] = aMagicNo[SW_CJK] = aMagicNo[SW_CTL] = nullptr; // determine script changes if not already done for current paragraph OSL_ENSURE( pScriptInfo, "No script info available"); if ( pScriptInfo->GetInvalidityA() != COMPLETE_STRING ) pScriptInfo->InitScriptInfo( rTextNode, bRTL ); if ( g_pBreakIt->GetBreakIter().is() ) { pFnt->SetActual( SwScriptInfo::WhichFont( 0, nullptr, pScriptInfo ) ); sal_Int32 nChg = 0; size_t nCnt = 0; do { if ( nCnt >= pScriptInfo->CountScriptChg() ) break; nChg = pScriptInfo->GetScriptChg( nCnt ); int nTmp = SW_SCRIPTS; switch ( pScriptInfo->GetScriptType( nCnt++ ) ) { 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 < SW_SCRIPTS ) { pFnt->ChkMagic( pShell, nTmp ); pFnt->GetMagic( aMagicNo[ nTmp ], aFntIdx[ nTmp ], nTmp ); } } while (nChg < rTextNode.GetText().getLength()); } 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 = rTextNode.GetDoc(); const IDocumentRedlineAccess& rIDRA = rTextNode.getIDocumentRedlineAccess(); const SwExtTextInput* pExtInp = pDoc->GetExtTextInput( rTextNode ); const bool bShow = IDocumentRedlineAccess::IsShowChanges( rIDRA.GetRedlineMode() ); if( pExtInp || bShow ) { const sal_uInt16 nRedlPos = rIDRA.GetRedlinePos( rTextNode, USHRT_MAX ); if( pExtInp || USHRT_MAX != nRedlPos ) { const std::vector *pArr = nullptr; sal_Int32 nInputStt = 0; if( pExtInp ) { pArr = &pExtInp->GetAttrs(); nInputStt = pExtInp->Start()->nContent.GetIndex(); Seek( 0 ); } pRedln = new SwRedlineItr( rTextNode, *pFnt, aAttrHandler, nRedlPos, bShow, pArr, nInputStt ); if( pRedln->IsOn() ) ++nChgCnt; } } } // The Redline-Iterator // The following information/states exist in RedlineIterator: // // nFirst is the first index of RedlineTable, which overlaps with the paragraph. // // nAct is the currently active (if bOn is set) or the next possible index. // nStart and nEnd give you the borders of the object within the paragraph. // // If bOn is set, the font has been manipulated according to it. // // If nAct is set to COMPLETE_STRING (via Reset()), then currently no // Redline is active, nStart and nEnd are invalid. SwRedlineItr::SwRedlineItr( const SwTextNode& rTextNd, SwFont& rFnt, SwAttrHandler& rAH, sal_Int32 nRed, bool bShw, const std::vector *pArr, sal_Int32 nExtStart ) : rDoc( *rTextNd.GetDoc() ), rAttrHandler( rAH ), pSet( nullptr ), nNdIdx( rTextNd.GetIndex() ), nFirst( nRed ), nAct( COMPLETE_STRING ), bOn( false ), bShow( bShw ) { if( pArr ) pExt = new SwExtend( *pArr, nExtStart ); else pExt = nullptr; Seek (rFnt, 0, COMPLETE_STRING); } SwRedlineItr::~SwRedlineItr() { Clear( nullptr ); delete pSet; delete pExt; } // The return value of SwRedlineItr::Seek tells you if the current font // has been manipulated by leaving (-1) or accessing (+1) of a section short SwRedlineItr::_Seek(SwFont& rFnt, sal_Int32 nNew, sal_Int32 nOld) { short nRet = 0; if( ExtOn() ) return 0; // Abbreviation: if we're within an ExtendTextInputs // there can't be other changes of attributes (not even by redlining) if( bShow ) { if( bOn ) { if( nNew >= nEnd ) { --nRet; _Clear( &rFnt ); // We go behind the current section ++nAct; // and check the next one } else if( nNew < nStart ) { --nRet; _Clear( &rFnt ); // We go in front of the current section if( nAct > nFirst ) nAct = nFirst; // the test has to run from the beginning else return nRet + EnterExtend( rFnt, nNew ); // There's none prior to us } else return nRet + EnterExtend( rFnt, nNew ); // We stayed in the same section } if( COMPLETE_STRING == nAct || nOld > nNew ) nAct = nFirst; nStart = COMPLETE_STRING; nEnd = COMPLETE_STRING; for( ; nAct < (sal_Int32)rDoc.getIDocumentRedlineAccess().GetRedlineTable().size() ; ++nAct ) { rDoc.getIDocumentRedlineAccess().GetRedlineTable()[ nAct ]->CalcStartEnd( nNdIdx, nStart, nEnd ); if( nNew < nEnd ) { if( nNew >= nStart ) // der einzig moegliche Kandidat { bOn = true; const SwRangeRedline *pRed = rDoc.getIDocumentRedlineAccess().GetRedlineTable()[ nAct ]; if (pSet) pSet->ClearItem(); else { SwAttrPool& rPool = const_cast(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 ); sal_uInt16 nWhich = aIter.FirstWhich(); while( nWhich ) { const SfxPoolItem* pItem; if( ( nWhich < RES_CHRATR_END ) && ( SfxItemState::SET == pSet->GetItemState( nWhich, true, &pItem ) ) ) { SwTextAttr* pAttr = MakeRedlineTextAttr( const_cast(rDoc), *const_cast(pItem) ); pAttr->SetPriorityAttr( true ); m_Hints.push_back(pAttr); rAttrHandler.PushAndChg( *pAttr, rFnt ); if( RES_CHRATR_COLOR == nWhich ) rFnt.SetNoCol( true ); } nWhich = aIter.NextWhich(); } ++nRet; } break; } nStart = COMPLETE_STRING; nEnd = COMPLETE_STRING; } } return nRet + EnterExtend( rFnt, nNew ); } void SwRedlineItr::FillHints( sal_uInt16 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::ChangeTextAttr( SwFont* pFnt, SwTextAttr &rHt, bool bChg ) { OSL_ENSURE( IsOn(), "SwRedlineItr::ChangeTextAttr: 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 = false; while (!m_Hints.empty()) { SwTextAttr *pPos = m_Hints.front(); m_Hints.pop_front(); if( pFnt ) rAttrHandler.PopAndChg( *pPos, *pFnt ); else rAttrHandler.Pop( *pPos ); SwTextAttr::Destroy(pPos, const_cast(rDoc).GetAttrPool() ); } if( pFnt ) pFnt->SetNoCol( false ); } sal_Int32 SwRedlineItr::_GetNextRedln( sal_Int32 nNext ) { nNext = NextExtend( nNext ); if( !bShow || COMPLETE_STRING == nFirst ) return nNext; if( COMPLETE_STRING == nAct ) { nAct = nFirst; rDoc.getIDocumentRedlineAccess().GetRedlineTable()[ nAct ]->CalcStartEnd( nNdIdx, nStart, nEnd ); } if( bOn || !nStart ) { if( nEnd < nNext ) nNext = nEnd; } else if( nStart < nNext ) nNext = nStart; return nNext; } bool SwRedlineItr::_ChkSpecialUnderline() const { // If the underlining or the escapement is caused by redlining, // we always apply the SpecialUnderlining, i.e. the underlining // below the base line for (size_t i = 0; i < m_Hints.size(); ++i) { const sal_uInt16 nWhich = m_Hints[i]->Which(); if( RES_CHRATR_UNDERLINE == nWhich || RES_CHRATR_ESCAPEMENT == nWhich ) return true; } return false; } bool SwRedlineItr::CheckLine( sal_Int32 nChkStart, sal_Int32 nChkEnd ) { if( nFirst == COMPLETE_STRING ) return false; if( nChkEnd == nChkStart ) // empty lines look one char further ++nChkEnd; sal_Int32 nOldStart = nStart; sal_Int32 nOldEnd = nEnd; sal_Int32 nOldAct = nAct; bool bRet = false; for( nAct = nFirst; nAct < (sal_Int32)rDoc.getIDocumentRedlineAccess().GetRedlineTable().size() ; ++nAct ) { rDoc.getIDocumentRedlineAccess().GetRedlineTable()[ nAct ]->CalcStartEnd( nNdIdx, nStart, nEnd ); if( nChkEnd < nStart ) break; if( nChkStart <= nEnd && ( nChkEnd > nStart || COMPLETE_STRING == nEnd ) ) { bRet = true; break; } } nStart = nOldStart; nEnd = nOldEnd; nAct = nOldAct; return bRet; } void SwExtend::ActualizeFont( SwFont &rFnt, sal_uInt16 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 = Application::GetSettings().GetStyleSettings(); rFnt.SetColor( rStyleSettings.GetHighlightTextColor() ); rFnt.SetBackColor( new Color( rStyleSettings.GetHighlightColor() ) ); } if ( nAttr & EXTTEXTINPUT_ATTR_GRAYWAVELINE ) rFnt.SetGreyWave( true ); } short SwExtend::Enter(SwFont& rFnt, sal_Int32 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; } bool SwExtend::_Leave(SwFont& rFnt, sal_Int32 nNew) { OSL_ENSURE( Inside(), "SwExtend: Leave without Enter" ); const sal_uInt16 nOldAttr = rArr[ nPos - nStart ]; nPos = nNew; if( Inside() ) { // We stayed within the ExtendText-section const sal_uInt16 nAttr = rArr[ nPos - nStart ]; if( nOldAttr != nAttr ) // Is there an (inner) change of attributes? { rFnt = *pFnt; ActualizeFont( rFnt, nAttr ); } } else { rFnt = *pFnt; delete pFnt; pFnt = nullptr; return true; } return false; } sal_Int32 SwExtend::Next( sal_Int32 nNext ) { if( nPos < nStart ) { if( nNext > nStart ) nNext = nStart; } else if( nPos < nEnd ) { sal_Int32 nIdx = nPos - nStart; const sal_uInt16 nAttr = rArr[ nIdx ]; while( static_cast(++nIdx) < rArr.size() && nAttr == rArr[ nIdx ] ) ; //nothing nIdx = nIdx + nStart; if( nNext > nIdx ) nNext = nIdx; } return nNext; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */