LCOV - code coverage report
Current view: top level - xmloff/source/text - XMLTextMarkImportContext.cxx (source / functions) Hit Total Coverage
Test: commit 10e77ab3ff6f4314137acd6e2702a6e5c1ce1fae Lines: 109 181 60.2 %
Date: 2014-11-03 Functions: 6 15 40.0 %
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 "XMLTextMarkImportContext.hxx"
      21             : 
      22             : 
      23             : #include <rtl/ustring.hxx>
      24             : #include <tools/debug.hxx>
      25             : #include <xmloff/xmluconv.hxx>
      26             : #include <xmloff/xmltoken.hxx>
      27             : #include <xmloff/xmlimp.hxx>
      28             : #include <xmloff/nmspmap.hxx>
      29             : #include <xmloff/xmlnmspe.hxx>
      30             : #include <xmloff/odffields.hxx>
      31             : #include <com/sun/star/xml/sax/XAttributeList.hpp>
      32             : #include <com/sun/star/text/XTextContent.hpp>
      33             : #include <com/sun/star/beans/XPropertySet.hpp>
      34             : #include <com/sun/star/lang/XMultiServiceFactory.hpp>
      35             : #include <com/sun/star/container/XNamed.hpp>
      36             : #include <com/sun/star/rdf/XMetadatable.hpp>
      37             : 
      38             : #include <com/sun/star/text/XFormField.hpp>
      39             : 
      40             : #include "RDFaImportHelper.hxx"
      41             : 
      42             : 
      43             : 
      44             : using namespace ::com::sun::star;
      45             : using namespace ::com::sun::star::text;
      46             : using namespace ::com::sun::star::uno;
      47             : using namespace ::com::sun::star::beans;
      48             : using namespace ::com::sun::star::lang;
      49             : using namespace ::com::sun::star::container;
      50             : using namespace ::com::sun::star::xml::sax;
      51             : using namespace ::xmloff::token;
      52             : 
      53             : 
      54             : 
      55           0 : XMLFieldParamImportContext::XMLFieldParamImportContext(
      56             :     SvXMLImport& rImport,
      57             :     XMLTextImportHelper& rHlp,
      58             :     sal_uInt16 nPrefix,
      59             :     const OUString& rLocalName ) :
      60             :         SvXMLImportContext(rImport, nPrefix, rLocalName),
      61           0 :         rHelper(rHlp)
      62             : {
      63           0 : }
      64             : 
      65             : 
      66           0 : void XMLFieldParamImportContext::StartElement(const ::com::sun::star::uno::Reference< ::com::sun::star::xml::sax::XAttributeList> & xAttrList)
      67             : {
      68           0 :     SvXMLImport& rImport = GetImport();
      69           0 :     OUString sName;
      70           0 :     OUString sValue;
      71             : 
      72           0 :     sal_Int16 nLength = xAttrList->getLength();
      73           0 :     for(sal_Int16 nAttr = 0; nAttr < nLength; nAttr++)
      74             :     {
      75           0 :         OUString sLocalName;
      76           0 :         sal_uInt16 nPrefix = rImport.GetNamespaceMap().
      77           0 :             GetKeyByAttrName( xAttrList->getNameByIndex(nAttr),
      78           0 :                               &sLocalName );
      79             : 
      80           0 :         if ( (XML_NAMESPACE_FIELD == nPrefix) &&
      81           0 :              IsXMLToken(sLocalName, XML_NAME)   )
      82             :         {
      83           0 :             sName = xAttrList->getValueByIndex(nAttr);
      84             :         }
      85           0 :         if ( (XML_NAMESPACE_FIELD == nPrefix) &&
      86           0 :              IsXMLToken(sLocalName, XML_VALUE)   )
      87             :         {
      88           0 :             sValue = xAttrList->getValueByIndex(nAttr);
      89             :         }
      90           0 :     }
      91           0 :     if (rHelper.hasCurrentFieldCtx() && !sName.isEmpty()) {
      92           0 :         rHelper.addFieldParam(sName, sValue);
      93           0 :     }
      94           0 : }
      95             : 
      96             : 
      97           0 : TYPEINIT1( XMLTextMarkImportContext, SvXMLImportContext);
      98             : 
      99        3978 : XMLTextMarkImportContext::XMLTextMarkImportContext(
     100             :     SvXMLImport& rImport,
     101             :     XMLTextImportHelper& rHlp,
     102             :     sal_uInt16 nPrefix,
     103             :     const OUString& rLocalName )
     104             :     : SvXMLImportContext(rImport, nPrefix, rLocalName)
     105             :     , m_rHelper(rHlp)
     106        3978 :     , m_bHaveAbout(false)
     107             : {
     108        3978 : }
     109             : 
     110             : enum lcl_MarkType { TypeReference, TypeReferenceStart, TypeReferenceEnd,
     111             :                     TypeBookmark, TypeBookmarkStart, TypeBookmarkEnd,
     112             :                     TypeFieldmark, TypeFieldmarkStart, TypeFieldmarkEnd
     113             :                   };
     114             : 
     115             : static SvXMLEnumMapEntry const lcl_aMarkTypeMap[] =
     116             : {
     117             :     { XML_REFERENCE_MARK,           TypeReference },
     118             :     { XML_REFERENCE_MARK_START,     TypeReferenceStart },
     119             :     { XML_REFERENCE_MARK_END,       TypeReferenceEnd },
     120             :     { XML_BOOKMARK,                 TypeBookmark },
     121             :     { XML_BOOKMARK_START,           TypeBookmarkStart },
     122             :     { XML_BOOKMARK_END,             TypeBookmarkEnd },
     123             :     { XML_FIELDMARK,                TypeFieldmark },
     124             :     { XML_FIELDMARK_START,          TypeFieldmarkStart },
     125             :     { XML_FIELDMARK_END,            TypeFieldmarkEnd },
     126             :     { XML_TOKEN_INVALID,            0 },
     127             : };
     128             : 
     129             : 
     130          30 : static const char *lcl_getFormFieldmarkName(OUString &name)
     131             : {
     132             :     static const char sCheckbox[]=ODF_FORMCHECKBOX;
     133             :     static const char sFormDropDown[]=ODF_FORMDROPDOWN;
     134          60 :     if (name.equalsAscii("msoffice.field.FORMCHECKBOX") ||
     135          30 :             name.equalsAscii("ecma.office-open-xml.field.FORMCHECKBOX") )
     136           0 :         return sCheckbox;
     137          30 :     else if (name.equalsAscii(ODF_FORMCHECKBOX) )
     138           0 :         return sCheckbox;
     139          60 :     if (name.equalsAscii(ODF_FORMDROPDOWN) ||
     140          30 :             name.equalsAscii("ecma.office-open-xml.field.FORMDROPDOWN") )
     141           0 :         return sFormDropDown;
     142             :     else
     143          30 :         return NULL;
     144             : }
     145             : 
     146           0 : static OUString lcl_getFieldmarkName(OUString &name)
     147             : {
     148             :     static const char sFormtext[]=ODF_FORMTEXT;
     149           0 :     if (name.equalsAscii("msoffice.field.FORMTEXT") ||
     150           0 :             name.equalsAscii("ecma.office-open-xml.field.FORMTEXT") )
     151           0 :         return OUString::createFromAscii(sFormtext);
     152           0 :     else if (name.equalsAscii(ODF_FORMTEXT) )
     153           0 :         return OUString::createFromAscii(sFormtext);
     154             :     else
     155           0 :         return name;
     156             : }
     157             : 
     158             : 
     159        3978 : void XMLTextMarkImportContext::StartElement(
     160             :     const Reference<XAttributeList> & xAttrList)
     161             : {
     162        3978 :     if (!FindName(GetImport(), xAttrList))
     163             :     {
     164           0 :         m_sBookmarkName = OUString();
     165             :     }
     166             : 
     167        3978 :     if (IsXMLToken(GetLocalName(), XML_FIELDMARK_END))
     168             :     {
     169           0 :         m_sBookmarkName = m_rHelper.FindActiveBookmarkName();
     170             :     }
     171             : 
     172        3978 :     if (IsXMLToken(GetLocalName(), XML_FIELDMARK_START) || IsXMLToken(GetLocalName(), XML_FIELDMARK))
     173             :     {
     174           0 :         if (m_sBookmarkName.isEmpty())
     175             :         {
     176           0 :             m_sBookmarkName = "Unknown";
     177             :         }
     178           0 :         m_rHelper.pushFieldCtx( m_sBookmarkName, m_sFieldName );
     179             :     }
     180        3978 : }
     181             : 
     182        3978 : void XMLTextMarkImportContext::EndElement()
     183             : {
     184        3978 :     SvXMLImportContext::EndElement();
     185             : 
     186        3978 :     static const OUString sAPI_reference_mark( "com.sun.star.text.ReferenceMark");
     187        3978 :     static const OUString sAPI_bookmark( "com.sun.star.text.Bookmark");
     188        3978 :     static const OUString sAPI_fieldmark( "com.sun.star.text.Fieldmark");
     189        3978 :     static const OUString sAPI_formfieldmark( "com.sun.star.text.FormFieldmark");
     190             : 
     191        3978 :     if (!m_sBookmarkName.isEmpty())
     192             :     {
     193             :         sal_uInt16 nTmp;
     194        3978 :         if (SvXMLUnitConverter::convertEnum(nTmp, GetLocalName(),
     195             :                                             lcl_aMarkTypeMap))
     196             :         {
     197        3978 :             switch ((lcl_MarkType)nTmp)
     198             :             {
     199             :                 case TypeReference:
     200             :                     // export point reference mark
     201           0 :                     CreateAndInsertMark(GetImport(),
     202             :                         sAPI_reference_mark,
     203             :                         m_sBookmarkName,
     204           0 :                         m_rHelper.GetCursorAsRange()->getStart(),
     205           0 :                         OUString());
     206           0 :                     break;
     207             : 
     208             :                 case TypeFieldmark:
     209             :                 case TypeBookmark:
     210             :                     {
     211          30 :                         const char *formFieldmarkName=lcl_getFormFieldmarkName(m_sFieldName);
     212          30 :                         bool bImportAsField=((lcl_MarkType)nTmp==TypeFieldmark && formFieldmarkName!=NULL); //@TODO handle abbreviation cases..
     213             :                         // export point bookmark
     214             :                         const Reference<XInterface> xContent(
     215          30 :                             CreateAndInsertMark(GetImport(),
     216             :                                         (bImportAsField?sAPI_formfieldmark:sAPI_bookmark),
     217             :                                 m_sBookmarkName,
     218          30 :                                 m_rHelper.GetCursorAsRange()->getStart(),
     219          90 :                                 m_sXmlId) );
     220          30 :                         if ((lcl_MarkType)nTmp==TypeFieldmark) {
     221           0 :                             if (xContent.is() && bImportAsField) {
     222             :                                 // setup fieldmark...
     223           0 :                                 Reference< ::com::sun::star::text::XFormField> xFormField(xContent, UNO_QUERY);
     224           0 :                                 xFormField->setFieldType(OUString::createFromAscii(formFieldmarkName));
     225           0 :                                 if (xFormField.is() && m_rHelper.hasCurrentFieldCtx()) {
     226           0 :                                     m_rHelper.setCurrentFieldParamsTo(xFormField);
     227           0 :                                 }
     228             :                             }
     229           0 :                             m_rHelper.popFieldCtx();
     230          30 :                         }
     231             :                     }
     232          30 :                     break;
     233             : 
     234             :                 case TypeFieldmarkStart:
     235             :                 case TypeBookmarkStart:
     236             :                     // save XTextRange for later construction of bookmark
     237             :                     {
     238             :                         ::boost::shared_ptr< ::xmloff::ParsedRDFaAttributes >
     239        1974 :                             pRDFaAttributes;
     240        1982 :                         if (m_bHaveAbout && (TypeBookmarkStart
     241           8 :                                 == static_cast<lcl_MarkType>(nTmp)))
     242             :                         {
     243          24 :                             pRDFaAttributes =
     244           8 :                                 GetImport().GetRDFaImportHelper().ParseRDFa(
     245             :                                     m_sAbout, m_sProperty,
     246           8 :                                     m_sContent, m_sDatatype);
     247             :                         }
     248             :                         m_rHelper.InsertBookmarkStartRange(
     249             :                             m_sBookmarkName,
     250        1974 :                             m_rHelper.GetCursorAsRange()->getStart(),
     251        3948 :                             m_sXmlId, pRDFaAttributes);
     252             :                     }
     253        1974 :                     break;
     254             : 
     255             :                 case TypeFieldmarkEnd:
     256             :                 case TypeBookmarkEnd:
     257             :                 {
     258             :                     // get old range, and construct
     259        1974 :                     Reference<XTextRange> xStartRange;
     260             :                     ::boost::shared_ptr< ::xmloff::ParsedRDFaAttributes >
     261        3948 :                         pRDFaAttributes;
     262        1974 :                     if (m_rHelper.FindAndRemoveBookmarkStartRange(
     263             :                             m_sBookmarkName, xStartRange,
     264        1974 :                             m_sXmlId, pRDFaAttributes))
     265             :                     {
     266             :                         Reference<XTextRange> xEndRange(
     267        1974 :                             m_rHelper.GetCursorAsRange()->getStart());
     268             : 
     269             :                         // check if beginning and end are in same XText
     270        1974 :                         if (xStartRange->getText() == xEndRange->getText())
     271             :                         {
     272             :                             // create range for insertion
     273             :                             Reference<XTextCursor> xInsertionCursor =
     274        1974 :                                 m_rHelper.GetText()->createTextCursorByRange(
     275        1974 :                                     xEndRange);
     276             :                             try {
     277        1974 :                             xInsertionCursor->gotoRange(xStartRange, sal_True);
     278           0 :                             } catch (uno::Exception&) {
     279             :                                 OSL_ENSURE(false,
     280             :                                     "cannot go to end position of bookmark");
     281             :                             }
     282             : 
     283             :                             //DBG_ASSERT(! xInsertionCursor->isCollapsed(),
     284             :                             //              "we want no point mark");
     285             :                             // can't assert, because someone could
     286             :                             // create a file with subsequence
     287             :                             // start/end elements
     288             : 
     289        1974 :                             bool bImportAsField=((lcl_MarkType)nTmp==TypeFieldmarkEnd && m_rHelper.hasCurrentFieldCtx());
     290             : 
     291             :                             // insert reference
     292             :                             const Reference<XInterface> xContent(
     293        1974 :                                 CreateAndInsertMark(GetImport(),
     294             :                                                 (bImportAsField?sAPI_fieldmark:sAPI_bookmark),
     295             :                                     m_sBookmarkName,
     296             :                                     xInsertionCursor,
     297        5922 :                                     m_sXmlId) );
     298        1974 :                             if (pRDFaAttributes)
     299             :                             {
     300             :                                 const Reference<rdf::XMetadatable>
     301           8 :                                     xMeta(xContent, UNO_QUERY);
     302           8 :                                 GetImport().GetRDFaImportHelper().AddRDFa(
     303           8 :                                     xMeta, pRDFaAttributes);
     304             :                             }
     305             : 
     306        1974 :                             if ((lcl_MarkType)nTmp==TypeFieldmarkEnd) {
     307           0 :                                 if (xContent.is() && bImportAsField) {
     308             :                                     // setup fieldmark...
     309           0 :                                     Reference< ::com::sun::star::text::XFormField> xFormField(xContent, UNO_QUERY);
     310           0 :                                     if (xFormField.is() && m_rHelper.hasCurrentFieldCtx()) {
     311           0 :                                         OUString givenTypeName=m_rHelper.getCurrentFieldType();
     312           0 :                                         OUString fieldmarkTypeName=lcl_getFieldmarkName(givenTypeName);
     313             : 
     314           0 :                                         xFormField->setFieldType(fieldmarkTypeName);
     315           0 :                                         m_rHelper.setCurrentFieldParamsTo(xFormField);
     316           0 :                                     }
     317             :                                 }
     318           0 :                                 m_rHelper.popFieldCtx();
     319        1974 :                             }
     320        1974 :                         }
     321             :                         // else: beginning/end in different XText -> ignore!
     322             :                     }
     323             :                     // else: no start found -> ignore!
     324        3948 :                     break;
     325             :                 }
     326             : 
     327             :                 case TypeReferenceStart:
     328             :                 case TypeReferenceEnd:
     329             :                     OSL_FAIL("reference start/end are handled in txtparai !");
     330           0 :                     break;
     331             : 
     332             :                 default:
     333             :                     OSL_FAIL("unknown mark type");
     334           0 :                     break;
     335             :             }
     336             :         }
     337             :     }
     338        3978 : }
     339             : 
     340           0 : SvXMLImportContext *XMLTextMarkImportContext::CreateChildContext( sal_uInt16 nPrefix,
     341             :                                         const OUString& rLocalName,
     342             :                                         const ::com::sun::star::uno::Reference< ::com::sun::star::xml::sax::XAttributeList >&  )
     343             : {
     344           0 :     return new XMLFieldParamImportContext(GetImport(), m_rHelper,
     345           0 :                 nPrefix, rLocalName);
     346             : }
     347             : 
     348             : 
     349        2080 : Reference<XTextContent> XMLTextMarkImportContext::CreateAndInsertMark(
     350             :     SvXMLImport& rImport,
     351             :     const OUString& sServiceName,
     352             :     const OUString& sMarkName,
     353             :     const Reference<XTextRange> & rRange,
     354             :     const OUString& i_rXmlId)
     355             : {
     356             :     // create mark
     357        2080 :     const Reference<XMultiServiceFactory> xFactory(rImport.GetModel(),
     358        2080 :         UNO_QUERY);
     359        4160 :     Reference<XInterface> xIfc;
     360             : 
     361        2080 :     if (xFactory.is())
     362             :     {
     363        2080 :         xIfc = xFactory->createInstance(sServiceName);
     364             : 
     365        2080 :         if (!xIfc.is())
     366             :         {
     367             :             OSL_FAIL("CreateAndInsertMark: cannot create service?");
     368        2080 :             return 0;
     369             :         }
     370             : 
     371             :         // set name (unless there is no name (text:meta))
     372        2080 :         const Reference<XNamed> xNamed(xIfc, UNO_QUERY);
     373        2080 :         if (xNamed.is())
     374             :         {
     375        2004 :             xNamed->setName(sMarkName);
     376             :         }
     377             :         else
     378             :         {
     379          76 :             if (!sMarkName.isEmpty())
     380             :             {
     381             :                 OSL_FAIL("name given, but XNamed not supported?");
     382           0 :                 return 0;
     383             :             }
     384             :         }
     385             : 
     386             :         // cast to XTextContent and attach to document
     387        2080 :         const Reference<XTextContent> xTextContent(xIfc, UNO_QUERY);
     388        2080 :         if (xTextContent.is())
     389             :         {
     390             :             try
     391             :             {
     392             :                 // if inserting marks, bAbsorb==sal_False will cause
     393             :                 // collapsing of the given XTextRange.
     394        4160 :                 rImport.GetTextImport()->GetText()->insertTextContent(rRange,
     395        2080 :                     xTextContent, sal_True);
     396             : 
     397             :                 // xml:id for RDF metadata -- after insertion!
     398        2080 :                 rImport.SetXmlId(xIfc, i_rXmlId);
     399             : 
     400        2080 :                 return xTextContent;
     401             :             }
     402           0 :             catch (com::sun::star::lang::IllegalArgumentException &)
     403             :             {
     404             :                 OSL_FAIL("CreateAndInsertMark: cannot insert?");
     405           0 :                 return 0;
     406             :             }
     407           0 :         }
     408             :     }
     409        2080 :     return 0;
     410             : }
     411             : 
     412        3978 : bool XMLTextMarkImportContext::FindName(
     413             :     SvXMLImport& rImport,
     414             :     const Reference<XAttributeList> & xAttrList)
     415             : {
     416        3978 :     bool bNameOK = false;
     417             : 
     418             :     // find name attribute first
     419        3978 :     const sal_Int16 nLength = xAttrList->getLength();
     420        8016 :     for(sal_Int16 nAttr = 0; nAttr < nLength; nAttr++)
     421             :     {
     422        4038 :         OUString sLocalName;
     423        4038 :         const sal_uInt16 nPrefix = rImport.GetNamespaceMap().
     424        4038 :             GetKeyByAttrName( xAttrList->getNameByIndex(nAttr),
     425        8076 :                               &sLocalName );
     426             : 
     427        8016 :         if ( (XML_NAMESPACE_TEXT == nPrefix) &&
     428        3978 :              IsXMLToken(sLocalName, XML_NAME)   )
     429             :         {
     430        3978 :             m_sBookmarkName = xAttrList->getValueByIndex(nAttr);
     431        3978 :             bNameOK = true;
     432             :         }
     433          80 :         else if ( (XML_NAMESPACE_XML == nPrefix) &&
     434          20 :              IsXMLToken(sLocalName, XML_ID)   )
     435             :         {
     436          20 :             m_sXmlId = xAttrList->getValueByIndex(nAttr);
     437             :         }
     438          40 :         else if ( XML_NAMESPACE_XHTML == nPrefix )
     439             :         {
     440             :             // RDFa
     441          32 :             if ( IsXMLToken( sLocalName, XML_ABOUT) )
     442             :             {
     443           8 :                 m_sAbout = xAttrList->getValueByIndex(nAttr);
     444           8 :                 m_bHaveAbout = true;
     445             :             }
     446          24 :             else if ( IsXMLToken( sLocalName, XML_PROPERTY) )
     447             :             {
     448           8 :                 m_sProperty = xAttrList->getValueByIndex(nAttr);
     449             :             }
     450          16 :             else if ( IsXMLToken( sLocalName, XML_CONTENT) )
     451             :             {
     452           8 :                 m_sContent = xAttrList->getValueByIndex(nAttr);
     453             :             }
     454           8 :             else if ( IsXMLToken( sLocalName, XML_DATATYPE) )
     455             :             {
     456           8 :                 m_sDatatype = xAttrList->getValueByIndex(nAttr);
     457             :             }
     458             :         }
     459           8 :         else if ( (XML_NAMESPACE_FIELD == nPrefix) &&
     460           0 :              IsXMLToken(sLocalName, XML_TYPE)   )
     461             :         {
     462           0 :             m_sFieldName = xAttrList->getValueByIndex(nAttr);
     463             :         }
     464        4038 :     }
     465             : 
     466        3978 :     return bNameOK;
     467             : }
     468             : 
     469             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.10