LCOV - code coverage report
Current view: top level - xmloff/source/transform - TransformerBase.cxx (source / functions) Hit Total Coverage
Test: commit 10e77ab3ff6f4314137acd6e2702a6e5c1ce1fae Lines: 391 706 55.4 %
Date: 2014-11-03 Functions: 30 39 76.9 %
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 <rtl/ref.hxx>
      21             : #include <rtl/ustrbuf.hxx>
      22             : #include <com/sun/star/i18n/CharacterClassification.hpp>
      23             : #include <com/sun/star/i18n/UnicodeType.hpp>
      24             : #include <com/sun/star/util/MeasureUnit.hpp>
      25             : #include <sax/tools/converter.hxx>
      26             : #include <comphelper/processfactory.hxx>
      27             : #include <xmloff/nmspmap.hxx>
      28             : #include <xmloff/xmlnmspe.hxx>
      29             : #include "IgnoreTContext.hxx"
      30             : #include "RenameElemTContext.hxx"
      31             : #include "ProcAttrTContext.hxx"
      32             : #include "ProcAddAttrTContext.hxx"
      33             : #include "MergeElemTContext.hxx"
      34             : #include "CreateElemTContext.hxx"
      35             : #include "MutableAttrList.hxx"
      36             : #include "TransformerActions.hxx"
      37             : #include "ElemTransformerAction.hxx"
      38             : #include "PropertyActionsOOo.hxx"
      39             : #include "TransformerTokenMap.hxx"
      40             : 
      41             : #include "TransformerBase.hxx"
      42             : #include "TContextVector.hxx"
      43             : 
      44             : using namespace ::osl;
      45             : using namespace ::xmloff::token;
      46             : using namespace ::com::sun::star;
      47             : using namespace ::com::sun::star::uno;
      48             : using namespace ::com::sun::star::beans;
      49             : using namespace ::com::sun::star::lang;
      50             : using namespace ::com::sun::star::i18n;
      51             : using namespace ::com::sun::star::xml::sax;
      52             : 
      53             : namespace
      54             : {
      55          12 : bool lcl_ConvertAttr( OUString & rOutAttribute, sal_Int32 nParam )
      56             : {
      57          12 :     bool bResult = false;
      58             :     enum XMLTokenEnum eTokenToRename =
      59          12 :         static_cast< enum XMLTokenEnum >( nParam & 0xffff );
      60          24 :     if( eTokenToRename != XML_TOKEN_INVALID &&
      61          12 :         IsXMLToken( rOutAttribute, eTokenToRename ))
      62             :     {
      63             :         enum XMLTokenEnum eReplacementToken =
      64           6 :             static_cast< enum XMLTokenEnum >( nParam >> 16 );
      65           6 :         rOutAttribute = GetXMLToken( eReplacementToken );
      66           6 :         bResult = true;
      67             :     }
      68          12 :     return bResult;
      69             : }
      70             : } // anonymous namespace
      71             : 
      72        3916 : XMLTransformerContext *XMLTransformerBase::CreateContext( sal_uInt16 nPrefix,
      73             :     const OUString& rLocalName, const OUString& rQName )
      74             : {
      75        3916 :     XMLTransformerActions::key_type aKey( nPrefix, rLocalName );
      76             :     XMLTransformerActions::const_iterator aIter =
      77        3916 :         GetElemActions().find( aKey );
      78             : 
      79        3916 :     if( !(aIter == GetElemActions().end()) )
      80             :     {
      81        2422 :         sal_uInt32 nActionType = (*aIter).second.m_nActionType;
      82        2422 :         if( (nActionType & XML_ETACTION_USER_DEFINED) != 0 )
      83             :         {
      84             :             XMLTransformerContext *pContext =
      85        1110 :                 CreateUserDefinedContext( (*aIter).second,
      86        2220 :                                     rQName );
      87             :             OSL_ENSURE( pContext && !pContext->IsPersistent(),
      88             :                         "unknown or not persistent action" );
      89        1110 :             return pContext;
      90             :         }
      91             : 
      92        1312 :         switch( nActionType )
      93             :         {
      94             :         case XML_ETACTION_COPY_CONTENT:
      95             :             return new XMLIgnoreTransformerContext( *this, rQName, false,
      96           2 :                                                 false );
      97             :         case XML_ETACTION_COPY:
      98           0 :             return new XMLTransformerContext( *this, rQName );
      99             :         case XML_ETACTION_RENAME_ELEM:
     100             :             return new XMLRenameElemTransformerContext( *this, rQName,
     101          56 :                     (*aIter).second.GetQNamePrefixFromParam1(),
     102          56 :                     (*aIter).second.GetQNameTokenFromParam1() );
     103             :         case XML_ETACTION_RENAME_ELEM_ADD_ATTR:
     104             :             return new XMLRenameElemTransformerContext( *this, rQName,
     105           0 :                     (*aIter).second.GetQNamePrefixFromParam1(),
     106           0 :                     (*aIter).second.GetQNameTokenFromParam1(),
     107           0 :                     (*aIter).second.GetQNamePrefixFromParam2(),
     108           0 :                     (*aIter).second.GetQNameTokenFromParam2(),
     109           0 :                        static_cast< XMLTokenEnum >( (*aIter).second.m_nParam3 ) );
     110             :         case XML_ETACTION_RENAME_ELEM_PROC_ATTRS:
     111             :             return new XMLProcAttrTransformerContext( *this, rQName,
     112         128 :                     (*aIter).second.GetQNamePrefixFromParam1(),
     113         128 :                     (*aIter).second.GetQNameTokenFromParam1(),
     114         256 :                        static_cast< sal_uInt16 >( (*aIter).second.m_nParam2 ) );
     115             :         case XML_ETACTION_RENAME_ELEM_ADD_PROC_ATTR:
     116             :             return new XMLProcAddAttrTransformerContext( *this, rQName,
     117          16 :                     (*aIter).second.GetQNamePrefixFromParam1(),
     118          16 :                     (*aIter).second.GetQNameTokenFromParam1(),
     119             :                        static_cast< sal_uInt16 >(
     120          16 :                         (*aIter).second.m_nParam3  >> 16 ),
     121          16 :                     (*aIter).second.GetQNamePrefixFromParam2(),
     122          16 :                     (*aIter).second.GetQNameTokenFromParam2(),
     123             :                        static_cast< XMLTokenEnum >(
     124          80 :                         (*aIter).second.m_nParam3 & 0xffff ) );
     125             :         case XML_ETACTION_RENAME_ELEM_COND:
     126             :             {
     127           0 :                 const XMLTransformerContext *pCurrent = GetCurrentContext();
     128           0 :                 if( pCurrent->HasQName(
     129           0 :                             (*aIter).second.GetQNamePrefixFromParam2(),
     130           0 :                             (*aIter).second.GetQNameTokenFromParam2() ) )
     131             :                     return new XMLRenameElemTransformerContext( *this, rQName,
     132           0 :                             (*aIter).second.GetQNamePrefixFromParam1(),
     133           0 :                             (*aIter).second.GetQNameTokenFromParam1() );
     134             :             }
     135           0 :             break;
     136             :         case XML_ETACTION_RENAME_ELEM_PROC_ATTRS_COND:
     137             :             {
     138           0 :                 const XMLTransformerContext *pCurrent = GetCurrentContext();
     139           0 :                 if( pCurrent->HasQName(
     140           0 :                             (*aIter).second.GetQNamePrefixFromParam3(),
     141           0 :                             (*aIter).second.GetQNameTokenFromParam3() ) )
     142             :                     return new XMLProcAttrTransformerContext( *this, rQName,
     143           0 :                             (*aIter).second.GetQNamePrefixFromParam1(),
     144           0 :                             (*aIter).second.GetQNameTokenFromParam1(),
     145           0 :                             static_cast< sal_uInt16 >( (*aIter).second.m_nParam2 ) );
     146             :                 else
     147             :                     return new XMLProcAttrTransformerContext( *this, rQName,
     148           0 :                             static_cast< sal_uInt16 >( (*aIter).second.m_nParam2 ) );
     149             :             }
     150             :         case XML_ETACTION_PROC_ATTRS:
     151             :             return new XMLProcAttrTransformerContext( *this, rQName,
     152        1076 :                        static_cast< sal_uInt16 >( (*aIter).second.m_nParam1 ) );
     153             :         case XML_ETACTION_PROC_ATTRS_COND:
     154             :             {
     155          24 :                 const XMLTransformerContext *pCurrent = GetCurrentContext();
     156          24 :                 if( pCurrent->HasQName(
     157          24 :                             (*aIter).second.GetQNamePrefixFromParam1(),
     158          48 :                             (*aIter).second.GetQNameTokenFromParam1() ) )
     159             :                     return new XMLProcAttrTransformerContext( *this, rQName,
     160          24 :                             static_cast< sal_uInt16 >( (*aIter).second.m_nParam2 ) );
     161             :             }
     162           0 :             break;
     163             :         case XML_ETACTION_MOVE_ATTRS_TO_ELEMS:
     164             :             return new XMLCreateElemTransformerContext( *this, rQName,
     165          10 :                        static_cast< sal_uInt16 >( (*aIter).second.m_nParam1 ) );
     166             :         case XML_ETACTION_MOVE_ELEMS_TO_ATTRS:
     167             :             return new XMLMergeElemTransformerContext( *this, rQName,
     168           0 :                        static_cast< sal_uInt16 >( (*aIter).second.m_nParam1 ) );
     169             :         default:
     170             :             OSL_ENSURE( false, "unknown action" );
     171           0 :             break;
     172             :         }
     173             :     }
     174             : 
     175             :     // default is copying
     176        1494 :     return new XMLTransformerContext( *this, rQName );
     177             : }
     178             : 
     179           0 : XMLTransformerActions *XMLTransformerBase::GetUserDefinedActions( sal_uInt16 )
     180             : {
     181           0 :     return 0;
     182             : }
     183             : 
     184         124 : XMLTransformerBase::XMLTransformerBase( XMLTransformerActionInit *pInit,
     185             :                                     ::xmloff::token::XMLTokenEnum *pTKMapInit )
     186             :     throw () :
     187         124 :     m_pNamespaceMap( new SvXMLNamespaceMap ),
     188         124 :     m_pReplaceNamespaceMap( new SvXMLNamespaceMap ),
     189         124 :     m_pContexts( new XMLTransformerContextVector ),
     190         124 :     m_pElemActions( new XMLTransformerActions( pInit ) ),
     191         620 :     m_pTokenMap( new XMLTransformerTokenMap( pTKMapInit ) )
     192             : {
     193         124 :     GetNamespaceMap().Add( GetXMLToken(XML_NP_XLINK), GetXMLToken(XML_N_XLINK), XML_NAMESPACE_XLINK );
     194         124 :     GetNamespaceMap().Add( GetXMLToken(XML_NP_DC), GetXMLToken(XML_N_DC), XML_NAMESPACE_DC );
     195         124 :     GetNamespaceMap().Add( GetXMLToken(XML_NP_MATH), GetXMLToken(XML_N_MATH), XML_NAMESPACE_MATH );
     196         124 :     GetNamespaceMap().Add( GetXMLToken(XML_NP_OOO), GetXMLToken(XML_N_OOO), XML_NAMESPACE_OOO );
     197         124 :     GetNamespaceMap().Add( GetXMLToken(XML_NP_DOM), GetXMLToken(XML_N_DOM), XML_NAMESPACE_DOM );
     198         124 :     GetNamespaceMap().Add( GetXMLToken(XML_NP_OOOW), GetXMLToken(XML_N_OOOW), XML_NAMESPACE_OOOW );
     199         124 :     GetNamespaceMap().Add( GetXMLToken(XML_NP_OOOC), GetXMLToken(XML_N_OOOC), XML_NAMESPACE_OOOC );
     200         124 : }
     201             : 
     202         248 : XMLTransformerBase::~XMLTransformerBase() throw ()
     203             : {
     204         124 :     delete m_pNamespaceMap;
     205         124 :     delete m_pReplaceNamespaceMap;
     206         124 :     delete m_pContexts;
     207         124 :     delete m_pElemActions;
     208         124 :     delete m_pTokenMap;
     209         124 : }
     210             : 
     211          82 : void SAL_CALL XMLTransformerBase::startDocument( void )
     212             :     throw( SAXException, RuntimeException, std::exception )
     213             : {
     214          82 :     m_xHandler->startDocument();
     215          82 : }
     216             : 
     217          82 : void SAL_CALL XMLTransformerBase::endDocument( void )
     218             :     throw( SAXException, RuntimeException, std::exception)
     219             : {
     220          82 :     m_xHandler->endDocument();
     221          82 : }
     222             : 
     223        4706 : void SAL_CALL XMLTransformerBase::startElement( const OUString& rName,
     224             :                                          const Reference< XAttributeList >& rAttrList )
     225             :     throw(SAXException, RuntimeException, std::exception)
     226             : {
     227        4706 :     SvXMLNamespaceMap *pRewindMap = 0;
     228             : 
     229        4706 :     bool bRect = rName == "presentation:show-shape";
     230             :     (void)bRect;
     231             : 
     232             :     // Process namespace attributes. This must happen before creating the
     233             :     // context, because namespace decaration apply to the element name itself.
     234        4706 :     XMLMutableAttributeList *pMutableAttrList = 0;
     235        4706 :     Reference< XAttributeList > xAttrList( rAttrList );
     236        4706 :     sal_Int16 nAttrCount = xAttrList.is() ? xAttrList->getLength() : 0;
     237       16332 :     for( sal_Int16 i=0; i < nAttrCount; i++ )
     238             :     {
     239       11626 :         const OUString& rAttrName = xAttrList->getNameByIndex( i );
     240       34878 :         if( ( rAttrName.getLength() >= 5 ) &&
     241       13314 :             ( rAttrName.startsWith( GetXMLToken(XML_XMLNS) ) ) &&
     242        1688 :             ( rAttrName.getLength() == 5 || ':' == rAttrName[5] ) )
     243             :         {
     244         844 :             if( !pRewindMap )
     245             :             {
     246          84 :                 pRewindMap = m_pNamespaceMap;
     247          84 :                 m_pNamespaceMap = new SvXMLNamespaceMap( *m_pNamespaceMap );
     248             :             }
     249         844 :             const OUString& rAttrValue = xAttrList->getValueByIndex( i );
     250             : 
     251         844 :             OUString aPrefix( ( rAttrName.getLength() == 5 )
     252             :                                  ? OUString()
     253        1688 :                                  : rAttrName.copy( 6 ) );
     254             :             // Add namespace, but only if it is known.
     255         844 :             sal_uInt16 nKey = m_pNamespaceMap->AddIfKnown( aPrefix, rAttrValue );
     256             :             // If namespace is unknown, try to match a name with similar
     257             :             // TC Id an version
     258         844 :             if( XML_NAMESPACE_UNKNOWN == nKey  )
     259             :             {
     260          64 :                 OUString aTestName( rAttrValue );
     261          64 :                 if( SvXMLNamespaceMap::NormalizeOasisURN( aTestName ) )
     262          10 :                     nKey = m_pNamespaceMap->AddIfKnown( aPrefix, aTestName );
     263             :             }
     264             :             // If that namespace is not known, too, add it as unknown
     265         844 :             if( XML_NAMESPACE_UNKNOWN == nKey  )
     266          64 :                 nKey = m_pNamespaceMap->Add( aPrefix, rAttrValue );
     267             : 
     268         844 :             const OUString& rRepName = m_pReplaceNamespaceMap->GetNameByKey( nKey );
     269         844 :             if( !rRepName.isEmpty() )
     270             :             {
     271         604 :                 if( !pMutableAttrList )
     272             :                 {
     273          78 :                     pMutableAttrList = new XMLMutableAttributeList( xAttrList );
     274          78 :                     xAttrList = pMutableAttrList;
     275             :                 }
     276             : 
     277         604 :                 pMutableAttrList->SetValueByIndex( i, rRepName );
     278         844 :             }
     279             :         }
     280       11626 :     }
     281             : 
     282             :     // Get element's namespace and local name.
     283        9412 :     OUString aLocalName;
     284             :     sal_uInt16 nPrefix =
     285        4706 :         m_pNamespaceMap->GetKeyByAttrName( rName, &aLocalName );
     286             : 
     287             :     // If there are contexts already, call a CreateChildContext at the topmost
     288             :     // context. Otherwise, create a default context.
     289        9412 :     ::rtl::Reference < XMLTransformerContext > xContext;
     290        4706 :     if( !m_pContexts->empty() )
     291             :     {
     292        9248 :         xContext = m_pContexts->back()->CreateChildContext( nPrefix,
     293             :                                                           aLocalName,
     294             :                                                           rName,
     295        9248 :                                                           xAttrList );
     296             :     }
     297             :     else
     298             :     {
     299          82 :         xContext = CreateContext( nPrefix, aLocalName, rName );
     300             :     }
     301             : 
     302             :     OSL_ENSURE( xContext.is(), "XMLTransformerBase::startElement: missing context" );
     303        4706 :     if( !xContext.is() )
     304           0 :         xContext = new XMLTransformerContext( *this, rName );
     305             : 
     306             :     // Remember old namespace map.
     307        4706 :     if( pRewindMap )
     308          84 :         xContext->SetRewindMap( pRewindMap );
     309             : 
     310             :     // Push context on stack.
     311        4706 :     m_pContexts->push_back( xContext );
     312             : 
     313             :     // Call a startElement at the new context.
     314        9412 :     xContext->StartElement( xAttrList );
     315        4706 : }
     316             : 
     317        4706 : void SAL_CALL XMLTransformerBase::endElement( const OUString&
     318             : #if OSL_DEBUG_LEVEL > 0
     319             : rName
     320             : #endif
     321             : )
     322             :     throw(SAXException, RuntimeException, std::exception)
     323             : {
     324        4706 :     if( !m_pContexts->empty() )
     325             :     {
     326             :         // Get topmost context
     327        4706 :         ::rtl::Reference< XMLTransformerContext > xContext = m_pContexts->back();
     328             : 
     329             : #if OSL_DEBUG_LEVEL > 0
     330             :         OSL_ENSURE( xContext->GetQName() == rName,
     331             :                 "XMLTransformerBase::endElement: popped context has wrong lname" );
     332             : #endif
     333             : 
     334             :         // Call a EndElement at the current context.
     335        4706 :         xContext->EndElement();
     336             : 
     337             :         // and remove it from the stack.
     338        4706 :         m_pContexts->pop_back();
     339             : 
     340             :         // Get a namespace map to rewind.
     341        4706 :         SvXMLNamespaceMap *pRewindMap = xContext->GetRewindMap();
     342             : 
     343             :         // Delete the current context.
     344        4706 :         xContext = 0;
     345             : 
     346             :         // Rewind a namespace map.
     347        4706 :         if( pRewindMap )
     348             :         {
     349          84 :             delete m_pNamespaceMap;
     350          84 :             m_pNamespaceMap = pRewindMap;
     351        4706 :         }
     352             :     }
     353        4706 : }
     354             : 
     355        1916 : void SAL_CALL XMLTransformerBase::characters( const OUString& rChars )
     356             :     throw(SAXException, RuntimeException, std::exception)
     357             : {
     358        1916 :     if( !m_pContexts->empty() )
     359             :     {
     360        1766 :         m_pContexts->back()->Characters( rChars );
     361             :     }
     362        1916 : }
     363             : 
     364          12 : void SAL_CALL XMLTransformerBase::ignorableWhitespace( const OUString& rWhitespaces )
     365             :     throw(SAXException, RuntimeException, std::exception)
     366             : {
     367          12 :     m_xHandler->ignorableWhitespace( rWhitespaces );
     368          12 : }
     369             : 
     370           0 : void SAL_CALL XMLTransformerBase::processingInstruction( const OUString& rTarget,
     371             :                                        const OUString& rData )
     372             :     throw(SAXException, RuntimeException, std::exception)
     373             : {
     374           0 :     m_xHandler->processingInstruction( rTarget, rData );
     375           0 : }
     376             : 
     377          76 : void SAL_CALL XMLTransformerBase::setDocumentLocator( const Reference< XLocator >& rLocator )
     378             :     throw(SAXException, RuntimeException, std::exception)
     379             : {
     380          76 :     m_xLocator = rLocator;
     381          76 : }
     382             : 
     383             : // XExtendedDocumentHandler
     384           0 : void SAL_CALL XMLTransformerBase::startCDATA( void ) throw(SAXException, RuntimeException, std::exception)
     385             : {
     386           0 :     if( m_xExtHandler.is() )
     387           0 :         m_xExtHandler->startCDATA();
     388           0 : }
     389             : 
     390           0 : void SAL_CALL XMLTransformerBase::endCDATA( void ) throw(RuntimeException, std::exception)
     391             : {
     392           0 :     if( m_xExtHandler.is() )
     393           0 :         m_xExtHandler->endCDATA();
     394           0 : }
     395             : 
     396          24 : void SAL_CALL XMLTransformerBase::comment( const OUString& rComment )
     397             :     throw(SAXException, RuntimeException, std::exception)
     398             : {
     399          24 :     if( m_xExtHandler.is() )
     400           0 :         m_xExtHandler->comment( rComment );
     401          24 : }
     402             : 
     403           0 : void SAL_CALL XMLTransformerBase::allowLineBreak( void )
     404             :     throw(SAXException, RuntimeException, std::exception)
     405             : {
     406           0 :     if( m_xExtHandler.is() )
     407           0 :         m_xExtHandler->allowLineBreak();
     408           0 : }
     409             : 
     410         678 : void SAL_CALL XMLTransformerBase::unknown( const OUString& rString )
     411             :     throw(SAXException, RuntimeException, std::exception)
     412             : {
     413         678 :     if( m_xExtHandler.is() )
     414           0 :         m_xExtHandler->unknown( rString );
     415         678 : }
     416             : 
     417             : // XInitialize
     418         130 : void SAL_CALL XMLTransformerBase::initialize( const Sequence< Any >& aArguments )
     419             :     throw(Exception, RuntimeException, std::exception)
     420             : {
     421         130 :     const sal_Int32 nAnyCount = aArguments.getLength();
     422         130 :     const Any* pAny = aArguments.getConstArray();
     423             : 
     424         476 :     for( sal_Int32 nIndex = 0; nIndex < nAnyCount; nIndex++, pAny++ )
     425             :     {
     426             :         // use isAssignableFrom instead of comparing the types to
     427             :         // allow XExtendedDocumentHandler instead of XDocumentHandler (used in
     428             :         // writeOasis2OOoLibraryElement in sfx2).
     429             :         // The Any shift operator can't be used to query the type because it
     430             :         // uses queryInterface, and the model also has a XPropertySet interface.
     431             : 
     432             :         // document handler
     433         346 :         if( cppu::UnoType<XDocumentHandler>::get().isAssignableFrom( pAny->getValueType() ) )
     434         130 :             m_xHandler.set( *pAny, UNO_QUERY );
     435             : 
     436             :         // property set to transport data across
     437         346 :         if( cppu::UnoType<XPropertySet>::get().isAssignableFrom( pAny->getValueType() ) )
     438          76 :             m_xPropSet.set( *pAny, UNO_QUERY );
     439             : 
     440             :         // xmodel
     441         346 :         if( cppu::UnoType<com::sun::star::frame::XModel>::get().isAssignableFrom( pAny->getValueType() ) )
     442           4 :             mxModel.set( *pAny, UNO_QUERY );
     443             :     }
     444             : 
     445         130 :     if( m_xPropSet.is() )
     446             :     {
     447          76 :         Any aAny;
     448         152 :         OUString sRelPath, sName;
     449             :         Reference< XPropertySetInfo > xPropSetInfo =
     450         152 :             m_xPropSet->getPropertySetInfo();
     451         152 :         OUString sPropName( "StreamRelPath"  );
     452          76 :         if( xPropSetInfo->hasPropertyByName(sPropName) )
     453             :         {
     454          72 :             aAny = m_xPropSet->getPropertyValue(sPropName);
     455          72 :             aAny >>= sRelPath;
     456             :         }
     457          76 :         sPropName = "StreamName";
     458          76 :         if( xPropSetInfo->hasPropertyByName(sPropName) )
     459             :         {
     460          72 :             aAny = m_xPropSet->getPropertyValue(sPropName);
     461          72 :             aAny >>= sName;
     462             :         }
     463          76 :         if( !sName.isEmpty() )
     464             :         {
     465          72 :             m_aExtPathPrefix = "../";
     466             : 
     467             :             // If there is a rel path within a package, then append
     468             :             // additional '../'. If the rel path contains an ':', then it is
     469             :             // an absolute URI (or invalid URI, because zip files don't
     470             :             // permit ':'), and it will be ignored.
     471          72 :             if( !sRelPath.isEmpty() )
     472             :             {
     473           0 :                 sal_Int32 nColPos = sRelPath.indexOf( ':' );
     474             :                 OSL_ENSURE( -1 == nColPos,
     475             :                             "StreamRelPath contains ':', absolute URI?" );
     476             : 
     477           0 :                 if( -1 == nColPos )
     478             :                 {
     479           0 :                     OUString sTmp = m_aExtPathPrefix;
     480           0 :                     sal_Int32 nPos = 0;
     481           0 :                     do
     482             :                     {
     483           0 :                         m_aExtPathPrefix += sTmp;
     484           0 :                         nPos = sRelPath.indexOf( '/', nPos + 1 );
     485             :                     }
     486           0 :                     while( -1 != nPos );
     487             :                 }
     488             :             }
     489             : 
     490          76 :         }
     491             :     }
     492             : 
     493             :     assert(m_xHandler.is()); // can't do anything without that
     494         130 : }
     495             : 
     496         204 : static sal_Int16 lcl_getUnit( const OUString& rValue )
     497             : {
     498         204 :     if( rValue.endsWithIgnoreAsciiCase( "cm" ) )
     499         204 :         return util::MeasureUnit::CM;
     500           0 :     else if ( rValue.endsWithIgnoreAsciiCase( "mm" ) )
     501           0 :         return util::MeasureUnit::MM;
     502             :     else
     503           0 :         return util::MeasureUnit::INCH;
     504             : }
     505             : 
     506        1396 : XMLMutableAttributeList *XMLTransformerBase::ProcessAttrList(
     507             :         Reference< XAttributeList >& rAttrList, sal_uInt16 nActionMap,
     508             :            bool bClone  )
     509             : {
     510        1396 :     XMLMutableAttributeList *pMutableAttrList = 0;
     511        1396 :     XMLTransformerActions *pActions = GetUserDefinedActions( nActionMap );
     512             :     OSL_ENSURE( pActions, "go no actions" );
     513        1396 :     if( pActions )
     514             :     {
     515        1396 :         sal_Int16 nAttrCount = rAttrList.is() ? rAttrList->getLength() : 0;
     516        4496 :         for( sal_Int16 i=0; i < nAttrCount; ++i )
     517             :         {
     518        3100 :             const OUString& rAttrName = rAttrList->getNameByIndex( i );
     519        6200 :             const OUString& rAttrValue = rAttrList->getValueByIndex( i );
     520        6200 :             OUString aLocalName;
     521        3100 :             sal_uInt16 nPrefix = GetNamespaceMap().GetKeyByAttrName( rAttrName,
     522        3100 :                                                            &aLocalName );
     523             : 
     524        6200 :             XMLTransformerActions::key_type aKey( nPrefix, aLocalName );
     525             :             XMLTransformerActions::const_iterator aIter =
     526        3100 :                     pActions->find( aKey );
     527        3100 :             if( !(aIter == pActions->end() ) )
     528             :             {
     529        1596 :                 if( !pMutableAttrList )
     530             :                 {
     531             :                     pMutableAttrList = new XMLMutableAttributeList( rAttrList,
     532         784 :                                                                     bClone );
     533         784 :                     rAttrList = pMutableAttrList;
     534             :                 }
     535             : 
     536        1596 :                 sal_uInt32 nAction = (*aIter).second.m_nActionType;
     537        1596 :                 bool bRename = false;
     538        1596 :                 switch( nAction )
     539             :                 {
     540             :                 case XML_ATACTION_RENAME:
     541         412 :                     bRename = true;
     542         412 :                     break;
     543             :                 case XML_ATACTION_COPY:
     544           0 :                     break;
     545             :                 case XML_ATACTION_REMOVE:
     546             :                 case XML_ATACTION_STYLE_DISPLAY_NAME:
     547           0 :                     pMutableAttrList->RemoveAttributeByIndex( i );
     548           0 :                     --i;
     549           0 :                     --nAttrCount;
     550           0 :                     break;
     551             :                 case XML_ATACTION_RENAME_IN2INCH:
     552           0 :                     bRename = true;
     553             :                 case XML_ATACTION_IN2INCH:
     554             :                     {
     555           0 :                         OUString aAttrValue( rAttrValue );
     556           0 :                         if( ReplaceSingleInWithInch( aAttrValue ) )
     557           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     558             :                     }
     559           0 :                     break;
     560             :                 case XML_ATACTION_INS2INCHS:
     561             :                     {
     562           0 :                         OUString aAttrValue( rAttrValue );
     563           0 :                         if( ReplaceInWithInch( aAttrValue ) )
     564           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     565             :                     }
     566           0 :                     break;
     567             :                 case XML_ATACTION_RENAME_INCH2IN:
     568           0 :                     bRename = true;
     569             :                 case XML_ATACTION_INCH2IN:
     570             :                     {
     571         330 :                         OUString aAttrValue( rAttrValue );
     572         330 :                         if( ReplaceSingleInchWithIn( aAttrValue ) )
     573           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     574             :                     }
     575         330 :                     break;
     576             :                 case XML_ATACTION_INCHS2INS:
     577             :                     {
     578           0 :                         OUString aAttrValue( rAttrValue );
     579           0 :                         if( ReplaceInchWithIn( aAttrValue ) )
     580           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     581             :                     }
     582           0 :                     break;
     583             :                 case XML_ATACTION_TWIPS2IN:
     584             :                     {
     585           0 :                         OUString aAttrValue( rAttrValue );
     586             : 
     587           0 :                         XMLTransformerBase::ReplaceSingleInchWithIn( aAttrValue );
     588           0 :                         if( isWriter() )
     589             :                         {
     590           0 :                             sal_Int16 const nDestUnit = lcl_getUnit(aAttrValue);
     591             : 
     592             :                             // convert twips value to inch
     593             :                             sal_Int32 nMeasure;
     594           0 :                             if (::sax::Converter::convertMeasure(nMeasure,
     595             :                                     aAttrValue, util::MeasureUnit::MM_100TH))
     596             :                             {
     597             : 
     598             :                                 // #i13778#,#i36248# apply correct twip-to-1/100mm
     599           0 :                                 nMeasure = (sal_Int32)( nMeasure >= 0
     600           0 :                                                         ? ((nMeasure*127+36)/72)
     601           0 :                                                         : ((nMeasure*127-36)/72) );
     602             : 
     603           0 :                                 OUStringBuffer aBuffer;
     604             :                                 ::sax::Converter::convertMeasure(aBuffer,
     605             :                                         nMeasure, util::MeasureUnit::MM_100TH,
     606           0 :                                         nDestUnit );
     607           0 :                                 aAttrValue = aBuffer.makeStringAndClear();
     608             :                             }
     609             :                         }
     610             : 
     611           0 :                         pMutableAttrList->SetValueByIndex( i, aAttrValue );
     612             :                     }
     613           0 :                     break;
     614             :                 case XML_ATACTION_RENAME_DECODE_STYLE_NAME_REF:
     615           0 :                     bRename = true;
     616             :                 case XML_ATACTION_DECODE_STYLE_NAME:
     617             :                 case XML_ATACTION_DECODE_STYLE_NAME_REF:
     618             :                     {
     619           6 :                         OUString aAttrValue( rAttrValue );
     620           6 :                         if( DecodeStyleName(aAttrValue) )
     621           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     622             :                     }
     623           6 :                     break;
     624             :                 case XML_ATACTION_ENCODE_STYLE_NAME:
     625             :                     {
     626          36 :                         OUString aAttrValue( rAttrValue );
     627          36 :                         if( EncodeStyleName(aAttrValue) )
     628             :                         {
     629           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     630             :                             OUString aNewAttrQName(
     631           0 :                                 GetNamespaceMap().GetQNameByKey(
     632             :                                     nPrefix,
     633             :                                 ::xmloff::token::GetXMLToken(
     634           0 :                                 XML_DISPLAY_NAME ) ) );
     635             :                             pMutableAttrList->AddAttribute( aNewAttrQName,
     636           0 :                                                             rAttrValue );
     637          36 :                         }
     638             :                     }
     639          36 :                     break;
     640             :                 case XML_ATACTION_RENAME_ENCODE_STYLE_NAME_REF:
     641          36 :                     bRename = true;
     642             :                 case XML_ATACTION_ENCODE_STYLE_NAME_REF:
     643             :                     {
     644         476 :                         OUString aAttrValue( rAttrValue );
     645         476 :                         if( EncodeStyleName(aAttrValue) )
     646           4 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     647             :                     }
     648         476 :                     break;
     649             :                 case XML_ATACTION_RENAME_NEG_PERCENT:
     650           0 :                     bRename = true;
     651             :                 case XML_ATACTION_NEG_PERCENT:
     652             :                     {
     653           0 :                         OUString aAttrValue( rAttrValue );
     654           0 :                         if( NegPercent( aAttrValue ) )
     655           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     656             :                     }
     657           0 :                     break;
     658             :                 case XML_ATACTION_RENAME_ADD_NAMESPACE_PREFIX:
     659          58 :                     bRename = true;
     660             :                 case XML_ATACTION_ADD_NAMESPACE_PREFIX:
     661             :                     {
     662          82 :                         OUString aAttrValue( rAttrValue );
     663             :                         sal_uInt16 nValPrefix =
     664             :                             static_cast<sal_uInt16>(
     665          58 :                                     bRename ? (*aIter).second.m_nParam2
     666         140 :                                             : (*aIter).second.m_nParam1);
     667          82 :                         if( AddNamespacePrefix( aAttrValue, nValPrefix ) )
     668          82 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     669             :                     }
     670          82 :                     break;
     671             :                 case XML_ATACTION_ADD_APP_NAMESPACE_PREFIX:
     672             :                     {
     673           0 :                         OUString aAttrValue( rAttrValue );
     674             :                         sal_uInt16 nValPrefix =
     675           0 :                             static_cast<sal_uInt16>((*aIter).second.m_nParam1);
     676           0 :                         if( IsXMLToken( GetClass(), XML_SPREADSHEET  ) )
     677           0 :                             nValPrefix = XML_NAMESPACE_OOOC;
     678           0 :                         else if( IsXMLToken( GetClass(), XML_TEXT  ) )
     679           0 :                             nValPrefix = XML_NAMESPACE_OOOW;
     680           0 :                         if( AddNamespacePrefix( aAttrValue, nValPrefix ) )
     681           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     682             :                     }
     683           0 :                     break;
     684             :                 case XML_ATACTION_RENAME_REMOVE_NAMESPACE_PREFIX:
     685           0 :                     bRename = true;
     686             :                 case XML_ATACTION_REMOVE_NAMESPACE_PREFIX:
     687             :                     {
     688           6 :                         OUString aAttrValue( rAttrValue );
     689             :                         sal_uInt16 nValPrefix =
     690             :                             static_cast<sal_uInt16>(
     691           0 :                                     bRename ? (*aIter).second.m_nParam2
     692           6 :                                             : (*aIter).second.m_nParam1);
     693           6 :                         if( RemoveNamespacePrefix( aAttrValue, nValPrefix ) )
     694           6 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     695             :                     }
     696           6 :                     break;
     697             :                 case XML_ATACTION_REMOVE_ANY_NAMESPACE_PREFIX:
     698             :                     {
     699           0 :                         OUString aAttrValue( rAttrValue );
     700           0 :                         if( RemoveNamespacePrefix( aAttrValue ) )
     701           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     702             :                     }
     703           0 :                     break;
     704             :                 case XML_ATACTION_URI_OOO:
     705             :                     {
     706          38 :                         OUString aAttrValue( rAttrValue );
     707          38 :                         if( ConvertURIToOASIS( aAttrValue,
     708          38 :                             static_cast< bool >((*aIter).second.m_nParam1)))
     709          10 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     710             :                     }
     711          38 :                     break;
     712             :                 case XML_ATACTION_URI_OASIS:
     713             :                     {
     714           0 :                         OUString aAttrValue( rAttrValue );
     715           0 :                         if( ConvertURIToOOo( aAttrValue,
     716           0 :                             static_cast< bool >((*aIter).second.m_nParam1)))
     717           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     718             :                     }
     719           0 :                     break;
     720             :                 case XML_ATACTION_RENAME_ATTRIBUTE:
     721             :                     {
     722           6 :                         OUString aAttrValue( rAttrValue );
     723             :                         RenameAttributeValue(
     724             :                             aAttrValue,
     725           6 :                             (*aIter).second.m_nParam1,
     726           6 :                             (*aIter).second.m_nParam2,
     727          18 :                             (*aIter).second.m_nParam3 );
     728           6 :                         pMutableAttrList->SetValueByIndex( i, aAttrValue );
     729             :                     }
     730           6 :                     break;
     731             :                 case XML_ATACTION_RNG2ISO_DATETIME:
     732             :                     {
     733           0 :                         OUString aAttrValue( rAttrValue );
     734           0 :                         if( ConvertRNGDateTimeToISO( aAttrValue ))
     735           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     736             :                     }
     737           0 :                     break;
     738             :                 case XML_ATACTION_RENAME_RNG2ISO_DATETIME:
     739             :                     {
     740           0 :                         OUString aAttrValue( rAttrValue );
     741           0 :                         if( ConvertRNGDateTimeToISO( aAttrValue ))
     742           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     743           0 :                         bRename = true;
     744             :                     }
     745           0 :                     break;
     746             :                 case XML_ATACTION_IN2TWIPS:
     747             :                     {
     748           0 :                         OUString aAttrValue( rAttrValue );
     749           0 :                         XMLTransformerBase::ReplaceSingleInWithInch( aAttrValue );
     750             : 
     751           0 :                         if( isWriter() )
     752             :                         {
     753           0 :                             sal_Int16 const nDestUnit = lcl_getUnit(aAttrValue);
     754             : 
     755             :                             // convert inch value to twips and export as faked inch
     756             :                             sal_Int32 nMeasure;
     757           0 :                             if (::sax::Converter::convertMeasure(nMeasure,
     758             :                                     aAttrValue, util::MeasureUnit::MM_100TH))
     759             :                             {
     760             : 
     761             :                                 // #i13778#,#i36248#/ apply correct 1/100mm-to-twip conversion
     762           0 :                                 nMeasure = (sal_Int32)( nMeasure >= 0
     763           0 :                                                         ? ((nMeasure*72+63)/127)
     764           0 :                                                         : ((nMeasure*72-63)/127) );
     765             : 
     766           0 :                                 OUStringBuffer aBuffer;
     767             :                                 ::sax::Converter::convertMeasure( aBuffer,
     768             :                                         nMeasure, util::MeasureUnit::MM_100TH,
     769           0 :                                         nDestUnit );
     770           0 :                                 aAttrValue = aBuffer.makeStringAndClear();
     771             :                             }
     772             :                         }
     773             : 
     774           0 :                         pMutableAttrList->SetValueByIndex( i, aAttrValue );
     775             :                     }
     776           0 :                     break;
     777             :                 case XML_ATACTION_SVG_WIDTH_HEIGHT_OOO:
     778             :                     {
     779         204 :                         OUString aAttrValue( rAttrValue );
     780         204 :                         ReplaceSingleInchWithIn( aAttrValue );
     781             : 
     782         204 :                         sal_Int16 const nDestUnit = lcl_getUnit( aAttrValue );
     783             : 
     784             :                         sal_Int32 nMeasure;
     785         204 :                         if (::sax::Converter::convertMeasure(nMeasure,
     786             :                                     aAttrValue, util::MeasureUnit::MM_100TH))
     787             :                         {
     788             : 
     789         204 :                             if( nMeasure > 0 )
     790         204 :                                 nMeasure -= 1;
     791           0 :                             else if( nMeasure < 0 )
     792           0 :                                 nMeasure += 1;
     793             : 
     794             : 
     795         204 :                             OUStringBuffer aBuffer;
     796             :                             ::sax::Converter::convertMeasure(aBuffer, nMeasure,
     797         204 :                                    util::MeasureUnit::MM_100TH, nDestUnit);
     798         204 :                             aAttrValue = aBuffer.makeStringAndClear();
     799             :                         }
     800             : 
     801         204 :                         pMutableAttrList->SetValueByIndex( i, aAttrValue );
     802             :                     }
     803         204 :                     break;
     804             :                 case XML_ATACTION_SVG_WIDTH_HEIGHT_OASIS:
     805             :                     {
     806           0 :                         OUString aAttrValue( rAttrValue );
     807           0 :                         ReplaceSingleInWithInch( aAttrValue );
     808             : 
     809           0 :                         sal_Int16 const nDestUnit = lcl_getUnit( aAttrValue );
     810             : 
     811             :                         sal_Int32 nMeasure;
     812           0 :                         if (::sax::Converter::convertMeasure(nMeasure,
     813             :                                 aAttrValue, util::MeasureUnit::MM_100TH))
     814             :                         {
     815             : 
     816           0 :                             if( nMeasure > 0 )
     817           0 :                                 nMeasure += 1;
     818           0 :                             else if( nMeasure < 0 )
     819           0 :                                 nMeasure -= 1;
     820             : 
     821             : 
     822           0 :                             OUStringBuffer aBuffer;
     823             :                             ::sax::Converter::convertMeasure(aBuffer, nMeasure,
     824           0 :                                     util::MeasureUnit::MM_100TH, nDestUnit );
     825           0 :                             aAttrValue = aBuffer.makeStringAndClear();
     826             :                         }
     827             : 
     828           0 :                         pMutableAttrList->SetValueByIndex( i, aAttrValue );
     829             :                     }
     830           0 :                     break;
     831             :                 case XML_ATACTION_DECODE_ID:
     832             :                     {
     833           0 :                         const sal_Int32 nLen = rAttrValue.getLength();
     834           0 :                         OUStringBuffer aBuffer;
     835             : 
     836             :                         sal_Int32 pos;
     837           0 :                         for( pos = 0; pos < nLen; pos++ )
     838             :                         {
     839           0 :                             sal_Unicode c = rAttrValue[pos];
     840           0 :                             if( (c >= '0') && (c <= '9') )
     841           0 :                                 aBuffer.append( c );
     842             :                             else
     843           0 :                                 aBuffer.append( (sal_Int32)c );
     844             :                         }
     845             : 
     846           0 :                         pMutableAttrList->SetValueByIndex( i, aBuffer.makeStringAndClear() );
     847             :                     }
     848           0 :                     break;
     849             :                 // #i50322# - special handling for the
     850             :                 // transparency of writer background graphics.
     851             :                 case XML_ATACTION_WRITER_BACK_GRAPHIC_TRANSPARENCY:
     852             :                     {
     853             :                         // determine, if it's the transparency of a document style
     854           0 :                         XMLTransformerContext* pFirstContext = (*m_pContexts)[0].get();
     855           0 :                         OUString aFirstContextLocalName;
     856             :                         /* sal_uInt16 nFirstContextPrefix = */
     857           0 :                             GetNamespaceMap().GetKeyByAttrName( pFirstContext->GetQName(),
     858           0 :                                                                 &aFirstContextLocalName );
     859             :                         bool bIsDocumentStyle(
     860             :                             ::xmloff::token::IsXMLToken( aFirstContextLocalName,
     861           0 :                                                          XML_DOCUMENT_STYLES ) );
     862             :                         // no conversion of transparency value for document
     863             :                         // styles, because former OpenOffice.org version writes
     864             :                         // writes always a transparency value of 100% and doesn't
     865             :                         // read the value. Thus, it's intepreted as 0%
     866           0 :                         if ( !bIsDocumentStyle )
     867             :                         {
     868           0 :                             OUString aAttrValue( rAttrValue );
     869           0 :                             NegPercent(aAttrValue);
     870           0 :                             pMutableAttrList->SetValueByIndex( i, aAttrValue );
     871             :                         }
     872           0 :                         bRename = true;
     873             :                     }
     874           0 :                     break;
     875             :                 case XML_ATACTION_SHAPEID:
     876             :                 {
     877           0 :                     OUString sNewValue( "shape"  );
     878           0 :                     sNewValue += rAttrValue;
     879           0 :                     pMutableAttrList->SetValueByIndex( i, sNewValue );
     880           0 :                     break;
     881             :                 }
     882             : 
     883             :                 default:
     884             :                     OSL_ENSURE( false, "unknown action" );
     885           0 :                     break;
     886             :                 }
     887             : 
     888        1596 :                 if( bRename )
     889             :                 {
     890             :                     OUString aNewAttrQName(
     891         506 :                         GetNamespaceMap().GetQNameByKey(
     892         506 :                             (*aIter).second.GetQNamePrefixFromParam1(),
     893             :                             ::xmloff::token::GetXMLToken(
     894        1518 :                                 (*aIter).second.GetQNameTokenFromParam1()) ) );
     895             :                     pMutableAttrList->RenameAttributeByIndex( i,
     896         506 :                                                               aNewAttrQName );
     897             :                 }
     898             :             }
     899        3100 :         }
     900             :     }
     901             : 
     902        1396 :     return pMutableAttrList;
     903             : }
     904             : 
     905        1396 : bool XMLTransformerBase::ReplaceSingleInchWithIn( OUString& rValue )
     906             : {
     907        1396 :     bool bRet = false;
     908        1396 :     sal_Int32 nPos = rValue.getLength();
     909        2792 :     while( nPos && rValue[nPos-1] <= ' ' )
     910           0 :         --nPos;
     911        2792 :     if( nPos > 2 &&
     912        2792 :         ('c'==rValue[nPos-2] || 'C'==rValue[nPos-2]) &&
     913        2236 :         ('h'==rValue[nPos-1] || 'H'==rValue[nPos-1]) )
     914             :     {
     915           0 :         rValue =rValue.copy( 0, nPos-2 );
     916           0 :         bRet = true;
     917             :     }
     918             : 
     919        1396 :     return bRet;
     920             : }
     921             : 
     922          60 : bool XMLTransformerBase::ReplaceInchWithIn( OUString& rValue )
     923             : {
     924          60 :     bool bRet = false;
     925          60 :     sal_Int32 nPos = 1;
     926         852 :     while( nPos < rValue.getLength()-3 )
     927             :     {
     928         732 :         sal_Unicode c = rValue[nPos];
     929         732 :         if( 'i'==c || 'I'==c )
     930             :         {
     931          20 :             c = rValue[nPos-1];
     932          20 :             if( (c >= '0' && c <= '9') || '.' == c )
     933             :             {
     934           0 :                 c = rValue[nPos+1];
     935           0 :                 if( 'n'==c || 'N'==c )
     936             :                 {
     937           0 :                     c = rValue[nPos+2];
     938           0 :                     if( 'c'==c || 'C'==c )
     939             :                     {
     940           0 :                         c = rValue[nPos+3];
     941           0 :                         if( 'h'==c || 'H'==c )
     942             :                         {
     943           0 :                             rValue = rValue.replaceAt( nPos,
     944           0 :                                 4, GetXMLToken(XML_UNIT_INCH) );
     945           0 :                             nPos += 2;
     946           0 :                             bRet = true;
     947           0 :                             continue;
     948             :                         }
     949             :                     }
     950             :                 }
     951             :             }
     952             :         }
     953         732 :         ++nPos;
     954             :     }
     955             : 
     956          60 :     return bRet;
     957             : }
     958             : 
     959         180 : bool XMLTransformerBase::ReplaceSingleInWithInch( OUString& rValue )
     960             : {
     961         180 :     bool bRet = false;
     962             : 
     963         180 :     sal_Int32 nPos = rValue.getLength();
     964         360 :     while( nPos && rValue[nPos-1] <= ' ' )
     965           0 :         --nPos;
     966         360 :     if( nPos > 2 &&
     967         360 :         ('i'==rValue[nPos-2] ||
     968         360 :             'I'==rValue[nPos-2]) &&
     969           0 :         ('n'==rValue[nPos-1] ||
     970           0 :             'N'==rValue[nPos-1]) )
     971             :     {
     972           0 :         nPos -= 2;
     973           0 :         rValue = rValue.replaceAt( nPos, rValue.getLength() - nPos,
     974           0 :                                            GetXMLToken(XML_INCH) );
     975           0 :         bRet = true;
     976             :     }
     977             : 
     978         180 :     return bRet;
     979             : }
     980             : 
     981           2 : bool XMLTransformerBase::ReplaceInWithInch( OUString& rValue )
     982             : {
     983           2 :     bool bRet = false;
     984           2 :     sal_Int32 nPos = 1;
     985           8 :     while( nPos < rValue.getLength()-1 )
     986             :     {
     987           4 :         sal_Unicode c = rValue[nPos];
     988           4 :         if( 'i'==c || 'I'==c )
     989             :         {
     990           0 :             c = rValue[nPos-1];
     991           0 :             if( (c >= '0' && c <= '9') || '.' == c )
     992             :             {
     993           0 :                 c = rValue[nPos+1];
     994           0 :                 if( 'n'==c || 'N'==c )
     995             :                 {
     996           0 :                     rValue = rValue.replaceAt( nPos,
     997           0 :                                     2, GetXMLToken(XML_INCH) );
     998           0 :                     nPos += 4;
     999           0 :                     bRet = true;
    1000           0 :                     continue;
    1001             :                 }
    1002             :             }
    1003             :         }
    1004           4 :         ++nPos;
    1005             :     }
    1006             : 
    1007           2 :     return bRet;
    1008             : }
    1009             : 
    1010        1166 : bool XMLTransformerBase::EncodeStyleName( OUString& rName ) const
    1011             : {
    1012             :     static const sal_Char aHexTab[] = "0123456789abcdef";
    1013             : 
    1014        1166 :     bool bEncoded = false;
    1015             : 
    1016        1166 :     sal_Int32 nLen = rName.getLength();
    1017        1166 :     OUStringBuffer aBuffer( nLen );
    1018             : 
    1019        7194 :     for( sal_Int32 i = 0; i < nLen; i++ )
    1020             :     {
    1021        6028 :         sal_Unicode c = rName[i];
    1022        6028 :         bool bValidChar = false;
    1023        6028 :         if( c < 0x00ffU )
    1024             :         {
    1025             :             bValidChar =
    1026        4870 :                 (c >= 0x0041 && c <= 0x005a) ||
    1027        4102 :                 (c >= 0x0061 && c <= 0x007a) ||
    1028           0 :                 (c >= 0x00c0 && c <= 0x00d6) ||
    1029           0 :                 (c >= 0x00d8 && c <= 0x00f6) ||
    1030       11862 :                 (c >= 0x00f8 && c <= 0x00ff) ||
    1031        1158 :                 ( i > 0 && ( (c >= 0x0030 && c <= 0x0039) ||
    1032        6234 :                              c == 0x00b7 || c == '-' || c == '.') );
    1033             :         }
    1034             :         else
    1035             :         {
    1036           0 :             if( (c >= 0xf900U && c <= 0xfffeU) ||
    1037           0 :                  (c >= 0x20ddU && c <= 0x20e0U))
    1038             :             {
    1039           0 :                 bValidChar = false;
    1040             :             }
    1041           0 :             else if( (c >= 0x02bbU && c <= 0x02c1U) || c == 0x0559 ||
    1042           0 :                      c == 0x06e5 || c == 0x06e6 )
    1043             :             {
    1044           0 :                 bValidChar = true;
    1045             :             }
    1046           0 :             else if( c == 0x0387 )
    1047             :             {
    1048           0 :                 bValidChar = i > 0;
    1049             :             }
    1050             :             else
    1051             :             {
    1052           0 :                 if( !xCharClass.is() )
    1053             :                 {
    1054             :                     const_cast < XMLTransformerBase * >(this)
    1055           0 :                         ->xCharClass = CharacterClassification::create( comphelper::getProcessComponentContext() );
    1056             :                 }
    1057           0 :                 sal_Int16 nType = xCharClass->getType( rName, i );
    1058             : 
    1059           0 :                 switch( nType )
    1060             :                 {
    1061             :                 case UnicodeType::UPPERCASE_LETTER:     // Lu
    1062             :                 case UnicodeType::LOWERCASE_LETTER:     // Ll
    1063             :                 case UnicodeType::TITLECASE_LETTER:     // Lt
    1064             :                 case UnicodeType::OTHER_LETTER:         // Lo
    1065             :                 case UnicodeType::LETTER_NUMBER:        // Nl
    1066           0 :                     bValidChar = true;
    1067           0 :                     break;
    1068             :                 case UnicodeType::NON_SPACING_MARK:     // Ms
    1069             :                 case UnicodeType::ENCLOSING_MARK:       // Me
    1070             :                 case UnicodeType::COMBINING_SPACING_MARK:   //Mc
    1071             :                 case UnicodeType::MODIFIER_LETTER:      // Lm
    1072             :                 case UnicodeType::DECIMAL_DIGIT_NUMBER: // Nd
    1073           0 :                     bValidChar = i > 0;
    1074           0 :                     break;
    1075             :                 }
    1076             :             }
    1077             :         }
    1078        6028 :         if( bValidChar )
    1079             :         {
    1080        5834 :             aBuffer.append( c );
    1081             :         }
    1082             :         else
    1083             :         {
    1084         194 :             aBuffer.append( '_' );
    1085         194 :             if( c > 0x0fff )
    1086             :                 aBuffer.append( static_cast< sal_Unicode >(
    1087           0 :                             aHexTab[ (c >> 12) & 0x0f ]  ) );
    1088         194 :             if( c > 0x00ff )
    1089             :                 aBuffer.append( static_cast< sal_Unicode >(
    1090           0 :                         aHexTab[ (c >> 8) & 0x0f ] ) );
    1091         194 :             if( c > 0x000f )
    1092             :                 aBuffer.append( static_cast< sal_Unicode >(
    1093         194 :                         aHexTab[ (c >> 4) & 0x0f ] ) );
    1094             :             aBuffer.append( static_cast< sal_Unicode >(
    1095         194 :                         aHexTab[ c & 0x0f ] ) );
    1096         194 :             aBuffer.append( '_' );
    1097         194 :             bEncoded = true;
    1098             :         }
    1099             :     }
    1100             : 
    1101        1166 :     if( aBuffer.getLength() > (1<<15)-1 )
    1102           0 :         bEncoded = false;
    1103             : 
    1104        1166 :     if( bEncoded )
    1105         170 :         rName = aBuffer.makeStringAndClear();
    1106        1166 :     return bEncoded;
    1107             : }
    1108             : 
    1109          84 : bool XMLTransformerBase::DecodeStyleName( OUString& rName )
    1110             : {
    1111          84 :     bool bEncoded = false;
    1112             : 
    1113          84 :     sal_Int32 nLen = rName.getLength();
    1114          84 :     OUStringBuffer aBuffer( nLen );
    1115             : 
    1116          84 :     bool bWithinHex = false;
    1117          84 :     sal_Unicode cEnc = 0;
    1118         834 :     for( sal_Int32 i = 0; i < nLen; i++ )
    1119             :     {
    1120         750 :         sal_Unicode c = rName[i];
    1121         750 :         if( '_' == c )
    1122             :         {
    1123          24 :             if( bWithinHex )
    1124             :             {
    1125          12 :                 aBuffer.append( cEnc );
    1126          12 :                 cEnc = 0;
    1127             :             }
    1128             :             else
    1129             :             {
    1130          12 :                 bEncoded = true;
    1131             :             }
    1132          24 :             bWithinHex = !bWithinHex;
    1133             :         }
    1134         726 :         else if( bWithinHex )
    1135             :         {
    1136             :             sal_Unicode cDigit;
    1137          24 :             if( c >= '0' && c <= '9' )
    1138             :             {
    1139          24 :                 cDigit = c - '0';
    1140             :             }
    1141           0 :             else if( c >= 'a' && c <= 'f' )
    1142             :             {
    1143           0 :                 cDigit = c - 'a' + 10;
    1144             :             }
    1145           0 :             else if( c >= 'A' && c <= 'F' )
    1146             :             {
    1147           0 :                 cDigit = c - 'A' + 10;
    1148             :             }
    1149             :             else
    1150             :             {
    1151             :                 // error
    1152           0 :                 bEncoded = false;
    1153           0 :                 break;
    1154             :             }
    1155          24 :             cEnc = (cEnc << 4) + cDigit;
    1156             :         }
    1157             :         else
    1158             :         {
    1159         702 :             aBuffer.append( c );
    1160             :         }
    1161             :     }
    1162             : 
    1163          84 :     if( bEncoded )
    1164           2 :         rName = aBuffer.makeStringAndClear();
    1165          84 :     return bEncoded;
    1166             : }
    1167             : 
    1168          34 : bool XMLTransformerBase::NegPercent( OUString& rValue )
    1169             : {
    1170          34 :     bool bRet = false;
    1171          34 :     bool bNeg = false;
    1172          34 :     double nVal = 0;
    1173             : 
    1174          34 :     sal_Int32 nPos = 0;
    1175          34 :     sal_Int32 nLen = rValue.getLength();
    1176             : 
    1177             :     // skip white space
    1178          68 :     while( nPos < nLen && ' ' == rValue[nPos] )
    1179           0 :         nPos++;
    1180             : 
    1181          34 :     if( nPos < nLen && '-' == rValue[nPos] )
    1182             :     {
    1183           0 :         bNeg = true;
    1184           0 :         nPos++;
    1185             :     }
    1186             : 
    1187             :     // get number
    1188         218 :     while( nPos < nLen &&
    1189         150 :            '0' <= rValue[nPos] &&
    1190          58 :            '9' >= rValue[nPos] )
    1191             :     {
    1192             :         // TODO: check overflow!
    1193          58 :         nVal *= 10;
    1194          58 :         nVal += (rValue[nPos] - '0');
    1195          58 :         nPos++;
    1196             :     }
    1197          34 :     if( nPos < nLen && '.' == rValue[nPos] )
    1198             :     {
    1199           0 :         nPos++;
    1200           0 :         double nDiv = 1.;
    1201             : 
    1202           0 :         while( nPos < nLen &&
    1203           0 :                '0' <= rValue[nPos] &&
    1204           0 :                '9' >= rValue[nPos] )
    1205             :         {
    1206             :             // TODO: check overflow!
    1207           0 :             nDiv *= 10;
    1208           0 :             nVal += ( static_cast<double>(rValue[nPos] - '0') / nDiv );
    1209           0 :             nPos++;
    1210             :         }
    1211             :     }
    1212             : 
    1213             :     // skip white space
    1214          68 :     while( nPos < nLen && ' ' == rValue[nPos] )
    1215           0 :         nPos++;
    1216             : 
    1217          34 :     if( nPos < nLen && '%' == rValue[nPos] )
    1218             :     {
    1219          34 :         if( bNeg )
    1220           0 :                nVal = -nVal;
    1221          34 :         nVal += .5;
    1222             : 
    1223          34 :         sal_Int32 nIntVal = 100 - static_cast<sal_Int32>( nVal );
    1224             : 
    1225          34 :         rValue = OUString::number(nIntVal) + "%";
    1226             : 
    1227          34 :         bRet = true;
    1228             :     }
    1229             : 
    1230          34 :     return bRet;
    1231             : }
    1232             : 
    1233          94 : bool XMLTransformerBase::AddNamespacePrefix( OUString& rName,
    1234             :                              sal_uInt16 nPrefix ) const
    1235             : {
    1236          94 :     rName = GetNamespaceMap().GetQNameByKey( nPrefix, rName, false );
    1237          94 :     return true;
    1238             : }
    1239             : 
    1240           6 : bool XMLTransformerBase::RemoveNamespacePrefix( OUString& rName,
    1241             :                             sal_uInt16 nPrefixOnly ) const
    1242             : {
    1243           6 :     OUString aLocalName;
    1244             :     sal_uInt16 nPrefix =
    1245           6 :         GetNamespaceMap()._GetKeyByAttrName( rName, &aLocalName, false );
    1246          12 :     bool bRet = XML_NAMESPACE_UNKNOWN != nPrefix &&
    1247          12 :                     (USHRT_MAX == nPrefixOnly || nPrefix == nPrefixOnly);
    1248           6 :     if( bRet )
    1249           6 :         rName = aLocalName;
    1250             : 
    1251           6 :     return bRet;
    1252             : }
    1253             : 
    1254          50 : bool XMLTransformerBase::ConvertURIToOASIS( OUString& rURI,
    1255             :                                         bool bSupportPackage ) const
    1256             : {
    1257          50 :     bool bRet = false;
    1258          50 :     if( !m_aExtPathPrefix.isEmpty() && !rURI.isEmpty() )
    1259             :     {
    1260          22 :         bool bRel = false;
    1261          22 :         switch( rURI[0] )
    1262             :         {
    1263             :         case '#':
    1264             :             // no rel path, but
    1265             :             // for package URIs, the '#' has to be removed
    1266          22 :             if( bSupportPackage )
    1267             :             {
    1268          22 :                 rURI = rURI.copy( 1 );
    1269          22 :                 bRet = true;
    1270             :             }
    1271          22 :             break;
    1272             :         case '/':
    1273             :             // no rel path; nothing to do
    1274           0 :             break;
    1275             :         case '.':
    1276             :             // a rel path; to keep URI simple, remove './', if there
    1277           0 :             bRel = true;
    1278           0 :             if( rURI.getLength() > 1 && '/' == rURI[1] )
    1279             :             {
    1280           0 :                 rURI = rURI.copy( 2 );
    1281           0 :                 bRet = true;
    1282             :             }
    1283           0 :             break;
    1284             :         default:
    1285             :             // check for a RFC2396 schema
    1286             :             {
    1287           0 :                 bRel = true;
    1288           0 :                 sal_Int32 nPos = 1;
    1289           0 :                 sal_Int32 nLen = rURI.getLength();
    1290           0 :                 while( nPos < nLen )
    1291             :                 {
    1292           0 :                     switch( rURI[nPos] )
    1293             :                     {
    1294             :                     case '/':
    1295             :                         // a relative path segement
    1296           0 :                         nPos = nLen;    // leave loop
    1297           0 :                         break;
    1298             :                     case ':':
    1299             :                         // a schema
    1300           0 :                         bRel = false;
    1301           0 :                         nPos = nLen;    // leave loop
    1302           0 :                         break;
    1303             :                     default:
    1304             :                         // we don't care about any other characters
    1305           0 :                         break;
    1306             :                     }
    1307           0 :                     ++nPos;
    1308             :                 }
    1309             :             }
    1310             :         }
    1311             : 
    1312          22 :         if( bRel )
    1313             :         {
    1314           0 :             OUString sTmp( m_aExtPathPrefix );
    1315           0 :             sTmp += rURI;
    1316           0 :             rURI = sTmp;
    1317           0 :             bRet = true;
    1318             :         }
    1319             :     }
    1320             : 
    1321          50 :     return bRet;
    1322             : }
    1323             : 
    1324           0 : bool XMLTransformerBase::ConvertURIToOOo( OUString& rURI,
    1325             :                                         bool bSupportPackage ) const
    1326             : {
    1327           0 :     bool bRet = false;
    1328           0 :     if( !rURI.isEmpty() )
    1329             :     {
    1330           0 :         bool bPackage = false;
    1331           0 :         switch( rURI[0] )
    1332             :         {
    1333             :         case '/':
    1334             :             // no rel path; nothing to to
    1335           0 :             break;
    1336             :         case '.':
    1337             :             // a rel path
    1338           0 :             if( rURI.startsWith( m_aExtPathPrefix ) )
    1339             :             {
    1340             :                 // an external URI; remove '../'
    1341           0 :                 rURI = rURI.copy( m_aExtPathPrefix.getLength() );
    1342           0 :                 bRet = true;
    1343             :             }
    1344             :             else
    1345             :             {
    1346           0 :                 bPackage = true;
    1347             :             }
    1348           0 :             break;
    1349             :         default:
    1350             :             // check for a RFC2396 schema
    1351             :             {
    1352           0 :                 bPackage = true;
    1353           0 :                 sal_Int32 nPos = 1;
    1354           0 :                 sal_Int32 nLen = rURI.getLength();
    1355           0 :                 while( nPos < nLen )
    1356             :                 {
    1357           0 :                     switch( rURI[nPos] )
    1358             :                     {
    1359             :                     case '/':
    1360             :                         // a relative path segement within the package
    1361           0 :                         nPos = nLen;    // leave loop
    1362           0 :                         break;
    1363             :                     case ':':
    1364             :                         // a schema
    1365           0 :                         bPackage = false;
    1366           0 :                         nPos = nLen;    // leave loop
    1367           0 :                         break;
    1368             :                     default:
    1369             :                         // we don't care about any other characters
    1370           0 :                         break;
    1371             :                     }
    1372           0 :                     ++nPos;
    1373             :                 }
    1374             :             }
    1375             :         }
    1376             : 
    1377           0 :         if( bPackage && bSupportPackage )
    1378             :         {
    1379           0 :             OUString sTmp( '#' );
    1380           0 :             if( rURI.startsWith( "./" ) )
    1381           0 :                 rURI = rURI.copy( 2 );
    1382           0 :             sTmp += rURI;
    1383           0 :             rURI = sTmp;
    1384           0 :             bRet = true;
    1385             :         }
    1386             :     }
    1387             : 
    1388           0 :     return bRet;
    1389             : }
    1390             : 
    1391           6 : bool XMLTransformerBase::RenameAttributeValue(
    1392             :     OUString& rOutAttributeValue,
    1393             :     sal_Int32 nParam1,
    1394             :     sal_Int32 nParam2,
    1395             :     sal_Int32 nParam3 )
    1396             : {
    1397          12 :     return ( lcl_ConvertAttr( rOutAttributeValue, nParam1) ||
    1398           6 :              lcl_ConvertAttr( rOutAttributeValue, nParam2) ||
    1399           6 :              lcl_ConvertAttr( rOutAttributeValue, nParam3) );
    1400             : }
    1401             : 
    1402             : // static
    1403           0 : bool XMLTransformerBase::ConvertRNGDateTimeToISO( OUString& rDateTime )
    1404             : {
    1405           0 :     if( !rDateTime.isEmpty() &&
    1406           0 :         rDateTime.indexOf( '.' ) != -1 )
    1407             :     {
    1408           0 :         rDateTime = rDateTime.replace( '.', ',');
    1409           0 :         return true;
    1410             :     }
    1411             : 
    1412           0 :     return false;
    1413             : }
    1414             : 
    1415          20 : XMLTokenEnum XMLTransformerBase::GetToken( const OUString& rStr ) const
    1416             : {
    1417             :     XMLTransformerTokenMap::const_iterator aIter =
    1418          20 :         m_pTokenMap->find( rStr );
    1419          20 :     if( aIter == m_pTokenMap->end() )
    1420           0 :         return XML_TOKEN_END;
    1421             :     else
    1422          20 :         return (*aIter).second;
    1423             : }
    1424             : 
    1425             : 
    1426             : 
    1427          24 : const XMLTransformerContext *XMLTransformerBase::GetCurrentContext() const
    1428             : {
    1429             :     OSL_ENSURE( !m_pContexts->empty(), "empty stack" );
    1430             : 
    1431             : 
    1432          24 :     return m_pContexts->empty() ? 0 : m_pContexts->back().get();
    1433             : }
    1434             : 
    1435           0 : const XMLTransformerContext *XMLTransformerBase::GetAncestorContext(
    1436             :                                                         sal_uInt32 n ) const
    1437             : {
    1438             :     XMLTransformerContextVector::size_type nSize =
    1439           0 :         m_pContexts->size();
    1440             :     XMLTransformerContextVector::size_type nPos =
    1441           0 :         static_cast<XMLTransformerContextVector::size_type>( n );
    1442             : 
    1443             :     OSL_ENSURE( nSize >nPos+2 , "invalid context" );
    1444             : 
    1445           0 :     return nSize > nPos+2 ? (*m_pContexts)[nSize-(nPos+2)].get() : 0;
    1446             : }
    1447             : 
    1448          16 : bool XMLTransformerBase::isWriter() const
    1449             : {
    1450          16 :     Reference< XServiceInfo > xSI( mxModel, UNO_QUERY );
    1451          54 :     return  xSI.is() &&
    1452          84 :         (   xSI->supportsService("com.sun.star.text.TextDocument") ||
    1453          56 :             xSI->supportsService("com.sun.star.text.WebDocument") ||
    1454          52 :             xSI->supportsService("com.sun.star.text.GlobalDocument") );
    1455             : }
    1456             : 
    1457             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.10