office-gobmx/xmloff/source/text/XMLIndexTOCContext.cxx
Caolán McNamara 23d7910c1e cid#1554844 COPY_INSTEAD_OF_MOVE
and

cid#1554878 COPY_INSTEAD_OF_MOVE
cid#1554907 COPY_INSTEAD_OF_MOVE
cid#1555082 COPY_INSTEAD_OF_MOVE
cid#1555224 COPY_INSTEAD_OF_MOVE
cid#1555301 COPY_INSTEAD_OF_MOVE
cid#1555378 COPY_INSTEAD_OF_MOVE
cid#1555395 COPY_INSTEAD_OF_MOVE
cid#1555452 COPY_INSTEAD_OF_MOVE
cid#1555466 COPY_INSTEAD_OF_MOVE
cid#1555514 COPY_INSTEAD_OF_MOVE
cid#1555566 COPY_INSTEAD_OF_MOVE
cid#1555753 COPY_INSTEAD_OF_MOVE
cid#1555782 COPY_INSTEAD_OF_MOVE
cid#1555825 COPY_INSTEAD_OF_MOVE
cid#1555842 COPY_INSTEAD_OF_MOVE
cid#1555891 COPY_INSTEAD_OF_MOVE
cid#1555917 COPY_INSTEAD_OF_MOVE
cid#1555961 COPY_INSTEAD_OF_MOVE
cid#1556021 COPY_INSTEAD_OF_MOVE
cid#1556055 COPY_INSTEAD_OF_MOVE
cid#1556307 COPY_INSTEAD_OF_MOVE
cid#1556349 COPY_INSTEAD_OF_MOVE
cid#1556436 COPY_INSTEAD_OF_MOVE
cid#1556665 COPY_INSTEAD_OF_MOVE
cid#1556696 COPY_INSTEAD_OF_MOVE
cid#1556718 COPY_INSTEAD_OF_MOVE
cid#1556775 COPY_INSTEAD_OF_MOVE
cid#1556835 COPY_INSTEAD_OF_MOVE
cid#1556877 COPY_INSTEAD_OF_MOVE
cid#1556994 COPY_INSTEAD_OF_MOVE
cid#1557010 COPY_INSTEAD_OF_MOVE
cid#1557074 COPY_INSTEAD_OF_MOVE
cid#1557111 COPY_INSTEAD_OF_MOVE
cid#1557142 COPY_INSTEAD_OF_MOVE
cid#1557163 COPY_INSTEAD_OF_MOVE
cid#1557164 COPY_INSTEAD_OF_MOVE
cid#1557214 COPY_INSTEAD_OF_MOVE
cid#1557254 COPY_INSTEAD_OF_MOVE
cid#1557322 COPY_INSTEAD_OF_MOVE
cid#1557327 COPY_INSTEAD_OF_MOVE
cid#1557492 COPY_INSTEAD_OF_MOVE
cid#1557565 COPY_INSTEAD_OF_MOVE
cid#1557663 COPY_INSTEAD_OF_MOVE
cid#1557798 COPY_INSTEAD_OF_MOVE

Change-Id: I2ae9c184dcc18d44a65437e18c49e16b27f4d81d
Reviewed-on: https://gerrit.libreoffice.org/c/core/+/174893
Tested-by: Jenkins
Reviewed-by: Caolán McNamara <caolan.mcnamara@collabora.com>
2024-10-14 15:28:24 +02:00

