Files
libreoffice/xmloff/source/text/XMLSectionImportContext.cxx

442 lines
15 KiB
C++
Raw Normal View History

/*************************************************************************
*
* $RCSfile: XMLSectionImportContext.cxx,v $
*
* $Revision: 1.12 $
*
* last change: $Author: dvo $ $Date: 2001-03-21 16:03:49 $
*
* The Contents of this file are made available subject to the terms of
* either of the following licenses
*
* - GNU Lesser General Public License Version 2.1
* - Sun Industry Standards Source License Version 1.1
*
* Sun Microsystems Inc., October, 2000
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2000 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1, as published by the Free Software Foundation.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
*
*
* Sun Industry Standards Source License Version 1.1
* =================================================
* The contents of this file are subject to the Sun Industry Standards
* Source License Version 1.1 (the "License"); You may not use this file
* except in compliance with the License. You may obtain a copy of the
* License at http://www.openoffice.org/license.html.
*
* Software provided under this License is provided on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
* WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
* MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
* See the License for the specific provisions governing your rights and
* obligations concerning the Software.
*
* The Initial Developer of the Original Code is: Sun Microsystems, Inc.
*
* Copyright: 2000 by Sun Microsystems, Inc.
*
* All Rights Reserved.
*
* Contributor(s): _______________________________________
*
*
************************************************************************/
#ifndef _XMLOFF_XMLSECTIONIMPORTCONTEXT_HXX_
#include "XMLSectionImportContext.hxx"
#endif
#ifndef _XMLOFF_XMLSECTIONSOURCEIMPORTCONTEXT_HXX_
#include "XMLSectionSourceImportContext.hxx"
#endif
#ifndef _XMLOFF_XMLSECTIONSOURCEDDEIMPORTCONTEXT_HXX_
#include "XMLSectionSourceDDEImportContext.hxx"
#endif
#ifndef _XMLOFF_XMLICTXT_HXX
#include "xmlictxt.hxx"
#endif
#ifndef _XMLOFF_XMLIMP_HXX
#include "xmlimp.hxx"
#endif
#ifndef _XMLOFF_TEXTIMP_HXX_
#include "txtimp.hxx"
#endif
#ifndef _XMLOFF_NMSPMAP_HXX
#include "nmspmap.hxx"
#endif
#ifndef _XMLOFF_XMLNMSPE_HXX
#include "xmlnmspe.hxx"
#endif
#ifndef _XMLOFF_XMLKYWD_HXX
#include "xmlkywd.hxx"
#endif
#ifndef _XMLOFF_XMLUCONV_HXX
#include "xmluconv.hxx"
#endif
#ifndef _XMLOFF_PRSTYLEI_HXX_
#include "prstylei.hxx"
#endif
#ifndef _COM_SUN_STAR_CONTAINER_XNAMED_HPP_
#include <com/sun/star/container/XNamed.hpp>
#endif
#ifndef _COM_SUN_STAR_UNO_REFERENCE_H_
#include <com/sun/star/uno/Reference.h>
#endif
#ifndef _COM_SUN_STAR_TEXT_XTEXTCONTENT_HPP_
#include <com/sun/star/text/XTextContent.hpp>
#endif
#ifndef _COM_SUN_STAR_BEANS_XPROPERTYSET_HPP_
#include <com/sun/star/beans/XPropertySet.hpp>
#endif
#ifndef _COM_SUN_STAR_LANG_XMULTISERVICEFACTORY_HPP_
#include <com/sun/star/lang/XMultiServiceFactory.hpp>
#endif
#ifndef _COM_SUN_STAR_TEXT_CONTROLCHARACTER_HPP_
#include <com/sun/star/text/ControlCharacter.hpp>
#endif
using ::rtl::OUString;
using ::com::sun::star::beans::XPropertySet;
using ::com::sun::star::uno::Reference;
using ::com::sun::star::xml::sax::XAttributeList;
using ::com::sun::star::lang::XMultiServiceFactory;
using ::com::sun::star::container::XNamed;
using namespace ::com::sun::star::uno;
using namespace ::com::sun::star::text;
TYPEINIT1( XMLSectionImportContext, SvXMLImportContext );
const sal_Char sAPI_TextSection[] = "com.sun.star.text.TextSection";
const sal_Char sAPI_IndexHeaderSection[] = "com.sun.star.text.IndexHeaderSection";
const sal_Char sAPI_IsProtected[] = "IsProtected";
const sal_Char sAPI_Condition[] = "Condition";
const sal_Char sAPI_IsVisible[] = "IsVisible";
const sal_Char sAPI_Password[] = "Password";
enum XMLSectionToken {
XML_TOK_SECTION_STYLE_NAME,
XML_TOK_SECTION_NAME,
XML_TOK_SECTION_CONDITION,
XML_TOK_SECTION_DISPLAY,
XML_TOK_SECTION_PROTECT,
XML_TOK_SECTION_PROTECTION_KEY
};
static __FAR_DATA SvXMLTokenMapEntry aSectionTokenMap[] =
{
{ XML_NAMESPACE_TEXT, sXML_style_name, XML_TOK_SECTION_STYLE_NAME },
{ XML_NAMESPACE_TEXT, sXML_name, XML_TOK_SECTION_NAME },
{ XML_NAMESPACE_TEXT, sXML_condition, XML_TOK_SECTION_CONDITION },
{ XML_NAMESPACE_TEXT, sXML_display, XML_TOK_SECTION_DISPLAY },
{ XML_NAMESPACE_TEXT, sXML_protect, XML_TOK_SECTION_PROTECT },
{ XML_NAMESPACE_TEXT, sXML_protection_key, XML_TOK_SECTION_PROTECTION_KEY},
XML_TOKEN_MAP_END
};
// section import: This one is fairly tricky due to a variety of
// limits of the core or the API. The main problem is that if you
// insert a section within another section, you can't move the cursor
// between the ends of the inner and the enclosing section. To avoid
// these problems, additional markers are first inserted and later deleted.
XMLSectionImportContext::XMLSectionImportContext(
SvXMLImport& rImport,
sal_uInt16 nPrfx,
const OUString& rLocalName ) :
SvXMLImportContext(rImport, nPrfx, rLocalName),
xStartRange(),
xEndRange(),
xSectionPropertySet(),
sTextSection(RTL_CONSTASCII_USTRINGPARAM(sAPI_TextSection)),
sIndexHeaderSection(RTL_CONSTASCII_USTRINGPARAM(
sAPI_IndexHeaderSection)),
sCondition(RTL_CONSTASCII_USTRINGPARAM(sAPI_Condition)),
sIsVisible(RTL_CONSTASCII_USTRINGPARAM(sAPI_IsVisible)),
sPassword(RTL_CONSTASCII_USTRINGPARAM(sAPI_Password)),
sIsProtected(RTL_CONSTASCII_USTRINGPARAM(sAPI_IsProtected)),
sStyleName(),
sName(),
sCond(),
sEmpty(),
bValid(sal_False),
bCondOK(sal_False),
bIsVisible(sal_True),
bSequenceOK(sal_False),
bProtect(sal_False)
{
}
XMLSectionImportContext::~XMLSectionImportContext()
{
}
void XMLSectionImportContext::StartElement(
const Reference<XAttributeList> & xAttrList)
{
// process attributes
ProcessAttributes(xAttrList);
// process index headers:
sal_Bool bIsIndexHeader = GetLocalName().equalsAsciiL(
sXML_index_title, sizeof(sXML_index_title)-1);
if (bIsIndexHeader)
{
bValid = sal_True;
}
UniReference<XMLTextImportHelper> rHelper = GetImport().GetTextImport();
// valid?
if (bValid)
{
// create text section (as XPropertySet)
Reference<XMultiServiceFactory> xFactory(
GetImport().GetModel(),UNO_QUERY);
if (xFactory.is())
{
Reference<XInterface> xIfc =
xFactory->createInstance( bIsIndexHeader ? sIndexHeaderSection
: sTextSection );
if (xIfc.is())
{
Reference<XPropertySet> xPropSet(xIfc, UNO_QUERY);
// save PropertySet (for CreateChildContext)
xSectionPropertySet = xPropSet;
// name
Reference<XNamed> xNamed(xPropSet, UNO_QUERY);
xNamed->setName(sName);
// stylename?
if (sStyleName.getLength() > 0)
{
XMLPropStyleContext* pStyle = rHelper->
FindSectionStyle(sStyleName);
if (pStyle != NULL)
{
pStyle->FillPropertySet( xPropSet );
}
}
// IsVisible and condition (not for index headers)
if (! bIsIndexHeader)
{
Any aAny;
aAny.setValue( &bIsVisible, ::getBooleanCppuType() );
xPropSet->setPropertyValue( sIsVisible, aAny );
if (bCondOK)
{
aAny <<= sCond;
xPropSet->setPropertyValue( sCondition, aAny );
}
}
// password (only for regular sections)
if (bSequenceOK &&
GetLocalName().equalsAsciiL(sXML_section,
sizeof(sXML_section)-1))
{
Any aAny;
aAny <<= aSequence;
xPropSet->setPropertyValue(sPassword, aAny);
}
// protection
Any aAny;
aAny.setValue( &bProtect, ::getBooleanCppuType() );
xPropSet->setPropertyValue( sIsProtected, aAny );
2001-01-19 17:38:06 +00:00
// insert marker, <paragraph>, marker; then insert
// section over the first marker character, and delete the
// last paragraph (and marker) when closing a section.
Reference<XTextRange> xStart =
rHelper->GetCursor()->getStart();
2001-01-19 17:38:06 +00:00
#ifdef PRODUCT
static const sal_Char sMarker[] = " ";
#else
static const sal_Char sMarker[] = "X";
#endif
OUString sMarkerString(RTL_CONSTASCII_USTRINGPARAM(sMarker));
rHelper->InsertString(sMarkerString);
rHelper->InsertControlCharacter(
ControlCharacter::APPEND_PARAGRAPH );
rHelper->InsertString(sMarkerString);
// select first marker
rHelper->GetCursor()->gotoRange(xStart, sal_False);
rHelper->GetCursor()->goRight(1, sal_True);
// convert section to XTextContent
Reference<XTextContent> xTextContent(xSectionPropertySet,
UNO_QUERY);
// and insert (over marker)
rHelper->GetText()->insertTextContent(
rHelper->GetCursorAsRange(), xTextContent, sal_True );
// and delete first marker (in section)
rHelper->GetText()->insertString(
rHelper->GetCursorAsRange(), sEmpty, sal_True);
2001-01-19 17:38:06 +00:00
// finally, check for redlines that should start at
// the section start node
rHelper->RedlineAdjustStartNodeCursor(sal_True); // start ???
}
}
}
}
void XMLSectionImportContext::ProcessAttributes(
const Reference<XAttributeList> & xAttrList )
{
SvXMLTokenMap aTokenMap(aSectionTokenMap);
sal_Int16 nLength = xAttrList->getLength();
for(sal_Int16 nAttr = 0; nAttr < nLength; nAttr++)
{
OUString sLocalName;
sal_uInt16 nPrefix = GetImport().GetNamespaceMap().
GetKeyByAttrName( xAttrList->getNameByIndex(nAttr),
&sLocalName );
OUString sAttr = xAttrList->getValueByIndex(nAttr);
switch (aTokenMap.Get(nPrefix, sLocalName))
{
case XML_TOK_SECTION_STYLE_NAME:
sStyleName = sAttr;
break;
case XML_TOK_SECTION_NAME:
sName = sAttr;
bValid = sal_True;
break;
case XML_TOK_SECTION_CONDITION:
sCond = sAttr;
bCondOK = sal_True;
break;
case XML_TOK_SECTION_DISPLAY:
if (sAttr.equalsAsciiL(sXML_true, sizeof(sXML_true)-1))
{
bIsVisible = sal_True;
}
else if ( sAttr.equalsAsciiL(sXML_none, sizeof(sXML_none)-1) ||
sAttr.equalsAsciiL(sXML_condition,
sizeof(sXML_condition)-1) )
{
bIsVisible = sal_False;
}
// else: ignore
break;
case XML_TOK_SECTION_PROTECTION_KEY:
SvXMLUnitConverter::decodeBase64(aSequence, sAttr);
bSequenceOK = sal_True;
break;
case XML_TOK_SECTION_PROTECT:
{
sal_Bool bTmp;
if (SvXMLUnitConverter::convertBool(bTmp, sAttr))
{
bProtect = bTmp;
}
break;
}
default:
; // ignore
break;
}
}
}
void XMLSectionImportContext::EndElement()
{
// get rid of last paragraph
UniReference<XMLTextImportHelper> rHelper = GetImport().GetTextImport();
rHelper->GetCursor()->goRight(1, sal_False);
rHelper->GetCursor()->goLeft(1, sal_True);
rHelper->GetText()->insertString(rHelper->GetCursorAsRange(),
sEmpty, sal_True);
// and delete second marker
rHelper->GetCursor()->goRight(1, sal_True);
rHelper->GetText()->insertString(rHelper->GetCursorAsRange(),
sEmpty, sal_True);
}
SvXMLImportContext* XMLSectionImportContext::CreateChildContext(
sal_uInt16 nPrefix,
const OUString& rLocalName,
const Reference<XAttributeList> & xAttrList )
{
SvXMLImportContext* pContext = NULL;
// section-source (-dde) elements
if ((XML_NAMESPACE_TEXT == nPrefix) &&
rLocalName.equalsAsciiL(sXML_section_source,
sizeof(sXML_section_source)-1))
{
pContext = new XMLSectionSourceImportContext(GetImport(),
nPrefix, rLocalName,
xSectionPropertySet);
}
else if ((XML_NAMESPACE_OFFICE == nPrefix) &&
rLocalName.equalsAsciiL(sXML_dde_source,
sizeof(sXML_dde_source)-1))
{
pContext = new XMLSectionSourceDDEImportContext(GetImport(),
nPrefix, rLocalName,
xSectionPropertySet);
}
else
{
// otherwise: text context
pContext = GetImport().GetTextImport()->CreateTextChildContext(
2000-11-17 06:37:42 +00:00
GetImport(), nPrefix, rLocalName, xAttrList,
XML_TEXT_TYPE_SECTION );
// if that fails, default context
if (NULL == pContext)
{
pContext = new SvXMLImportContext( GetImport(),
nPrefix, rLocalName );
}
}
return pContext;
}