/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; SwXTextMarkup::SwXTextMarkup( SwTxtNode *const pTxtNode, const ModelToViewHelper& rMap) : mpTxtNode(pTxtNode) , maConversionMap(rMap) { // FME 2007-07-16 #i79641# SwXTextMarkup is allowed to be removed ... SetIsAllowedToBeRemovedInModifyCall(true); mpTxtNode->Add(this); } SwXTextMarkup::~SwXTextMarkup() { } uno::Reference< container::XStringKeyMap > SAL_CALL SwXTextMarkup::getMarkupInfoContainer() throw (uno::RuntimeException, std::exception) { SolarMutexGuard aGuard; uno::Reference< container::XStringKeyMap > xProp = new SwXStringKeyMap; return xProp; } void SAL_CALL SwXTextMarkup::commitTextRangeMarkup(::sal_Int32 nType, const ::rtl::OUString & aIdentifier, const uno::Reference< text::XTextRange> & xRange, const uno::Reference< container::XStringKeyMap > & xMarkupInfoContainer) throw (uno::RuntimeException, std::exception) { SolarMutexGuard aGuard; uno::Reference xRangeTunnel( xRange, uno::UNO_QUERY); if(!xRangeTunnel.is()) return; SwXTextRange* pRange = 0; OTextCursorHelper* pCursor = 0; if(xRangeTunnel.is()) { pRange = reinterpret_cast( sal::static_int_cast< sal_IntPtr >( xRangeTunnel->getSomething(SwXTextRange::getUnoTunnelId()))); pCursor = reinterpret_cast( sal::static_int_cast< sal_IntPtr >( xRangeTunnel->getSomething(OTextCursorHelper::getUnoTunnelId()))); } if (pRange) { SwDoc* pDoc = reinterpret_cast(pRange->GetDoc()); if (!pDoc) return; SwUnoInternalPaM aPam(*pDoc); ::sw::XTextRangeToSwPaM(aPam, xRange); SwPosition* startPos = aPam.Start(); SwPosition* endPos = aPam.End(); commitStringMarkup (nType, aIdentifier, startPos->nContent.GetIndex(), endPos->nContent.GetIndex() - startPos->nContent.GetIndex(), xMarkupInfoContainer); } else if (pCursor) { SwPaM aPam(*pCursor->GetPaM()); SwPosition* startPos = aPam.Start(); SwPosition* endPos = aPam.End(); commitStringMarkup (nType, aIdentifier, startPos->nContent.GetIndex(), endPos->nContent.GetIndex() - startPos->nContent.GetIndex(), xMarkupInfoContainer); } } void SAL_CALL SwXTextMarkup::commitStringMarkup( ::sal_Int32 nType, const OUString & rIdentifier, ::sal_Int32 nStart, ::sal_Int32 nLength, const uno::Reference< container::XStringKeyMap > & xMarkupInfoContainer) throw (uno::RuntimeException, std::exception) { SolarMutexGuard aGuard; // paragraph already dead or modified? if ( !mpTxtNode || nLength <= 0 ) return; if ( nType == text::TextMarkupType::SMARTTAG && !SwSmartTagMgr::Get().IsSmartTagTypeEnabled( rIdentifier ) ) return; // get appropriate list to use... SwWrongList* pWList = 0; bool bRepaint = false; if ( nType == text::TextMarkupType::SPELLCHECK ) { pWList = mpTxtNode->GetWrong(); if ( !pWList ) { pWList = new SwWrongList( WRONGLIST_SPELL ); mpTxtNode->SetWrong( pWList ); } } else if ( nType == text::TextMarkupType::PROOFREADING || nType == text::TextMarkupType::SENTENCE ) { IGrammarContact *pGrammarContact = getGrammarContact( *mpTxtNode ); if( pGrammarContact ) { pWList = pGrammarContact->getGrammarCheck( *mpTxtNode, true ); OSL_ENSURE( pWList, "GrammarContact _has_ to deliver a wrong list" ); } else { pWList = mpTxtNode->GetGrammarCheck(); if ( !pWList ) { mpTxtNode->SetGrammarCheck( new SwGrammarMarkUp() ); pWList = mpTxtNode->GetGrammarCheck(); } } bRepaint = pWList == mpTxtNode->GetGrammarCheck(); if( pWList->GetBeginInv() < COMPLETE_STRING ) ((SwGrammarMarkUp*)pWList)->ClearGrammarList(); } else if ( nType == text::TextMarkupType::SMARTTAG ) { pWList = mpTxtNode->GetSmartTags(); if ( !pWList ) { pWList = new SwWrongList( WRONGLIST_SMARTTAG ); mpTxtNode->SetSmartTags( pWList ); } } else { OSL_FAIL( "Unknown mark-up type" ); return; } const ModelToViewHelper::ModelPosition aStartPos = maConversionMap.ConvertToModelPosition( nStart ); const ModelToViewHelper::ModelPosition aEndPos = maConversionMap.ConvertToModelPosition( nStart + nLength - 1); const bool bStartInField = aStartPos.mbIsField; const bool bEndInField = aEndPos.mbIsField; bool bCommit = false; if ( bStartInField && bEndInField && aStartPos.mnPos == aEndPos.mnPos ) { nStart = aStartPos.mnSubPos; const sal_Int32 nFieldPosModel = aStartPos.mnPos; const sal_uInt16 nInsertPos = pWList->GetWrongPos( nFieldPosModel ); SwWrongList* pSubList = pWList->SubList( nInsertPos ); if ( !pSubList ) { if( nType == text::TextMarkupType::PROOFREADING || nType == text::TextMarkupType::SENTENCE ) pSubList = new SwGrammarMarkUp(); else pSubList = new SwWrongList( pWList->GetWrongListType() ); pWList->InsertSubList( nFieldPosModel, 1, nInsertPos, pSubList ); } pWList = pSubList; bCommit = true; } else if ( !bStartInField && !bEndInField ) { nStart = aStartPos.mnPos; bCommit = true; nLength = aEndPos.mnPos + 1 - aStartPos.mnPos; } else if( nType == text::TextMarkupType::PROOFREADING || nType == text::TextMarkupType::SENTENCE ) { bCommit = true; nStart = aStartPos.mnPos; sal_Int32 nEnd = aEndPos.mnPos; if( bStartInField && nType != text::TextMarkupType::SENTENCE ) { const sal_Int32 nFieldPosModel = aStartPos.mnPos; const sal_uInt16 nInsertPos = pWList->GetWrongPos( nFieldPosModel ); SwWrongList* pSubList = pWList->SubList( nInsertPos ); if ( !pSubList ) { pSubList = new SwGrammarMarkUp(); pWList->InsertSubList( nFieldPosModel, 1, nInsertPos, pSubList ); } const sal_Int32 nTmpStart = maConversionMap.ConvertToViewPosition( aStartPos.mnPos ); const sal_Int32 nTmpLen = maConversionMap.ConvertToViewPosition( aStartPos.mnPos + 1 ) - nTmpStart - aStartPos.mnSubPos; if( nTmpLen > 0 ) { pSubList->Insert( rIdentifier, xMarkupInfoContainer, aStartPos.mnSubPos, nTmpLen ); } ++nStart; } if( bEndInField && nType != text::TextMarkupType::SENTENCE ) { const sal_Int32 nFieldPosModel = aEndPos.mnPos; const sal_uInt16 nInsertPos = pWList->GetWrongPos( nFieldPosModel ); SwWrongList* pSubList = pWList->SubList( nInsertPos ); if ( !pSubList ) { pSubList = new SwGrammarMarkUp(); pWList->InsertSubList( nFieldPosModel, 1, nInsertPos, pSubList ); } const sal_Int32 nTmpLen = aEndPos.mnSubPos + 1; pSubList->Insert( rIdentifier, xMarkupInfoContainer, 0, nTmpLen ); } else ++nEnd; if( nEnd > nStart ) nLength = nEnd - nStart; else bCommit = false; } if ( bCommit ) { if( nType == text::TextMarkupType::SENTENCE ) ((SwGrammarMarkUp*)pWList)->setSentence( nStart ); else pWList->Insert( rIdentifier, xMarkupInfoContainer, nStart, nLength ); } if( bRepaint ) finishGrammarCheck( *mpTxtNode ); } static void lcl_commitGrammarMarkUp( const ModelToViewHelper& rConversionMap, SwGrammarMarkUp* pWList, ::sal_Int32 nType, const OUString & rIdentifier, ::sal_Int32 nStart, ::sal_Int32 nLength, const uno::Reference< container::XStringKeyMap > & xMarkupInfoContainer) { OSL_ENSURE( nType == text::TextMarkupType::PROOFREADING || nType == text::TextMarkupType::SENTENCE, "Wrong mark-up type" ); const ModelToViewHelper::ModelPosition aStartPos = rConversionMap.ConvertToModelPosition( nStart ); const ModelToViewHelper::ModelPosition aEndPos = rConversionMap.ConvertToModelPosition( nStart + nLength - 1); const bool bStartInField = aStartPos.mbIsField; const bool bEndInField = aEndPos.mbIsField; bool bCommit = false; if ( bStartInField && bEndInField && aStartPos.mnPos == aEndPos.mnPos ) { nStart = aStartPos.mnSubPos; const sal_Int32 nFieldPosModel = aStartPos.mnPos; const sal_uInt16 nInsertPos = pWList->GetWrongPos( nFieldPosModel ); SwGrammarMarkUp* pSubList = (SwGrammarMarkUp*)pWList->SubList( nInsertPos ); if ( !pSubList ) { pSubList = new SwGrammarMarkUp(); pWList->InsertSubList( nFieldPosModel, 1, nInsertPos, pSubList ); } pWList = pSubList; bCommit = true; } else if ( !bStartInField && !bEndInField ) { nStart = aStartPos.mnPos; bCommit = true; nLength = aEndPos.mnPos + 1 - aStartPos.mnPos; } else { bCommit = true; nStart = aStartPos.mnPos; sal_Int32 nEnd = aEndPos.mnPos; if( bStartInField && nType != text::TextMarkupType::SENTENCE ) { const sal_Int32 nFieldPosModel = aStartPos.mnPos; const sal_uInt16 nInsertPos = pWList->GetWrongPos( nFieldPosModel ); SwGrammarMarkUp* pSubList = (SwGrammarMarkUp*)pWList->SubList( nInsertPos ); if ( !pSubList ) { pSubList = new SwGrammarMarkUp(); pWList->InsertSubList( nFieldPosModel, 1, nInsertPos, pSubList ); } const sal_Int32 nTmpStart = rConversionMap.ConvertToViewPosition( aStartPos.mnPos ); const sal_Int32 nTmpLen = rConversionMap.ConvertToViewPosition( aStartPos.mnPos + 1 ) - nTmpStart - aStartPos.mnSubPos; if( nTmpLen > 0 ) pSubList->Insert( rIdentifier, xMarkupInfoContainer, aStartPos.mnSubPos, nTmpLen ); ++nStart; } if( bEndInField && nType != text::TextMarkupType::SENTENCE ) { const sal_Int32 nFieldPosModel = aEndPos.mnPos; const sal_uInt16 nInsertPos = pWList->GetWrongPos( nFieldPosModel ); SwGrammarMarkUp* pSubList = (SwGrammarMarkUp*)pWList->SubList( nInsertPos ); if ( !pSubList ) { pSubList = new SwGrammarMarkUp(); pWList->InsertSubList( nFieldPosModel, 1, nInsertPos, pSubList ); } const sal_Int32 nTmpLen = aEndPos.mnSubPos + 1; pSubList->Insert( rIdentifier, xMarkupInfoContainer, 0, nTmpLen ); } else ++nEnd; if( nEnd > nStart ) nLength = nEnd - nStart; else bCommit = false; } if ( bCommit ) { if( nType == text::TextMarkupType::SENTENCE ) ((SwGrammarMarkUp*)pWList)->setSentence( nStart+nLength ); else pWList->Insert( rIdentifier, xMarkupInfoContainer, nStart, nLength ); } } void SAL_CALL SwXTextMarkup::commitMultiTextMarkup( const uno::Sequence< text::TextMarkupDescriptor > &rMarkups ) throw (lang::IllegalArgumentException, uno::RuntimeException, std::exception) { SolarMutexGuard aGuard; // paragraph already dead or modified? if ( !mpTxtNode ) return; // check for equal length of all sequnces sal_Int32 nLen = rMarkups.getLength(); // for grammar checking there should be exactly one sentence markup // and 0..n grammar markups. // Different markups are not expected but may be applied anyway since // that should be no problem... // but it has to be implemented, at the moment only this function is for // grammar markups and sentence markup only! sal_Int32 nSentenceMarkUpIndex = -1; const text::TextMarkupDescriptor *pMarkups = rMarkups.getConstArray(); sal_Int32 i; for( i = 0; i < nLen; ++i ) { if (pMarkups[i].nType == text::TextMarkupType::SENTENCE) { if (nSentenceMarkUpIndex == -1) nSentenceMarkUpIndex = i; else // there is already one sentence markup throw lang::IllegalArgumentException(); } else if( pMarkups[i].nType != text::TextMarkupType::PROOFREADING ) return; } if( nSentenceMarkUpIndex == -1 ) return; // get appropriate list to use... SwGrammarMarkUp* pWList = 0; bool bRepaint = false; IGrammarContact *pGrammarContact = getGrammarContact( *mpTxtNode ); if( pGrammarContact ) { pWList = pGrammarContact->getGrammarCheck( *mpTxtNode, true ); OSL_ENSURE( pWList, "GrammarContact _has_ to deliver a wrong list" ); } else { pWList = mpTxtNode->GetGrammarCheck(); if ( !pWList ) { mpTxtNode->SetGrammarCheck( new SwGrammarMarkUp() ); pWList = mpTxtNode->GetGrammarCheck(); pWList->SetInvalid( 0, COMPLETE_STRING ); } } bRepaint = pWList == mpTxtNode->GetGrammarCheck(); bool bAcceptGrammarError = false; if( pWList->GetBeginInv() < COMPLETE_STRING ) { const ModelToViewHelper::ModelPosition aSentenceEnd = maConversionMap.ConvertToModelPosition( pMarkups[nSentenceMarkUpIndex].nOffset + pMarkups[nSentenceMarkUpIndex].nLength ); bAcceptGrammarError = aSentenceEnd.mnPos > pWList->GetBeginInv(); pWList->ClearGrammarList( aSentenceEnd.mnPos ); } if( bAcceptGrammarError ) { for( i = 0; i < nLen; ++i ) { const text::TextMarkupDescriptor &rDesc = pMarkups[i]; lcl_commitGrammarMarkUp( maConversionMap, pWList, rDesc.nType, rDesc.aIdentifier, rDesc.nOffset, rDesc.nLength, rDesc.xMarkupInfoContainer ); } } else { bRepaint = false; i = nSentenceMarkUpIndex; const text::TextMarkupDescriptor &rDesc = pMarkups[i]; lcl_commitGrammarMarkUp( maConversionMap, pWList, rDesc.nType, rDesc.aIdentifier, rDesc.nOffset, rDesc.nLength, rDesc.xMarkupInfoContainer ); } if( bRepaint ) finishGrammarCheck( *mpTxtNode ); return; } void SwXTextMarkup::Modify( const SfxPoolItem* /*pOld*/, const SfxPoolItem* /*pNew*/ ) { // FME 2007-07-16 #i79641# In my opinion this is perfectly legal, // therefore I remove the assertion in SwModify::_Remove() if ( GetRegisteredIn() ) GetRegisteredInNonConst()->Remove( this ); SolarMutexGuard aGuard; mpTxtNode = 0; } SwXStringKeyMap::SwXStringKeyMap() { } uno::Any SAL_CALL SwXStringKeyMap::getValue(const OUString & aKey) throw (uno::RuntimeException, container::NoSuchElementException, std::exception) { std::map< OUString, uno::Any >::const_iterator aIter = maMap.find( aKey ); if ( aIter == maMap.end() ) throw container::NoSuchElementException(); return (*aIter).second; } sal_Bool SAL_CALL SwXStringKeyMap::hasValue(const OUString & aKey) throw (uno::RuntimeException, std::exception) { return maMap.find( aKey ) != maMap.end(); } void SAL_CALL SwXStringKeyMap::insertValue(const OUString & aKey, const uno::Any & aValue) throw (uno::RuntimeException, lang::IllegalArgumentException, container::ElementExistException, std::exception) { std::map< OUString, uno::Any >::const_iterator aIter = maMap.find( aKey ); if ( aIter != maMap.end() ) throw container::ElementExistException(); maMap[ aKey ] = aValue; } ::sal_Int32 SAL_CALL SwXStringKeyMap::getCount() throw (uno::RuntimeException, std::exception) { return maMap.size(); } OUString SAL_CALL SwXStringKeyMap::getKeyByIndex(::sal_Int32 nIndex) throw (uno::RuntimeException, lang::IndexOutOfBoundsException, std::exception) { if ( (sal_uInt32)nIndex >= maMap.size() ) throw lang::IndexOutOfBoundsException(); return OUString(); } uno::Any SAL_CALL SwXStringKeyMap::getValueByIndex(::sal_Int32 nIndex) throw (uno::RuntimeException, lang::IndexOutOfBoundsException, std::exception) { if ( (sal_uInt32)nIndex >= maMap.size() ) throw lang::IndexOutOfBoundsException(); return uno::Any(); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */