libreoffice/oox/source/drawingml/chart/plotareacontext.cxx
knordback 95b2c8b82e tdf#165742 Step 4.3: Establish a narrow export path for chartex
This is a subtask of tdf#165742: Chartex charts are lost on input from OOXML
and re-export.

Fix various aspects of input and output for chartex (and specifically
for funnel charts). This now allows for loading and re-exporting a
very basic funnel chart, with the result that MS Office successfully
loads the exported chart, and it looks similar (though not quite
identical) to the original.

Change-Id: I6eeb277e31250031604f7cdd21b80848a9c642ca
Reviewed-on: https://gerrit.libreoffice.org/c/core/+/184758
Reviewed-by: Tomaž Vajngerl <quikee@gmail.com>
Tested-by: Jenkins
2025-05-09 14:41:45 +02:00

254 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 <drawingml/chart/plotareacontext.hxx>
#include <drawingml/shapepropertiescontext.hxx>
#include <drawingml/chart/axiscontext.hxx>
#include <drawingml/chart/plotareamodel.hxx>
#include <drawingml/chart/seriescontext.hxx>
#include <drawingml/chart/typegroupcontext.hxx>
#include <drawingml/chart/datatablecontext.hxx>
#include <oox/core/xmlfilterbase.hxx>
#include <oox/helper/attributelist.hxx>
#include <oox/token/namespaces.hxx>
#include <oox/token/tokens.hxx>
namespace oox::drawingml::chart {
using ::oox::core::ContextHandler2Helper;
using ::oox::core::ContextHandlerRef;
View3DContext::View3DContext( ContextHandler2Helper& rParent, View3DModel& rModel ) :
ContextBase< View3DModel >( rParent, rModel )
{
}
View3DContext::~View3DContext()
{
}
ContextHandlerRef View3DContext::onCreateContext( sal_Int32 nElement, const AttributeList& rAttribs )
{
bool bMSO2007Doc = getFilter().isMSO2007Document();
switch( getCurrentElement() )
{
case C_TOKEN( view3D ):
switch( nElement )
{
case C_TOKEN( depthPercent ):
mrModel.mnDepthPercent = rAttribs.getInteger( XML_val, 100 );
return nullptr;
case C_TOKEN( hPercent ):
mrModel.monHeightPercent = rAttribs.getInteger( XML_val, 100 );
return nullptr;
case C_TOKEN( perspective ):
mrModel.mnPerspective = rAttribs.getInteger( XML_val, 30 );
return nullptr;
case C_TOKEN( rAngAx ):
mrModel.mbRightAngled = rAttribs.getBool( XML_val, !bMSO2007Doc );
return nullptr;
case C_TOKEN( rotX ):
// default value dependent on chart type
mrModel.monRotationX = rAttribs.getInteger( XML_val );
return nullptr;
case C_TOKEN( rotY ):
// default value dependent on chart type
mrModel.monRotationY = rAttribs.getInteger( XML_val );
return nullptr;
}
break;
}
return nullptr;
}
WallFloorContext::WallFloorContext( ContextHandler2Helper& rParent, WallFloorModel& rModel ) :
ContextBase< WallFloorModel >( rParent, rModel )
{
}
WallFloorContext::~WallFloorContext()
{
}
ContextHandlerRef WallFloorContext::onCreateContext( sal_Int32 nElement, const AttributeList& )
{
bool bMSO2007Doc = getFilter().isMSO2007Document();
switch( getCurrentElement() )
{
case C_TOKEN( backWall ):
case C_TOKEN( floor ):
case C_TOKEN( sideWall ):
switch( nElement )
{
case C_TOKEN( pictureOptions ):
return new PictureOptionsContext( *this, mrModel.mxPicOptions.create(bMSO2007Doc) );
case C_TOKEN( spPr ):
return new ShapePropertiesContext( *this, mrModel.mxShapeProp.create() );
}
break;
}
return nullptr;
}
PlotAreaContext::PlotAreaContext( ContextHandler2Helper& rParent, PlotAreaModel& rModel ) :
ContextBase< PlotAreaModel >( rParent, rModel )
{
}
PlotAreaContext::~PlotAreaContext()
{
}
ContextHandlerRef PlotAreaContext::onCreateContext( sal_Int32 nElement, const AttributeList& rAttribs)
{
bool bMSO2007Doc = getFilter().isMSO2007Document();
switch( getCurrentElement() )
{
case C_TOKEN( plotArea ):
switch( nElement )
{
case C_TOKEN( area3DChart ):
case C_TOKEN( areaChart ):
return new AreaTypeGroupContext( *this, mrModel.maTypeGroups.create( nElement, bMSO2007Doc ) );
case C_TOKEN( bar3DChart ):
case C_TOKEN( barChart ):
return new BarTypeGroupContext( *this, mrModel.maTypeGroups.create( nElement, bMSO2007Doc ) );
case C_TOKEN( bubbleChart ):
return new BubbleTypeGroupContext( *this, mrModel.maTypeGroups.create( nElement, bMSO2007Doc ) );
case C_TOKEN( line3DChart ):
case C_TOKEN( lineChart ):
case C_TOKEN( stockChart ):
return new LineTypeGroupContext( *this, mrModel.maTypeGroups.create( nElement, bMSO2007Doc ) );
case C_TOKEN( ofPieChart ):
return new OfPieTypeGroupContext( *this, mrModel.maTypeGroups.create( nElement, bMSO2007Doc ) );
case C_TOKEN( doughnutChart ):
case C_TOKEN( pie3DChart ):
case C_TOKEN( pieChart ):
return new PieTypeGroupContext( *this, mrModel.maTypeGroups.create( nElement, bMSO2007Doc ) );
case C_TOKEN( radarChart ):
return new RadarTypeGroupContext( *this, mrModel.maTypeGroups.create( nElement, bMSO2007Doc ) );
case C_TOKEN( scatterChart ):
return new ScatterTypeGroupContext( *this, mrModel.maTypeGroups.create( nElement, bMSO2007Doc ) );
case C_TOKEN( surface3DChart ):
case C_TOKEN( surfaceChart ):
return new SurfaceTypeGroupContext( *this, mrModel.maTypeGroups.create( nElement, bMSO2007Doc ) );
case C_TOKEN( catAx ):
return new CatAxisContext( *this, mrModel.maAxes.create( nElement, bMSO2007Doc ) );
case C_TOKEN( dateAx ):
return new DateAxisContext( *this, mrModel.maAxes.create( nElement, bMSO2007Doc ) );
case C_TOKEN( serAx ):
return new SerAxisContext( *this, mrModel.maAxes.create( nElement, bMSO2007Doc ) );
case C_TOKEN( valAx ):
return new ValAxisContext( *this, mrModel.maAxes.create( nElement, bMSO2007Doc ) );
case C_TOKEN( layout ):
return new LayoutContext( *this, mrModel.mxLayout.create() );
case C_TOKEN( spPr ):
return new ShapePropertiesContext( *this, mrModel.mxShapeProp.getOrCreate() );
case C_TOKEN(dTable):
return new DataTableContext( *this, mrModel.mxDataTable.create() );
}
break;
case CX_TOKEN(plotArea) :
switch (nElement) {
case CX_TOKEN(plotAreaRegion) :
return this;
case CX_TOKEN(axis) :
// TODO
return nullptr;
case CX_TOKEN(spPr) :
return new ShapePropertiesContext( *this, mrModel.mxShapeProp.getOrCreate() );
case CX_TOKEN(extLst) :
// TODO
return nullptr;
}
break;
case CX_TOKEN(plotAreaRegion) :
switch (nElement) {
case CX_TOKEN(plotSurface) :
// TODO
return nullptr;
case CX_TOKEN(series) :
if (rAttribs.hasAttribute(XML_layoutId)) {
sal_Int32 nChartType = 0;
OUString sChartId = rAttribs.getStringDefaulted(XML_layoutId);
assert(!sChartId.isEmpty());
if (sChartId == "boxWhisker") {
nChartType = CX_TOKEN(boxWhisker);
} else if (sChartId == "clusteredColumn") {
nChartType = CX_TOKEN(clusteredColumn);
} else if (sChartId == "funnel") {
nChartType = CX_TOKEN(funnel);
} else if (sChartId == "paretoLine") {
nChartType = CX_TOKEN(paretoLine);
} else if (sChartId == "regionMap") {
nChartType = CX_TOKEN(regionMap);
} else if (sChartId == "sunburst") {
nChartType = CX_TOKEN(sunburst);
} else if (sChartId == "treemap") {
nChartType = CX_TOKEN(treemap);
} else if (sChartId == "waterfall") {
nChartType = CX_TOKEN(waterfall);
}
assert(nChartType != 0);
// This is a little awkward. The existing parsing
// structures are set up for the ECMA-376 charts, which
// are structured in the XML as
// ...
// <c:plotArea>
// <c:barChart> (or whatever)
// <c:series ... />
// <c:barChart/>
// <c:plotArea/>
//
// By contrast, chartex is like this:
// ...
// <cx:plotArea>
// <cx:plotAreaRegion>
// <cx:series layoutId="funnel" ... /> (or other chart type)
// <cx:plotAreaRegion/>
// <cx:plotArea/>
//
// The best way I've figured out to bridge this
// difference is via the explicit CreateSeries() call
// below, since the structure wants a TypeGroup but
// we're already in the series handling. There may well
// be a better solution.
rtl::Reference<ChartexTypeGroupContext> rTGCtx = new ChartexTypeGroupContext( *this,
mrModel.maTypeGroups.create( nChartType, false ) );
rTGCtx->CreateSeries();
return rTGCtx;
}
break;
}
break;
}
return nullptr;
}
} // namespace oox::drawingml::chart
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */