office-gobmx/xmloff/source/chart/SchXMLImport.cxx
Noel Grandin 12008e1054 loplugin:ostr in xmloff
Change-Id: I24f5626e57e03217e4bcf77289c785e255b9f904
Reviewed-on: https://gerrit.libreoffice.org/c/core/+/167667
Reviewed-by: Noel Grandin <noel.grandin@collabora.co.uk>
Tested-by: Jenkins
2024-05-15 20:37:38 +02:00

396 lines
15 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 <SchXMLImport.hxx>
#include "SchXMLChartContext.hxx"
#include "contexts.hxx"
#include "SchXMLTools.hxx"
#include <sal/log.hxx>
#include <comphelper/processfactory.hxx>
#include <utility>
#include <xmloff/namespacemap.hxx>
#include <xmloff/prstylei.hxx>
#include <xmloff/xmlnamespace.hxx>
#include <xmloff/xmltoken.hxx>
#include <xmloff/xmluconv.hxx>
#include <xmloff/xmlictxt.hxx>
#include <xmloff/xmlstyle.hxx>
#include <com/sun/star/chart/XChartDocument.hpp>
#include <com/sun/star/container/XChild.hpp>
#include <com/sun/star/lang/XMultiServiceFactory.hpp>
#include <com/sun/star/uno/XComponentContext.hpp>
#include <com/sun/star/chart2/data/XDataReceiver.hpp>
#include <com/sun/star/chart2/XChartDocument.hpp>
#include <com/sun/star/chart2/XCoordinateSystemContainer.hpp>
#include <com/sun/star/chart2/XChartTypeContainer.hpp>
#include <com/sun/star/chart2/XDataSeriesContainer.hpp>
#include <com/sun/star/document/XDocumentPropertiesSupplier.hpp>
#include <comphelper/diagnose_ex.hxx>
using namespace com::sun::star;
using namespace ::xmloff::token;
using ::com::sun::star::uno::Reference;
using ::com::sun::star::uno::Sequence;
namespace
{
class lcl_MatchesChartType
{
public:
explicit lcl_MatchesChartType( OUString aChartTypeName ) :
m_aChartTypeName(std::move( aChartTypeName ))
{}
bool operator () ( const Reference< chart2::XChartType > & xChartType ) const
{
return (xChartType.is() &&
xChartType->getChartType() == m_aChartTypeName );
}
private:
OUString m_aChartTypeName;
};
} // anonymous namespace
// TokenMaps for distinguishing different
// tokens in different contexts
// element maps
// attribute maps
SchXMLImportHelper::SchXMLImportHelper() :
mpAutoStyles( nullptr )
{
}
SvXMLImportContext* SchXMLImportHelper::CreateChartContext(
SvXMLImport& rImport,
const Reference< frame::XModel >& rChartModel )
{
SvXMLImportContext* pContext = nullptr;
Reference< chart::XChartDocument > xDoc( rChartModel, uno::UNO_QUERY );
if( xDoc.is())
{
mxChartDoc = xDoc;
pContext = new SchXMLChartContext( *this, rImport );
}
else
{
SAL_WARN("xmloff.chart", "No valid XChartDocument given as XModel" );
}
return pContext;
}
void SchXMLImportHelper::FillAutoStyle(const OUString& rAutoStyleName, const uno::Reference<beans::XPropertySet>& rProp)
{
if (!rProp.is())
return;
const SvXMLStylesContext* pStylesCtxt = GetAutoStylesContext();
if (pStylesCtxt)
{
SvXMLStyleContext* pStyle = const_cast<SvXMLStyleContext*>(pStylesCtxt->FindStyleChildContext(SchXMLImportHelper::GetChartFamilyID(), rAutoStyleName));
if (XMLPropStyleContext* pPropStyle = dynamic_cast<XMLPropStyleContext*>(pStyle))
pPropStyle->FillPropertySet(rProp);
}
}
// get various token maps
//static
void SchXMLImportHelper::DeleteDataSeries(
const Reference< chart2::XDataSeries > & xSeries,
const Reference< chart2::XChartDocument > & xDoc )
{
if( !xDoc.is() )
return;
try
{
Reference< chart2::XCoordinateSystemContainer > xCooSysCnt(
xDoc->getFirstDiagram(), uno::UNO_QUERY_THROW );
const Sequence< Reference< chart2::XCoordinateSystem > > aCooSysSeq(
xCooSysCnt->getCoordinateSystems());
for( const auto& rCooSys : aCooSysSeq )
{
Reference< chart2::XChartTypeContainer > xCTCnt( rCooSys, uno::UNO_QUERY_THROW );
const Sequence< Reference< chart2::XChartType > > aChartTypes( xCTCnt->getChartTypes());
for( const auto& rChartType : aChartTypes )
{
Reference< chart2::XDataSeriesContainer > xSeriesCnt( rChartType, uno::UNO_QUERY_THROW );
const Sequence< Reference< chart2::XDataSeries > > aSeriesSeq( xSeriesCnt->getDataSeries());
if (std::find(aSeriesSeq.begin(), aSeriesSeq.end(), xSeries) != aSeriesSeq.end())
{
xSeriesCnt->removeDataSeries(xSeries);
return;
}
}
}
}
catch( const uno::Exception &)
{
DBG_UNHANDLED_EXCEPTION("xmloff.chart");
}
}
// static
Reference< chart2::XDataSeries > SchXMLImportHelper::GetNewDataSeries(
const Reference< chart2::XChartDocument > & xDoc,
sal_Int32 nCoordinateSystemIndex,
const OUString & rChartTypeName,
bool bPushLastChartType /* = false */ )
{
Reference< chart2::XDataSeries > xResult;
if(!xDoc.is())
return xResult;
try
{
Reference< chart2::XCoordinateSystemContainer > xCooSysCnt(
xDoc->getFirstDiagram(), uno::UNO_QUERY_THROW );
Sequence< Reference< chart2::XCoordinateSystem > > aCooSysSeq(
xCooSysCnt->getCoordinateSystems());
Reference< uno::XComponentContext > xContext(
comphelper::getProcessComponentContext() );
if( nCoordinateSystemIndex < aCooSysSeq.getLength())
{
Reference< chart2::XChartType > xCurrentType;
{
Reference< chart2::XChartTypeContainer > xCTCnt( aCooSysSeq[ nCoordinateSystemIndex ], uno::UNO_QUERY_THROW );
Sequence< Reference< chart2::XChartType > > aChartTypes( xCTCnt->getChartTypes());
// find matching chart type group
const Reference< chart2::XChartType > * pBegin = aChartTypes.getConstArray();
const Reference< chart2::XChartType > * pEnd = pBegin + aChartTypes.getLength();
const Reference< chart2::XChartType > * pIt =
::std::find_if( pBegin, pEnd, lcl_MatchesChartType( rChartTypeName ));
if( pIt != pEnd )
xCurrentType.set( *pIt );
// if chart type is set at series and differs from current one,
// create a new chart type
if( !xCurrentType.is())
{
xCurrentType.set(
xContext->getServiceManager()->createInstanceWithContext( rChartTypeName, xContext ),
uno::UNO_QUERY );
if( xCurrentType.is())
{
if( bPushLastChartType && aChartTypes.hasElements())
{
sal_Int32 nIndex( aChartTypes.getLength() - 1 );
aChartTypes.realloc( aChartTypes.getLength() + 1 );
auto pChartTypes = aChartTypes.getArray();
pChartTypes[ nIndex + 1 ] = aChartTypes[ nIndex ];
pChartTypes[ nIndex ] = xCurrentType;
xCTCnt->setChartTypes( aChartTypes );
}
else
xCTCnt->addChartType( xCurrentType );
}
}
}
if( xCurrentType.is())
{
Reference< chart2::XDataSeriesContainer > xSeriesCnt( xCurrentType, uno::UNO_QUERY_THROW );
if( xContext.is() )
{
xResult.set(
xContext->getServiceManager()->createInstanceWithContext(
u"com.sun.star.chart2.DataSeries"_ustr,
xContext ), uno::UNO_QUERY_THROW );
}
if( xResult.is() )
xSeriesCnt->addDataSeries( xResult );
}
}
}
catch( const uno::Exception & )
{
DBG_UNHANDLED_EXCEPTION("xmloff.chart");
}
return xResult;
}
SchXMLImport::SchXMLImport(
const Reference< uno::XComponentContext >& xContext,
OUString const & implementationName, SvXMLImportFlags nImportFlags ) :
SvXMLImport( xContext, implementationName, nImportFlags ),
maImportHelper(new SchXMLImportHelper)
{
GetNamespaceMap().Add( GetXMLToken(XML_NP_XLINK), GetXMLToken(XML_N_XLINK), XML_NAMESPACE_XLINK );
GetNamespaceMap().Add( GetXMLToken(XML_NP_CHART_EXT), GetXMLToken(XML_N_CHART_EXT), XML_NAMESPACE_CHART_EXT);
}
SchXMLImport::~SchXMLImport() noexcept
{
uno::Reference< chart2::XChartDocument > xChartDoc( GetModel(), uno::UNO_QUERY );
if (xChartDoc.is())
{
if (xChartDoc->hasControllersLocked())
xChartDoc->unlockControllers();
if (auto xPropSet = xChartDoc.query<beans::XPropertySet>())
{
try
{
// The view of the chart might not received a notification about the updates,
// which is only sent when the chart model is set modified; during the load,
// setting modified is disabled. So, the view needs an explicit notification.
// See ChartDocumentWrapper::setPropertyValue
xPropSet->setPropertyValue(u"ODFImport_UpdateView"_ustr, css::uno::Any(xChartDoc));
}
catch (css::beans::UnknownPropertyException&)
{
// That's absolutely fine!
}
}
}
}
// create the main context (subcontexts are created
// by the one created here)
SvXMLImportContext *SchXMLImport::CreateFastContext( sal_Int32 nElement,
const uno::Reference< xml::sax::XFastAttributeList >& /*xAttrList*/ )
{
SvXMLImportContext* pContext = nullptr;
switch (nElement)
{
case XML_ELEMENT( OFFICE, XML_DOCUMENT ):
case XML_ELEMENT( OFFICE, XML_DOCUMENT_META ):
{
uno::Reference<document::XDocumentPropertiesSupplier> xDPS(
GetModel(), uno::UNO_QUERY);
// mst@: right now, this seems to be not supported, so it is untested
if (xDPS.is()) {
pContext = (nElement == XML_ELEMENT(OFFICE, XML_DOCUMENT_META))
? new SvXMLMetaDocumentContext(*this, xDPS->getDocumentProperties())
// flat OpenDocument file format
: new SchXMLFlatDocContext_Impl(*maImportHelper, *this, nElement,
xDPS->getDocumentProperties());
}
}
break;
// accept <office:document>
case XML_ELEMENT(OFFICE, XML_DOCUMENT_STYLES):
case XML_ELEMENT(OFFICE, XML_DOCUMENT_CONTENT):
pContext = new SchXMLDocContext(*maImportHelper, *this, nElement);
break;
}
return pContext;
}
SvXMLImportContext* SchXMLImport::CreateStylesContext()
{
//#i103287# make sure that the version information is set before importing all the properties (especially stroke-opacity!)
SchXMLTools::setBuildIDAtImportInfo( GetModel(), getImportInfo() );
SvXMLStylesContext* pStylesCtxt = new SvXMLStylesContext( *this );
// set context at base class, so that all auto-style classes are imported
SetAutoStyles( pStylesCtxt );
maImportHelper->SetAutoStylesContext( pStylesCtxt );
return pStylesCtxt;
}
void SAL_CALL SchXMLImport::setTargetDocument(const uno::Reference<lang::XComponent>& xDoc)
{
uno::Reference<chart2::XChartDocument> xOldDoc(GetModel(), uno::UNO_QUERY);
if (xOldDoc.is() && xOldDoc->hasControllersLocked())
xOldDoc->unlockControllers();
SvXMLImport::setTargetDocument(xDoc);
uno::Reference<chart2::XChartDocument> xChartDoc(GetModel(), uno::UNO_QUERY);
if (!xChartDoc.is())
return;
try
{
// prevent rebuild of view during load (necessary especially if loaded not
// via load api, which is the case for example if binary files are loaded)
xChartDoc->lockControllers();
uno::Reference<container::XChild> xChild(xChartDoc, uno::UNO_QUERY);
uno::Reference<chart2::data::XDataReceiver> xDataReceiver(xChartDoc, uno::UNO_QUERY);
if (xChild.is() && xDataReceiver.is())
{
Reference<lang::XMultiServiceFactory> xFact(xChild->getParent(), uno::UNO_QUERY);
if (xFact.is())
{
//if the parent has a number formatter we will use the numberformatter of the parent
Reference<util::XNumberFormatsSupplier> xNumberFormatsSupplier(xFact, uno::UNO_QUERY);
xDataReceiver->attachNumberFormatsSupplier(xNumberFormatsSupplier);
}
}
}
catch (const uno::Exception &)
{
TOOLS_INFO_EXCEPTION("xmloff.chart", "SchXMLChartContext::StartElement(): Exception caught");
}
}
// first version: everything comes from one storage
extern "C" SAL_DLLPUBLIC_EXPORT uno::XInterface*
com_sun_star_comp_Chart_XMLOasisImporter_get_implementation(uno::XComponentContext* pCtx,
uno::Sequence<uno::Any> const& /*rSeq*/)
{
return cppu::acquire(new SchXMLImport(pCtx, u"SchXMLImport"_ustr, SvXMLImportFlags::ALL));
}
// multiple storage version: one for content / styles / meta
extern "C" SAL_DLLPUBLIC_EXPORT uno::XInterface*
com_sun_star_comp_Chart_XMLOasisMetaImporter_get_implementation(
uno::XComponentContext* pCtx, uno::Sequence<uno::Any> const& /*rSeq*/)
{
return cppu::acquire(new SchXMLImport(pCtx, u"SchXMLImport.Meta"_ustr, SvXMLImportFlags::META));
}
extern "C" SAL_DLLPUBLIC_EXPORT uno::XInterface*
com_sun_star_comp_Chart_XMLOasisStylesImporter_get_implementation(
uno::XComponentContext* pCtx, uno::Sequence<uno::Any> const& /*rSeq*/)
{
return cppu::acquire(new SchXMLImport(pCtx, u"SchXMLImport.Styles"_ustr, SvXMLImportFlags::STYLES));
}
extern "C" SAL_DLLPUBLIC_EXPORT uno::XInterface*
com_sun_star_comp_Chart_XMLOasisContentImporter_get_implementation(
uno::XComponentContext* pCtx, uno::Sequence<uno::Any> const& /*rSeq*/)
{
return cppu::acquire(new SchXMLImport(pCtx, u"SchXMLImport.Content"_ustr,
SvXMLImportFlags::CONTENT | SvXMLImportFlags::AUTOSTYLES
| SvXMLImportFlags::FONTDECLS));
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */