23d7910c1e
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>
338 lines
11 KiB
C++
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: */
|