338 lines
11 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* This file incorporates work covered by the following license notice:
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
*/
#include "XMLIndexTOCContext.hxx"
#include <com/sun/star/frame/XModel.hpp>
#include <com/sun/star/lang/XMultiServiceFactory.hpp>
#include <com/sun/star/lang/IllegalArgumentException.hpp>
#include <com/sun/star/uno/XInterface.hpp>
#include <com/sun/star/text/XTextContent.hpp>
#include <com/sun/star/beans/XPropertySet.hpp>
#include <sax/tools/converter.hxx>
#include <sal/log.hxx>
#include "XMLIndexTOCSourceContext.hxx"
#include "XMLIndexObjectSourceContext.hxx"
#include "XMLIndexAlphabeticalSourceContext.hxx"
#include "XMLIndexUserSourceContext.hxx"
#include "XMLIndexBibliographySourceContext.hxx"
#include "XMLIndexTableSourceContext.hxx"
#include "XMLIndexIllustrationSourceContext.hxx"
#include "XMLIndexBodyContext.hxx"
#include <xmloff/xmlictxt.hxx>
#include <xmloff/xmlimp.hxx>
#include <xmloff/txtimp.hxx>
#include <xmloff/xmlnamespace.hxx>
#include <xmloff/xmltoken.hxx>
#include <xmloff/prstylei.hxx>
#include <xmloff/xmlerror.hxx>
#include <xmloff/xmluconv.hxx>
#include <xmloff/xmlement.hxx>
#include <rtl/ustring.hxx>
#include <osl/diagnose.h>
using namespace ::com::sun::star::uno;
using namespace ::com::sun::star::text;
using namespace ::xmloff::token;
using ::com::sun::star::uno::Reference;
using ::com::sun::star::lang::XMultiServiceFactory;
using ::com::sun::star::lang::IllegalArgumentException;
constexpr OUString aIndexServiceMap[]
{
u"com.sun.star.text.ContentIndex"_ustr,
u"com.sun.star.text.DocumentIndex"_ustr,
u"com.sun.star.text.TableIndex"_ustr,
u"com.sun.star.text.ObjectIndex"_ustr,
u"com.sun.star.text.Bibliography"_ustr,
u"com.sun.star.text.UserIndex"_ustr,
u"com.sun.star.text.IllustrationsIndex"_ustr
};
const XMLTokenEnum aIndexSourceElementMap[] =
{
XML_TABLE_OF_CONTENT_SOURCE,
XML_ALPHABETICAL_INDEX_SOURCE,
XML_TABLE_INDEX_SOURCE,
XML_OBJECT_INDEX_SOURCE,
XML_BIBLIOGRAPHY_SOURCE,
XML_USER_INDEX_SOURCE,
XML_ILLUSTRATION_INDEX_SOURCE
};
SvXMLEnumMapEntry<IndexTypeEnum> const aIndexTypeMap[] =
{
{ XML_TABLE_OF_CONTENT, TEXT_INDEX_TOC },
{ XML_ALPHABETICAL_INDEX, TEXT_INDEX_ALPHABETICAL },
{ XML_TABLE_INDEX, TEXT_INDEX_TABLE },
{ XML_OBJECT_INDEX, TEXT_INDEX_OBJECT },
{ XML_BIBLIOGRAPHY, TEXT_INDEX_BIBLIOGRAPHY },
{ XML_USER_INDEX, TEXT_INDEX_USER },
{ XML_ILLUSTRATION_INDEX, TEXT_INDEX_ILLUSTRATION },
{ XML_TOKEN_INVALID, IndexTypeEnum(0) }
};
XMLIndexTOCContext::XMLIndexTOCContext(SvXMLImport& rImport,
sal_Int32 nElement)
: SvXMLImportContext(rImport)
, eIndexType(TEXT_INDEX_UNKNOWN)
, bValid(false)
{
if (IsTokenInNamespace(nElement, XML_NAMESPACE_TEXT))
{
if (SvXMLUnitConverter::convertEnum(eIndexType, SvXMLImport::getNameFromToken(nElement), aIndexTypeMap))
{
// check for array index:
OSL_ENSURE(unsigned(eIndexType) < (SAL_N_ELEMENTS(aIndexServiceMap)), "index out of range");
OSL_ENSURE(SAL_N_ELEMENTS(aIndexServiceMap) ==
SAL_N_ELEMENTS(aIndexSourceElementMap),
"service and source element maps must be same size");
bValid = true;
}
}
}
XMLIndexTOCContext::~XMLIndexTOCContext()
{
}
void XMLIndexTOCContext::startFastElement(
sal_Int32 nElement,
const css::uno::Reference< css::xml::sax::XFastAttributeList >& xAttrList )
{
if (!bValid)
return;
// find text:style-name attribute and set section style
// find text:protected and set value
// find text:name and set value (if not empty)
bool bProtected = false;
OUString sIndexName;
OUString sXmlId;
XMLPropStyleContext* pStyle(nullptr);
for( auto& aIter : sax_fastparser::castToFastAttributeList(xAttrList) )
{
switch(aIter.getToken())
{
case XML_ELEMENT(TEXT, XML_STYLE_NAME):
{
pStyle = GetImport().GetTextImport()->FindSectionStyle(
aIter.toString());
break;
}
case XML_ELEMENT(TEXT, XML_PROTECTED):
{
bool bTmp(false);
if (::sax::Converter::convertBool(bTmp, aIter.toView()))
{
bProtected = bTmp;
}
break;
}
case XML_ELEMENT(TEXT, XML_NAME):
{
sIndexName = aIter.toString();
break;
}
case XML_ELEMENT(XML, XML_ID):
{
sXmlId = aIter.toString();
break;
}
default:
XMLOFF_WARN_UNKNOWN("xmloff", aIter);
}
}
// create table of content (via MultiServiceFactory)
Reference<XMultiServiceFactory> xFactory(GetImport().GetModel(),
UNO_QUERY);
if( xFactory.is() )
{
Reference<XInterface> xIfc = xFactory->createInstance(aIndexServiceMap[eIndexType]);
if( xIfc.is() )
{
// get Property set
xTOCPropertySet.set(xIfc, UNO_QUERY);
// insert section
// a) insert section
// The inserted index consists of an empty paragraph
// only, as well as an empty paragraph *after* the index
// b) insert marker after index, and put Cursor inside of the
// index
// preliminaries
#ifndef DBG_UTIL
static constexpr OUStringLiteral sMarker(u" ");
#else
static constexpr OUStringLiteral sMarker(u"Y");
#endif
rtl::Reference<XMLTextImportHelper> rImport =
GetImport().GetTextImport();
// a) insert index
Reference<XTextContent> xTextContent(xIfc, UNO_QUERY);
try
{
GetImport().GetTextImport()->InsertTextContent(
xTextContent);
}
catch(const IllegalArgumentException& e)
{
// illegal argument? Then we can't accept indices here!
Sequence<OUString> aSeq { SvXMLImport::getNameFromToken(nElement) };
GetImport().SetError(
XMLERROR_FLAG_ERROR | XMLERROR_NO_INDEX_ALLOWED_HERE,
aSeq, e.Message, nullptr );
// set bValid to false, and return prematurely
bValid = false;
return;
}
// xml:id for RDF metadata
GetImport().SetXmlId(xIfc, sXmlId);
// b) insert marker and move cursor
rImport->InsertString(sMarker);
rImport->GetCursor()->goLeft(2, false);
}
}
// finally, check for redlines that should start at
// the section start node
if( bValid )
GetImport().GetTextImport()->RedlineAdjustStartNodeCursor();
if (pStyle != nullptr)
{
pStyle->FillPropertySet( xTOCPropertySet );
}
xTOCPropertySet->setPropertyValue( u"IsProtected"_ustr, Any(bProtected) );
if (!sIndexName.isEmpty())
{
xTOCPropertySet->setPropertyValue( u"Name"_ustr, Any(sIndexName) );
}
}
void XMLIndexTOCContext::endFastElement(sal_Int32 )
{
// complete import of index by removing the markers (if the index
// was actually inserted, that is)
if( !bValid )
return;
// preliminaries
rtl::Reference<XMLTextImportHelper> rHelper= GetImport().GetTextImport();
// get rid of last paragraph (unless it's the only paragraph)
rHelper->GetCursor()->goRight(1, false);
if( xBodyContextRef.is() && xBodyContextRef->HasContent() )
{
rHelper->GetCursor()->goLeft(1, true);
rHelper->GetText()->insertString(rHelper->GetCursorAsRange(),
u""_ustr, true);
}
// and delete second marker
rHelper->GetCursor()->goRight(1, true);
rHelper->GetText()->insertString(rHelper->GetCursorAsRange(),
u""_ustr, true);
// check for Redlines on our end node
GetImport().GetTextImport()->RedlineAdjustStartNodeCursor();
}
css::uno::Reference< css::xml::sax::XFastContextHandler > XMLIndexTOCContext::createFastChildContext(
sal_Int32 nElement,
const css::uno::Reference< css::xml::sax::XFastAttributeList >& )
{
SvXMLImportContextRef xContext;
// not valid -> ignore
if (!bValid)
return nullptr;
if (nElement == XML_ELEMENT(TEXT, XML_INDEX_BODY) )
{
rtl::Reference<XMLIndexBodyContext> xNewBodyContext = new XMLIndexBodyContext(GetImport());
xContext = xNewBodyContext;
if ( !xBodyContextRef.is() || !xBodyContextRef->HasContent() )
{
xBodyContextRef = std::move(xNewBodyContext);
}
}
else if (nElement == XML_ELEMENT(TEXT, aIndexSourceElementMap[eIndexType]))
{
// instantiate source context for the appropriate index type
switch (eIndexType)
{
case TEXT_INDEX_TOC:
xContext = new XMLIndexTOCSourceContext(
GetImport(), xTOCPropertySet);
break;
case TEXT_INDEX_OBJECT:
xContext = new XMLIndexObjectSourceContext(
GetImport(), xTOCPropertySet);
break;
case TEXT_INDEX_ALPHABETICAL:
xContext = new XMLIndexAlphabeticalSourceContext(
GetImport(), xTOCPropertySet);
break;
case TEXT_INDEX_USER:
xContext = new XMLIndexUserSourceContext(
GetImport(), xTOCPropertySet);
break;
case TEXT_INDEX_BIBLIOGRAPHY:
xContext = new XMLIndexBibliographySourceContext(
GetImport(), xTOCPropertySet);
break;
case TEXT_INDEX_TABLE:
xContext = new XMLIndexTableSourceContext(
GetImport(), xTOCPropertySet);
break;
case TEXT_INDEX_ILLUSTRATION:
xContext = new XMLIndexIllustrationSourceContext(
GetImport(), xTOCPropertySet);
break;
default:
OSL_FAIL("index type not implemented");
break;
}
}
// else: ignore
return xContext;
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */