LCOV - code coverage report
Current view: top level - sw/source/core/fields - ddefld.cxx (source / functions) Hit Total Coverage
Test: commit c8344322a7af75b84dd3ca8f78b05543a976dfd5 Lines: 1 209 0.5 %
Date: 2015-06-13 12:38:46 Functions: 2 28 7.1 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include <osl/thread.h>
      21             : #include <sfx2/linkmgr.hxx>
      22             : #include <doc.hxx>
      23             : #include <IDocumentLinksAdministration.hxx>
      24             : #include <IDocumentState.hxx>
      25             : #include <IDocumentLayoutAccess.hxx>
      26             : #include <editsh.hxx>
      27             : #include <ndtxt.hxx>
      28             : #include <fmtfld.hxx>
      29             : #include <txtfld.hxx>
      30             : #include <ddefld.hxx>
      31             : #include <swtable.hxx>
      32             : #include <swbaslnk.hxx>
      33             : #include <swddetbl.hxx>
      34             : #include <unofldmid.h>
      35             : #include <hints.hxx>
      36             : #include <calbck.hxx>
      37             : 
      38             : using namespace ::com::sun::star;
      39             : 
      40             : #define DDE_TXT_ENCODING    osl_getThreadTextEncoding()
      41             : 
      42           0 : class SwIntrnlRefLink : public SwBaseLink
      43             : {
      44             :     SwDDEFieldType& rFieldType;
      45             : public:
      46           0 :     SwIntrnlRefLink( SwDDEFieldType& rType, SfxLinkUpdateMode nUpdateType, SotClipboardFormatId nFormat )
      47             :         : SwBaseLink( nUpdateType, nFormat ),
      48           0 :         rFieldType( rType )
      49           0 :     {}
      50             : 
      51             :     virtual void Closed() SAL_OVERRIDE;
      52             :     virtual ::sfx2::SvBaseLink::UpdateResult DataChanged(
      53             :         const OUString& rMimeType, const ::com::sun::star::uno::Any & rValue ) SAL_OVERRIDE;
      54             : 
      55             :     virtual const SwNode* GetAnchor() const SAL_OVERRIDE;
      56             :     virtual bool IsInRange( sal_uLong nSttNd, sal_uLong nEndNd, sal_Int32 nStt = 0,
      57             :                             sal_Int32 nEnd = -1 ) const SAL_OVERRIDE;
      58             : };
      59             : 
      60           0 : ::sfx2::SvBaseLink::UpdateResult SwIntrnlRefLink::DataChanged( const OUString& rMimeType,
      61             :                                 const uno::Any & rValue )
      62             : {
      63           0 :     switch( SotExchange::GetFormatIdFromMimeType( rMimeType ) )
      64             :     {
      65             :     case SotClipboardFormatId::STRING:
      66           0 :         if( !IsNoDataFlag() )
      67             :         {
      68           0 :             uno::Sequence< sal_Int8 > aSeq;
      69           0 :             rValue >>= aSeq;
      70           0 :             OUString sStr( reinterpret_cast<char const *>(aSeq.getConstArray()), aSeq.getLength(), DDE_TXT_ENCODING );
      71             : 
      72             :             // remove not needed CR-LF at the end
      73           0 :             sal_Int32 n = sStr.getLength();
      74           0 :             while( n && 0 == sStr[ n-1 ] )
      75           0 :                 --n;
      76           0 :             if( n && 0x0a == sStr[ n-1 ] )
      77           0 :                 --n;
      78           0 :             if( n && 0x0d == sStr[ n-1 ] )
      79           0 :                 --n;
      80             : 
      81           0 :             bool bDel = n != sStr.getLength();
      82           0 :             if( bDel )
      83           0 :                 sStr = sStr.copy( 0, n );
      84             : 
      85           0 :             rFieldType.SetExpansion( sStr );
      86             :             // set Expansion first! (otherwise this flag will be deleted)
      87           0 :             rFieldType.SetCRLFDelFlag( bDel );
      88             :         }
      89           0 :         break;
      90             : 
      91             :     // other formats
      92             :     default:
      93           0 :         return SUCCESS;
      94             :     }
      95             : 
      96             :     OSL_ENSURE( rFieldType.GetDoc(), "no pDoc" );
      97             : 
      98             :     // no dependencies left?
      99           0 :     if( rFieldType.HasWriterListeners() && !rFieldType.IsModifyLocked() && !ChkNoDataFlag() )
     100             :     {
     101           0 :         SwViewShell* pSh = rFieldType.GetDoc()->getIDocumentLayoutAccess().GetCurrentViewShell();
     102           0 :         SwEditShell* pESh = rFieldType.GetDoc()->GetEditShell();
     103             : 
     104             :         // Search for fields. If no valid found, disconnect.
     105           0 :         SwMsgPoolItem aUpdateDDE( RES_UPDATEDDETBL );
     106           0 :         bool bCallModify = false;
     107           0 :         rFieldType.LockModify();
     108             : 
     109           0 :         SwIterator<SwClient,SwFieldType> aIter(rFieldType);
     110           0 :         for(SwClient* pLast = aIter.First(); pLast; pLast = aIter.Next())
     111             :         {
     112             :             // a DDE table or a DDE field attribute in the text
     113           0 :             if( !pLast->IsA( TYPE( SwFormatField ) ) ||
     114           0 :                 static_cast<SwFormatField*>(pLast)->GetTextField() )
     115             :             {
     116           0 :                 if( !bCallModify )
     117             :                 {
     118           0 :                     if( pESh )
     119           0 :                         pESh->StartAllAction();
     120           0 :                     else if( pSh )
     121           0 :                         pSh->StartAction();
     122             :                 }
     123           0 :                 pLast->ModifyNotification( 0, &aUpdateDDE );
     124           0 :                 bCallModify = true;
     125             :             }
     126             :         }
     127             : 
     128           0 :         rFieldType.UnlockModify();
     129             : 
     130           0 :         if( bCallModify )
     131             :         {
     132           0 :             if( pESh )
     133           0 :                 pESh->EndAllAction();
     134           0 :             else if( pSh )
     135           0 :                 pSh->EndAction();
     136             : 
     137           0 :             if( pSh )
     138           0 :                 pSh->GetDoc()->getIDocumentState().SetModified();
     139           0 :         }
     140             :     }
     141             : 
     142           0 :     return SUCCESS;
     143             : }
     144             : 
     145           0 : void SwIntrnlRefLink::Closed()
     146             : {
     147           0 :     if( rFieldType.GetDoc() && !rFieldType.GetDoc()->IsInDtor() )
     148             :     {
     149             :         // advise goes, convert all fields into text?
     150           0 :         SwViewShell* pSh = rFieldType.GetDoc()->getIDocumentLayoutAccess().GetCurrentViewShell();
     151           0 :         SwEditShell* pESh = rFieldType.GetDoc()->GetEditShell();
     152           0 :         if( pESh )
     153             :         {
     154           0 :             pESh->StartAllAction();
     155           0 :             pESh->FieldToText( &rFieldType );
     156           0 :             pESh->EndAllAction();
     157             :         }
     158             :         else
     159             :         {
     160           0 :             pSh->StartAction();
     161             :             // am Doc aufrufen ??
     162           0 :             pSh->EndAction();
     163             :         }
     164             :     }
     165           0 :     SvBaseLink::Closed();
     166           0 : }
     167             : 
     168           0 : const SwNode* SwIntrnlRefLink::GetAnchor() const
     169             : {
     170             :     // here, any anchor of the normal NodesArray should be sufficient
     171           0 :     const SwNode* pNd = 0;
     172           0 :     SwIterator<SwClient,SwFieldType> aIter(rFieldType);
     173           0 :     for(SwClient* pLast = aIter.First(); pLast; pLast = aIter.Next())
     174             :     {
     175             :         // a DDE table or a DDE field attribute in the text
     176           0 :         if( !pLast->IsA( TYPE( SwFormatField ) ))
     177             :         {
     178           0 :             SwDepend* pDep = static_cast<SwDepend*>(pLast);
     179           0 :             SwDDETable* pDDETable = static_cast<SwDDETable*>(pDep->GetToTell());
     180           0 :             pNd = pDDETable->GetTabSortBoxes()[0]->GetSttNd();
     181             :         }
     182           0 :         else if( static_cast<SwFormatField*>(pLast)->GetTextField() )
     183           0 :             pNd = static_cast<SwFormatField*>(pLast)->GetTextField()->GetpTextNode();
     184             : 
     185           0 :         if( pNd && &rFieldType.GetDoc()->GetNodes() == &pNd->GetNodes() )
     186           0 :             break;
     187           0 :         pNd = 0;
     188             :     }
     189           0 :     return pNd;
     190             : }
     191             : 
     192           0 : bool SwIntrnlRefLink::IsInRange( sal_uLong nSttNd, sal_uLong nEndNd,
     193             :                                 sal_Int32 nStt, sal_Int32 nEnd ) const
     194             : {
     195             :     // here, any anchor of the normal NodesArray should be sufficient
     196           0 :     SwNodes* pNds = &rFieldType.GetDoc()->GetNodes();
     197           0 :     SwIterator<SwClient,SwFieldType> aIter(rFieldType);
     198           0 :     for(SwClient* pLast = aIter.First(); pLast; pLast = aIter.Next())
     199             :     {
     200             :         // a DDE table or a DDE field attribute in the text
     201           0 :         if( !pLast->IsA( TYPE( SwFormatField ) ))
     202             :         {
     203           0 :             SwDepend* pDep = static_cast<SwDepend*>(pLast);
     204           0 :             SwDDETable* pDDETable = static_cast<SwDDETable*>(pDep->GetToTell());
     205           0 :             const SwTableNode* pTableNd = pDDETable->GetTabSortBoxes()[0]->
     206           0 :                             GetSttNd()->FindTableNode();
     207           0 :             if( pTableNd->GetNodes().IsDocNodes() &&
     208           0 :                 nSttNd < pTableNd->EndOfSectionIndex() &&
     209           0 :                 nEndNd > pTableNd->GetIndex() )
     210           0 :                 return true;
     211             :         }
     212           0 :         else if( static_cast<SwFormatField*>(pLast)->GetTextField() )
     213             :         {
     214           0 :             const SwTextField* pTField = static_cast<SwFormatField*>(pLast)->GetTextField();
     215           0 :             const SwTextNode* pNd = pTField->GetpTextNode();
     216           0 :             if( pNd && pNds == &pNd->GetNodes() )
     217             :             {
     218           0 :                 sal_uLong nNdPos = pNd->GetIndex();
     219           0 :                 if( nSttNd <= nNdPos && nNdPos <= nEndNd &&
     220           0 :                     ( nNdPos != nSttNd || pTField->GetStart() >= nStt ) &&
     221           0 :                     ( nNdPos != nEndNd || pTField->GetStart() < nEnd ))
     222           0 :                     return true;
     223             :             }
     224             :         }
     225             :     }
     226             : 
     227           0 :     return false;
     228             : }
     229             : 
     230           0 : SwDDEFieldType::SwDDEFieldType(const OUString& rName,
     231             :                                const OUString& rCmd, SfxLinkUpdateMode nUpdateType )
     232             :     : SwFieldType( RES_DDEFLD ),
     233           0 :     aName( rName ), pDoc( 0 ), nRefCnt( 0 )
     234             : {
     235           0 :     bCRLFFlag = bDeleted = false;
     236           0 :     refLink = new SwIntrnlRefLink( *this, nUpdateType, SotClipboardFormatId::STRING );
     237           0 :     SetCmd( rCmd );
     238           0 : }
     239             : 
     240           0 : SwDDEFieldType::~SwDDEFieldType()
     241             : {
     242           0 :     if( pDoc && !pDoc->IsInDtor() )
     243           0 :         pDoc->getIDocumentLinksAdministration().GetLinkManager().Remove( refLink );
     244           0 :     refLink->Disconnect();
     245           0 : }
     246             : 
     247           0 : SwFieldType* SwDDEFieldType::Copy() const
     248             : {
     249           0 :     SwDDEFieldType* pType = new SwDDEFieldType( aName, GetCmd(), GetType() );
     250           0 :     pType->aExpansion = aExpansion;
     251           0 :     pType->bCRLFFlag = bCRLFFlag;
     252           0 :     pType->bDeleted = bDeleted;
     253           0 :     pType->SetDoc( pDoc );
     254           0 :     return pType;
     255             : }
     256             : 
     257           0 : OUString SwDDEFieldType::GetName() const
     258             : {
     259           0 :     return aName;
     260             : }
     261             : 
     262           0 : void SwDDEFieldType::SetCmd( const OUString& _aStr )
     263             : {
     264           0 :     OUString aStr = _aStr;
     265           0 :     sal_Int32 nIndex = 0;
     266           0 :     do
     267             :     {
     268           0 :         aStr = aStr.replaceFirst("  ", " ", &nIndex);
     269           0 :     } while (nIndex>=0);
     270           0 :     refLink->SetLinkSourceName( aStr );
     271           0 : }
     272             : 
     273           0 : OUString SwDDEFieldType::GetCmd() const
     274             : {
     275           0 :     return refLink->GetLinkSourceName();
     276             : }
     277             : 
     278           0 : void SwDDEFieldType::SetDoc( SwDoc* pNewDoc )
     279             : {
     280           0 :     if( pNewDoc == pDoc )
     281           0 :         return;
     282             : 
     283           0 :     if( pDoc && refLink.Is() )
     284             :     {
     285             :         OSL_ENSURE( !nRefCnt, "How do we get the references?" );
     286           0 :         pDoc->getIDocumentLinksAdministration().GetLinkManager().Remove( refLink );
     287             :     }
     288             : 
     289           0 :     pDoc = pNewDoc;
     290           0 :     if( pDoc && nRefCnt )
     291             :     {
     292           0 :         refLink->SetVisible( pDoc->getIDocumentLinksAdministration().IsVisibleLinks() );
     293           0 :         pDoc->getIDocumentLinksAdministration().GetLinkManager().InsertDDELink( refLink );
     294             :     }
     295             : }
     296             : 
     297           0 : void SwDDEFieldType::_RefCntChgd()
     298             : {
     299           0 :     if( nRefCnt )
     300             :     {
     301           0 :         refLink->SetVisible( pDoc->getIDocumentLinksAdministration().IsVisibleLinks() );
     302           0 :         pDoc->getIDocumentLinksAdministration().GetLinkManager().InsertDDELink( refLink );
     303           0 :         if( pDoc->getIDocumentLayoutAccess().GetCurrentViewShell() )
     304           0 :             UpdateNow();
     305             :     }
     306             :     else
     307             :     {
     308           0 :         Disconnect();
     309           0 :         pDoc->getIDocumentLinksAdministration().GetLinkManager().Remove( refLink );
     310             :     }
     311           0 : }
     312             : 
     313           0 : bool SwDDEFieldType::QueryValue( uno::Any& rVal, sal_uInt16 nWhichId ) const
     314             : {
     315           0 :     sal_Int32 nPart = -1;
     316           0 :     switch( nWhichId )
     317             :     {
     318           0 :     case FIELD_PROP_PAR2:      nPart = 2; break;
     319           0 :     case FIELD_PROP_PAR4:      nPart = 1; break;
     320           0 :     case FIELD_PROP_SUBTYPE:   nPart = 0; break;
     321             :     case FIELD_PROP_BOOL1:
     322           0 :         rVal <<= GetType() == SfxLinkUpdateMode::ALWAYS;
     323           0 :         break;
     324             :     case FIELD_PROP_PAR5:
     325           0 :         rVal <<= aExpansion;
     326           0 :         break;
     327             :     default:
     328             :         OSL_FAIL("illegal property");
     329             :     }
     330           0 :     if ( nPart>=0 )
     331           0 :         rVal <<= GetCmd().getToken(nPart, sfx2::cTokenSeparator);
     332           0 :     return true;
     333             : }
     334             : 
     335           0 : bool SwDDEFieldType::PutValue( const uno::Any& rVal, sal_uInt16 nWhichId )
     336             : {
     337           0 :     sal_Int32 nPart = -1;
     338           0 :     switch( nWhichId )
     339             :     {
     340           0 :     case FIELD_PROP_PAR2:      nPart = 2; break;
     341           0 :     case FIELD_PROP_PAR4:      nPart = 1; break;
     342           0 :     case FIELD_PROP_SUBTYPE:   nPart = 0; break;
     343             :     case FIELD_PROP_BOOL1:
     344           0 :         SetType( *static_cast<sal_Bool const *>(rVal.getValue()) ?
     345             :                  SfxLinkUpdateMode::ALWAYS :
     346           0 :                  SfxLinkUpdateMode::ONCALL );
     347           0 :         break;
     348             :     case FIELD_PROP_PAR5:
     349           0 :         rVal >>= aExpansion;
     350           0 :         break;
     351             :     default:
     352             :         OSL_FAIL("illegal property");
     353             :     }
     354           0 :     if( nPart>=0 )
     355             :     {
     356           0 :         const OUString sOldCmd( GetCmd() );
     357           0 :         OUString sNewCmd;
     358           0 :         sal_Int32 nIndex = 0;
     359           0 :         for (sal_Int32 i=0; i<3; ++i)
     360             :         {
     361           0 :             OUString sToken = sOldCmd.getToken(0, sfx2::cTokenSeparator, nIndex);
     362           0 :             if (i==nPart)
     363             :             {
     364           0 :                 rVal >>= sToken;
     365             :             }
     366           0 :             sNewCmd += (i < 2)
     367           0 :                 ? sToken + OUString(sfx2::cTokenSeparator) : sToken;
     368           0 :         }
     369           0 :         SetCmd( sNewCmd );
     370             :     }
     371           0 :     return true;
     372             : }
     373             : 
     374           0 : SwDDEField::SwDDEField( SwDDEFieldType* pInitType )
     375           0 :     : SwField(pInitType)
     376             : {
     377           0 : }
     378             : 
     379           0 : SwDDEField::~SwDDEField()
     380             : {
     381           0 :     if( GetTyp()->HasOnlyOneListener() )
     382           0 :         static_cast<SwDDEFieldType*>(GetTyp())->Disconnect();
     383           0 : }
     384             : 
     385           0 : OUString SwDDEField::Expand() const
     386             : {
     387           0 :     OUString aStr = static_cast<SwDDEFieldType*>(GetTyp())->GetExpansion();
     388           0 :     aStr = aStr.replaceAll("\r", OUString());
     389           0 :     aStr = aStr.replaceAll("\t", " ");
     390           0 :     aStr = aStr.replaceAll("\n", "|");
     391           0 :     if (aStr.endsWith("|"))
     392             :     {
     393           0 :         return aStr.copy(0, aStr.getLength()-1);
     394             :     }
     395           0 :     return aStr;
     396             : }
     397             : 
     398           0 : SwField* SwDDEField::Copy() const
     399             : {
     400           0 :     return new SwDDEField(static_cast<SwDDEFieldType*>(GetTyp()));
     401             : }
     402             : 
     403             : /// get field type name
     404           0 : OUString SwDDEField::GetPar1() const
     405             : {
     406           0 :     return static_cast<const SwDDEFieldType*>(GetTyp())->GetName();
     407             : }
     408             : 
     409             : /// get field type command
     410           0 : OUString SwDDEField::GetPar2() const
     411             : {
     412           0 :     return static_cast<const SwDDEFieldType*>(GetTyp())->GetCmd();
     413             : }
     414             : 
     415             : /// set field type command
     416           0 : void SwDDEField::SetPar2(const OUString& rStr)
     417             : {
     418           0 :     static_cast<SwDDEFieldType*>(GetTyp())->SetCmd(rStr);
     419         177 : }
     420             : 
     421             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.11