office-gobmx/oox/source/drawingml/chart/objectformatter.cxx
Takeshi Abe 96cb767123 Prefer equalsIgnoreAsciiCaseAsciiL(RTL_CONSTASCII_STRINGPARAM("..."))
to equalsIgnoreAsciiCaseAscii("...")
2012-02-19 23:28:55 +09:00

1181 lines
55 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*************************************************************************
*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* Copyright 2000, 2010 Oracle and/or its affiliates.
*
* OpenOffice.org - a multi-platform office productivity suite
*
* This file is part of OpenOffice.org.
*
* OpenOffice.org is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License version 3
* only, as published by the Free Software Foundation.
*
* OpenOffice.org 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 version 3 for more details
* (a copy is included in the LICENSE file that accompanied this code).
*
* You should have received a copy of the GNU Lesser General Public License
* version 3 along with OpenOffice.org. If not, see
* <http://www.openoffice.org/license.html>
* for a copy of the LGPLv3 License.
*
************************************************************************/
#include "oox/drawingml/chart/objectformatter.hxx"
#include <com/sun/star/util/XNumberFormatsSupplier.hpp>
#include <com/sun/star/util/XNumberFormatTypes.hpp>
#include <osl/thread.h>
#include <rtl/strbuf.hxx>
#include "oox/core/xmlfilterbase.hxx"
#include "oox/drawingml/fillproperties.hxx"
#include "oox/drawingml/lineproperties.hxx"
#include "oox/drawingml/shapepropertymap.hxx"
#include "oox/drawingml/textbody.hxx"
#include "oox/drawingml/textparagraph.hxx"
#include "oox/drawingml/theme.hxx"
#include "oox/drawingml/chart/chartspacemodel.hxx"
#include "oox/helper/modelobjecthelper.hxx"
namespace oox {
namespace drawingml {
namespace chart {
// ============================================================================
using namespace ::com::sun::star::chart2;
using namespace ::com::sun::star::frame;
using namespace ::com::sun::star::graphic;
using namespace ::com::sun::star::lang;
using namespace ::com::sun::star::uno;
using namespace ::com::sun::star::util;
using ::oox::core::XmlFilterBase;
using ::rtl::OStringBuffer;
using ::rtl::OUString;
using ::rtl::OUStringToOString;
// ============================================================================
namespace {
struct AutoFormatPatternEntry
{
sal_Int32 mnColorToken; /// Theme color token.
sal_Int32 mnModToken; /// Color modification token.
sal_Int32 mnModValue; /// Color modification value.
};
#define AUTOFORMAT_PATTERN_COLOR( color_token ) \
{ color_token, XML_TOKEN_INVALID, 0 }
#define AUTOFORMAT_PATTERN_COLORMOD( color_token, mod_token, mod_value ) \
{ color_token, mod_token, mod_value }
#define AUTOFORMAT_PATTERN_END() \
AUTOFORMAT_PATTERN_COLOR( XML_TOKEN_INVALID )
static const AutoFormatPatternEntry spAutoFormatPattern1[] =
{
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 88500 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 55000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 78000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 92500 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 70000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 30000 ),
AUTOFORMAT_PATTERN_END()
};
static const AutoFormatPatternEntry spAutoFormatPattern2[] =
{
AUTOFORMAT_PATTERN_COLOR( XML_accent1 ),
AUTOFORMAT_PATTERN_COLOR( XML_accent2 ),
AUTOFORMAT_PATTERN_COLOR( XML_accent3 ),
AUTOFORMAT_PATTERN_COLOR( XML_accent4 ),
AUTOFORMAT_PATTERN_COLOR( XML_accent5 ),
AUTOFORMAT_PATTERN_COLOR( XML_accent6 ),
AUTOFORMAT_PATTERN_END()
};
static const AutoFormatPatternEntry spAutoFormatPattern3[] =
{
AUTOFORMAT_PATTERN_COLORMOD( XML_accent1, XML_shade, 50000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_accent2, XML_shade, 50000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_accent3, XML_shade, 50000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_accent4, XML_shade, 50000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_accent5, XML_shade, 50000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_accent6, XML_shade, 50000 ),
AUTOFORMAT_PATTERN_END()
};
static const AutoFormatPatternEntry spAutoFormatPattern4[] =
{
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 5000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 55000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 78000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 15000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 70000 ),
AUTOFORMAT_PATTERN_COLORMOD( XML_dk1, XML_tint, 30000 ),
AUTOFORMAT_PATTERN_END()
};
#undef AUTOFORMAT_PATTERN_COLOR
#undef AUTOFORMAT_PATTERN_COLORMOD
#undef AUTOFORMAT_PATTERN_END
// ----------------------------------------------------------------------------
struct AutoFormatEntry
{
sal_Int32 mnFirstStyleIdx; /// First chart style index.
sal_Int32 mnLastStyleIdx; /// Last chart style index.
sal_Int32 mnThemedIdx; /// Themed style index.
sal_Int32 mnColorToken; /// Theme color token.
sal_Int32 mnModToken; /// Color modification token.
sal_Int32 mnModValue; /// Color modification value.
sal_Int32 mnRelLineWidth; /// Relative line width (percent).
const AutoFormatPatternEntry* mpPattern;/// Color cycling pattern for data series.
bool mbFadedColor; /// True = Faded color for data series.
};
#define AUTOFORMAT_COLOR( first, last, themed_style, color_token ) \
{ first, last, themed_style, color_token, XML_TOKEN_INVALID, 0, 100, 0, false }
#define AUTOFORMAT_ACCENTS( first, themed_style ) \
AUTOFORMAT_COLOR( first, first, themed_style, XML_accent1 ), \
AUTOFORMAT_COLOR( first + 1, first + 1, themed_style, XML_accent2 ), \
AUTOFORMAT_COLOR( first + 2, first + 2, themed_style, XML_accent3 ), \
AUTOFORMAT_COLOR( first + 3, first + 3, themed_style, XML_accent4 ), \
AUTOFORMAT_COLOR( first + 4, first + 4, themed_style, XML_accent5 ), \
AUTOFORMAT_COLOR( first + 5, first + 5, themed_style, XML_accent6 )
#define AUTOFORMAT_COLORMOD( first, last, themed_style, color_token, mod_token, mod_value ) \
{ first, last, themed_style, color_token, mod_token, mod_value, 100, 0, false }
#define AUTOFORMAT_ACCENTSMOD( first, themed_style, mod_token, mod_value ) \
AUTOFORMAT_COLORMOD( first, first, themed_style, XML_accent1, mod_token, mod_value ), \
AUTOFORMAT_COLORMOD( first + 1, first + 1, themed_style, XML_accent2, mod_token, mod_value ), \
AUTOFORMAT_COLORMOD( first + 2, first + 2, themed_style, XML_accent3, mod_token, mod_value ), \
AUTOFORMAT_COLORMOD( first + 3, first + 3, themed_style, XML_accent4, mod_token, mod_value ), \
AUTOFORMAT_COLORMOD( first + 4, first + 4, themed_style, XML_accent5, mod_token, mod_value ), \
AUTOFORMAT_COLORMOD( first + 5, first + 5, themed_style, XML_accent6, mod_token, mod_value )
#define AUTOFORMAT_PATTERN( first, last, themed_style, line_width, pattern ) \
{ first, last, themed_style, XML_TOKEN_INVALID, XML_TOKEN_INVALID, 0, line_width, pattern, false }
#define AUTOFORMAT_FADED( first, last, themed_style, color_token, line_width ) \
{ first, last, themed_style, color_token, XML_TOKEN_INVALID, 0, line_width, 0, true }
#define AUTOFORMAT_FADEDACCENTS( first, themed_style, line_width ) \
AUTOFORMAT_FADED( first, first, themed_style, XML_accent1, line_width ), \
AUTOFORMAT_FADED( first + 1, first + 1, themed_style, XML_accent2, line_width ), \
AUTOFORMAT_FADED( first + 2, first + 2, themed_style, XML_accent3, line_width ), \
AUTOFORMAT_FADED( first + 3, first + 3, themed_style, XML_accent4, line_width ), \
AUTOFORMAT_FADED( first + 4, first + 4, themed_style, XML_accent5, line_width ), \
AUTOFORMAT_FADED( first + 5, first + 5, themed_style, XML_accent6, line_width )
#define AUTOFORMAT_INVISIBLE( first, last ) \
AUTOFORMAT_COLOR( first, last, -1, XML_TOKEN_INVALID )
#define AUTOFORMAT_END() \
AUTOFORMAT_INVISIBLE( -1, -1 )
static const AutoFormatEntry spNoFormats[] =
{
AUTOFORMAT_INVISIBLE( 1, 48 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spDataTableLines[] =
{
AUTOFORMAT_COLORMOD( 1, 32, THEMED_STYLE_SUBTLE, XML_tx1, XML_tint, 75000 ),
AUTOFORMAT_COLORMOD( 33, 40, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 75000 ),
// 41...48: no line, same as Chart2
AUTOFORMAT_END()
};
static const AutoFormatEntry spPlotArea2dFills[] =
{
AUTOFORMAT_COLOR( 1, 32, THEMED_STYLE_SUBTLE, XML_bg1 ),
AUTOFORMAT_COLORMOD( 33, 34, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 20000 ),
AUTOFORMAT_ACCENTSMOD( 35, THEMED_STYLE_SUBTLE, XML_tint, 20000 ), // tint not documented!?
AUTOFORMAT_COLORMOD( 41, 48, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 95000 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spFloorLines[] =
{
AUTOFORMAT_COLORMOD( 1, 32, THEMED_STYLE_SUBTLE, XML_tx1, XML_tint, 75000 ),
AUTOFORMAT_COLORMOD( 33, 40, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 75000 ),
// 41...48: no line, same as Chart2
AUTOFORMAT_END()
};
static const AutoFormatEntry spWallFloorFills[] =
{
AUTOFORMAT_INVISIBLE( 1, 32 ),
AUTOFORMAT_COLORMOD( 33, 34, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 20000 ),
AUTOFORMAT_ACCENTSMOD( 35, THEMED_STYLE_SUBTLE, XML_tint, 20000 ), // tint not documented!?
AUTOFORMAT_COLORMOD( 41, 48, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 95000 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spAxisLines[] =
{
AUTOFORMAT_COLORMOD( 1, 32, THEMED_STYLE_SUBTLE, XML_tx1, XML_tint, 75000 ), // tint not documented!?
AUTOFORMAT_COLORMOD( 33, 48, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 75000 ), // tint not documented!?
AUTOFORMAT_END()
};
static const AutoFormatEntry spMajorGridLines[] =
{
AUTOFORMAT_COLORMOD( 1, 32, THEMED_STYLE_SUBTLE, XML_tx1, XML_tint, 75000 ), // tint not documented!?
AUTOFORMAT_COLORMOD( 33, 48, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 75000 ), // tint not documented!?
AUTOFORMAT_END()
};
static const AutoFormatEntry spMinorGridLines[] =
{
AUTOFORMAT_COLORMOD( 1, 40, THEMED_STYLE_SUBTLE, XML_tx1, XML_tint, 50000 ),
AUTOFORMAT_COLORMOD( 41, 48, THEMED_STYLE_SUBTLE, XML_tx1, XML_tint, 90000 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spOtherLines[] =
{
AUTOFORMAT_COLOR( 1, 32, THEMED_STYLE_SUBTLE, XML_tx1 ),
AUTOFORMAT_COLOR( 33, 34, THEMED_STYLE_SUBTLE, XML_dk1 ),
AUTOFORMAT_COLORMOD( 35, 40, THEMED_STYLE_SUBTLE, XML_dk1, XML_shade, 25000 ),
AUTOFORMAT_COLOR( 41, 48, THEMED_STYLE_SUBTLE, XML_lt1 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spLinearSeriesLines[] =
{
AUTOFORMAT_PATTERN( 1, 1, THEMED_STYLE_SUBTLE, 300, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 2, 2, THEMED_STYLE_SUBTLE, 300, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 3, THEMED_STYLE_SUBTLE, 300 ),
AUTOFORMAT_PATTERN( 9, 9, THEMED_STYLE_SUBTLE, 500, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 10, 10, THEMED_STYLE_SUBTLE, 500, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 11, THEMED_STYLE_SUBTLE, 500 ),
AUTOFORMAT_PATTERN( 17, 17, THEMED_STYLE_SUBTLE, 500, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 18, 18, THEMED_STYLE_SUBTLE, 500, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 19, THEMED_STYLE_SUBTLE, 500 ),
AUTOFORMAT_PATTERN( 25, 25, THEMED_STYLE_SUBTLE, 700, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 26, 26, THEMED_STYLE_SUBTLE, 700, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 27, THEMED_STYLE_SUBTLE, 700 ),
AUTOFORMAT_PATTERN( 33, 33, THEMED_STYLE_SUBTLE, 500, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 34, 34, THEMED_STYLE_SUBTLE, 500, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 35, THEMED_STYLE_SUBTLE, 500 ),
AUTOFORMAT_PATTERN( 41, 42, THEMED_STYLE_SUBTLE, 500, spAutoFormatPattern4 ),
AUTOFORMAT_PATTERN( 42, 42, THEMED_STYLE_SUBTLE, 500, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 43, THEMED_STYLE_SUBTLE, 500 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spFilledSeriesLines[] =
{
AUTOFORMAT_INVISIBLE( 1, 8 ),
AUTOFORMAT_COLOR( 9, 16, THEMED_STYLE_SUBTLE, XML_lt1 ),
AUTOFORMAT_INVISIBLE( 17, 32 ),
AUTOFORMAT_COLORMOD( 33, 33, THEMED_STYLE_SUBTLE, XML_dk1, XML_shade, 50000 ),
AUTOFORMAT_PATTERN( 34, 34, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern3 ),
AUTOFORMAT_ACCENTSMOD( 35, THEMED_STYLE_SUBTLE, XML_shade, 50000 ),
AUTOFORMAT_INVISIBLE( 41, 48 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spFilledSeries2dFills[] =
{
AUTOFORMAT_PATTERN( 1, 1, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 2, 2, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 3, THEMED_STYLE_SUBTLE, 100 ),
AUTOFORMAT_PATTERN( 9, 9, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 10, 10, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 11, THEMED_STYLE_SUBTLE, 100 ),
AUTOFORMAT_PATTERN( 17, 17, THEMED_STYLE_INTENSE, 100, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 18, 18, THEMED_STYLE_INTENSE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 19, THEMED_STYLE_INTENSE, 100 ),
AUTOFORMAT_PATTERN( 25, 25, THEMED_STYLE_INTENSE, 100, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 26, 26, THEMED_STYLE_INTENSE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 27, THEMED_STYLE_INTENSE, 100 ),
AUTOFORMAT_PATTERN( 33, 33, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 34, 34, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 35, THEMED_STYLE_SUBTLE, 100 ),
AUTOFORMAT_PATTERN( 41, 42, THEMED_STYLE_INTENSE, 100, spAutoFormatPattern4 ),
AUTOFORMAT_PATTERN( 42, 42, THEMED_STYLE_INTENSE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 43, THEMED_STYLE_INTENSE, 100 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spFilledSeries3dFills[] =
{
AUTOFORMAT_PATTERN( 1, 1, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 2, 2, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 3, THEMED_STYLE_SUBTLE, 100 ),
AUTOFORMAT_PATTERN( 9, 9, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 10, 10, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 11, THEMED_STYLE_SUBTLE, 100 ),
AUTOFORMAT_PATTERN( 17, 17, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 18, 18, THEMED_STYLE_INTENSE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 19, THEMED_STYLE_SUBTLE, 100 ),
AUTOFORMAT_PATTERN( 25, 25, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 26, 26, THEMED_STYLE_INTENSE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 27, THEMED_STYLE_SUBTLE, 100 ),
AUTOFORMAT_PATTERN( 33, 33, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern1 ),
AUTOFORMAT_PATTERN( 34, 34, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 35, THEMED_STYLE_SUBTLE, 100 ),
AUTOFORMAT_PATTERN( 41, 42, THEMED_STYLE_SUBTLE, 100, spAutoFormatPattern4 ),
AUTOFORMAT_PATTERN( 42, 42, THEMED_STYLE_INTENSE, 100, spAutoFormatPattern2 ),
AUTOFORMAT_FADEDACCENTS( 43, THEMED_STYLE_SUBTLE, 100 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spFilledSeriesEffects[] =
{
// 1...8: no effect, same as Chart2
AUTOFORMAT_COLOR( 9, 16, THEMED_STYLE_SUBTLE, XML_dk1 ),
AUTOFORMAT_COLOR( 17, 24, THEMED_STYLE_MODERATE, XML_dk1 ),
AUTOFORMAT_COLOR( 25, 32, THEMED_STYLE_INTENSE, XML_dk1 ),
// 33...40: no effect, same as Chart2
AUTOFORMAT_COLOR( 41, 48, THEMED_STYLE_INTENSE, XML_dk1 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spUpDownBarLines[] =
{
AUTOFORMAT_COLOR( 1, 16, THEMED_STYLE_SUBTLE, XML_tx1 ),
AUTOFORMAT_INVISIBLE( 17, 32 ),
AUTOFORMAT_COLOR( 33, 34, THEMED_STYLE_SUBTLE, XML_dk1 ),
AUTOFORMAT_ACCENTSMOD( 35, THEMED_STYLE_SUBTLE, XML_shade, 25000 ),
AUTOFORMAT_INVISIBLE( 41, 48 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spUpBarFills[] =
{
AUTOFORMAT_COLORMOD( 1, 1, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 25000 ),
AUTOFORMAT_COLORMOD( 2, 2, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 5000 ),
AUTOFORMAT_ACCENTSMOD( 3, THEMED_STYLE_SUBTLE, XML_tint, 25000 ),
AUTOFORMAT_COLORMOD( 9, 9, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 25000 ),
AUTOFORMAT_COLORMOD( 10, 10, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 5000 ),
AUTOFORMAT_ACCENTSMOD( 11, THEMED_STYLE_SUBTLE, XML_tint, 25000 ),
AUTOFORMAT_COLORMOD( 17, 17, THEMED_STYLE_INTENSE, XML_dk1, XML_tint, 25000 ),
AUTOFORMAT_COLORMOD( 18, 18, THEMED_STYLE_INTENSE, XML_dk1, XML_tint, 5000 ),
AUTOFORMAT_ACCENTSMOD( 19, THEMED_STYLE_INTENSE, XML_tint, 25000 ),
AUTOFORMAT_COLORMOD( 25, 25, THEMED_STYLE_INTENSE, XML_dk1, XML_tint, 25000 ),
AUTOFORMAT_COLORMOD( 26, 26, THEMED_STYLE_INTENSE, XML_dk1, XML_tint, 5000 ),
AUTOFORMAT_ACCENTSMOD( 27, THEMED_STYLE_INTENSE, XML_tint, 25000 ),
AUTOFORMAT_COLOR( 33, 40, THEMED_STYLE_SUBTLE, XML_lt1 ),
AUTOFORMAT_COLORMOD( 41, 41, THEMED_STYLE_INTENSE, XML_dk1, XML_tint, 25000 ),
AUTOFORMAT_COLOR( 42, 42, THEMED_STYLE_INTENSE, XML_lt1 ),
AUTOFORMAT_ACCENTSMOD( 43, THEMED_STYLE_INTENSE, XML_tint, 25000 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spDownBarFills[] =
{
AUTOFORMAT_COLORMOD( 1, 1, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 85000 ),
AUTOFORMAT_COLORMOD( 2, 2, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 95000 ),
AUTOFORMAT_ACCENTSMOD( 3, THEMED_STYLE_SUBTLE, XML_shade, 25000 ),
AUTOFORMAT_COLORMOD( 9, 9, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 85000 ),
AUTOFORMAT_COLORMOD( 10, 10, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 95000 ),
AUTOFORMAT_ACCENTSMOD( 11, THEMED_STYLE_SUBTLE, XML_shade, 25000 ),
AUTOFORMAT_COLORMOD( 17, 17, THEMED_STYLE_INTENSE, XML_dk1, XML_tint, 85000 ),
AUTOFORMAT_COLORMOD( 18, 18, THEMED_STYLE_INTENSE, XML_dk1, XML_tint, 95000 ),
AUTOFORMAT_ACCENTSMOD( 19, THEMED_STYLE_INTENSE, XML_shade, 25000 ),
AUTOFORMAT_COLORMOD( 25, 25, THEMED_STYLE_INTENSE, XML_dk1, XML_tint, 85000 ),
AUTOFORMAT_COLORMOD( 26, 26, THEMED_STYLE_INTENSE, XML_dk1, XML_tint, 95000 ),
AUTOFORMAT_ACCENTSMOD( 27, THEMED_STYLE_INTENSE, XML_shade, 25000 ),
AUTOFORMAT_COLORMOD( 33, 33, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 85000 ),
AUTOFORMAT_COLORMOD( 34, 34, THEMED_STYLE_SUBTLE, XML_dk1, XML_tint, 95000 ),
AUTOFORMAT_ACCENTSMOD( 27, THEMED_STYLE_SUBTLE, XML_shade, 25000 ),
AUTOFORMAT_COLORMOD( 41, 41, THEMED_STYLE_INTENSE, XML_dk1, XML_tint, 85000 ),
AUTOFORMAT_COLOR( 42, 42, THEMED_STYLE_INTENSE, XML_dk1 ),
AUTOFORMAT_ACCENTSMOD( 43, THEMED_STYLE_INTENSE, XML_shade, 25000 ),
AUTOFORMAT_END()
};
static const AutoFormatEntry spUpDownBarEffects[] =
{
// 1...8: no effect, same as Chart2
AUTOFORMAT_COLOR( 9, 16, THEMED_STYLE_SUBTLE, XML_dk1 ),
AUTOFORMAT_COLOR( 17, 24, THEMED_STYLE_MODERATE, XML_dk1 ),
AUTOFORMAT_COLOR( 25, 32, THEMED_STYLE_INTENSE, XML_dk1 ),
// 33...40: no effect, same as Chart2
AUTOFORMAT_COLOR( 41, 48, THEMED_STYLE_INTENSE, XML_dk1 ),
AUTOFORMAT_END()
};
#undef AUTOFORMAT_COLOR
#undef AUTOFORMAT_ACCENTS
#undef AUTOFORMAT_COLORMOD
#undef AUTOFORMAT_ACCENTSMOD
#undef AUTOFORMAT_PATTERN
#undef AUTOFORMAT_FADED
#undef AUTOFORMAT_FADEDACCENTS
#undef AUTOFORMAT_INVISIBLE
#undef AUTOFORMAT_END
const AutoFormatEntry* lclGetAutoFormatEntry( const AutoFormatEntry* pEntries, sal_Int32 nStyle )
{
for( ; pEntries && (pEntries->mnFirstStyleIdx >= 0); ++pEntries )
if( (pEntries->mnFirstStyleIdx <= nStyle) && (nStyle <= pEntries->mnLastStyleIdx) )
return pEntries;
return 0;
}
// ----------------------------------------------------------------------------
struct AutoTextEntry
{
sal_Int32 mnFirstStyleIdx; /// First chart style index.
sal_Int32 mnLastStyleIdx; /// Last chart style index.
sal_Int32 mnThemedFont; /// Themed font (minor/major).
sal_Int32 mnColorToken; /// Theme color token.
sal_Int32 mnDefFontSize; /// Default font size (1/100 points).
sal_Int32 mnRelFontSize; /// Font size relative to chart global font (percent).
bool mbBold; /// True = bold font.
};
#define AUTOTEXT_COLOR( first, last, themed_font, color_token, def_font_size, rel_font_size, bold ) \
{ first, last, themed_font, color_token, def_font_size, rel_font_size, bold }
#define AUTOTEXT_END() \
AUTOTEXT_COLOR( -1, -1, XML_none, XML_TOKEN_INVALID, 1000, 100, false )
static const AutoTextEntry spChartTitleTexts[] =
{
AUTOTEXT_COLOR( 1, 40, XML_minor, XML_tx1, 1800, 120, true ),
AUTOTEXT_COLOR( 41, 48, XML_minor, XML_lt1, 1800, 120, true ),
AUTOTEXT_END()
};
static const AutoTextEntry spAxisTitleTexts[] =
{
AUTOTEXT_COLOR( 1, 40, XML_minor, XML_tx1, 1000, 100, true ),
AUTOTEXT_COLOR( 41, 48, XML_minor, XML_lt1, 1000, 100, true ),
AUTOTEXT_END()
};
static const AutoTextEntry spOtherTexts[] =
{
AUTOTEXT_COLOR( 1, 40, XML_minor, XML_tx1, 1000, 100, false ),
AUTOTEXT_COLOR( 41, 48, XML_minor, XML_lt1, 1000, 100, false ),
AUTOTEXT_END()
};
#undef AUTOTEXT_COLOR
#undef AUTOTEXT_END
const AutoTextEntry* lclGetAutoTextEntry( const AutoTextEntry* pEntries, sal_Int32 nStyle )
{
for( ; pEntries && (pEntries->mnFirstStyleIdx >= 0); ++pEntries )
if( (pEntries->mnFirstStyleIdx <= nStyle) && (nStyle <= pEntries->mnLastStyleIdx) )
return pEntries;
return 0;
}
// ----------------------------------------------------------------------------
/** Property identifiers for common chart objects, to be used in ShapePropertyInfo. */
static const sal_Int32 spnCommonPropIds[] =
{
PROP_LineStyle, PROP_LineWidth, PROP_LineColor, PROP_LineTransparence, PROP_LineDashName,
PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID,
PROP_FillStyle, PROP_FillColor, PROP_FillTransparence, PROP_FillGradientName,
PROP_FillBitmapName, PROP_FillBitmapMode, PROP_FillBitmapSizeX, PROP_FillBitmapSizeY,
PROP_FillBitmapPositionOffsetX, PROP_FillBitmapPositionOffsetY, PROP_FillBitmapRectanglePoint
};
/** Property identifiers for linear data series, to be used in ShapePropertyInfo. */
static const sal_Int32 spnLinearPropIds[] =
{
PROP_LineStyle, PROP_LineWidth, PROP_Color, PROP_Transparency, PROP_LineDashName,
PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID,
PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID,
PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID,
PROP_INVALID, PROP_INVALID, PROP_INVALID
};
/** Property identifiers for filled data series, to be used in ShapePropertyInfo. */
static const sal_Int32 spnFilledPropIds[] =
{
PROP_BorderStyle, PROP_BorderWidth, PROP_BorderColor, PROP_BorderTransparency, PROP_BorderDashName,
PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID, PROP_INVALID,
PROP_FillStyle, PROP_Color, PROP_Transparency, PROP_GradientName,
PROP_FillBitmapName, PROP_FillBitmapMode, PROP_FillBitmapSizeX, PROP_FillBitmapSizeY,
PROP_FillBitmapPositionOffsetX, PROP_FillBitmapPositionOffsetY, PROP_FillBitmapRectanglePoint
};
/** Property info for common chart objects, to be used in ShapePropertyMap. */
static const ShapePropertyInfo saCommonPropInfo( spnCommonPropIds, false, true, true, true );
/** Property info for linear data series, to be used in ShapePropertyMap. */
static const ShapePropertyInfo saLinearPropInfo( spnLinearPropIds, false, true, true, true );
/** Property info for filled data series, to be used in ShapePropertyMap. */
static const ShapePropertyInfo saFilledPropInfo( spnFilledPropIds, false, true, true, true );
// ----------------------------------------------------------------------------
/** Contains information about formatting of a specific chart object type. */
struct ObjectTypeFormatEntry
{
ObjectType meObjType; /// Object type for automatic format.
const ShapePropertyInfo* mpPropInfo; /// Property info for the ShapePropertyMap class.
const AutoFormatEntry* mpAutoLines; /// Automatic line formatting for all chart styles.
const AutoFormatEntry* mpAutoFills; /// Automatic fill formatting for all chart styles.
const AutoFormatEntry* mpAutoEffects; /// Automatic effect formatting for all chart styles.
const AutoTextEntry* mpAutoTexts; /// Automatic text attributes for all chart styles.
bool mbIsFrame; /// True = object is a frame, false = object is a line.
};
#define TYPEFORMAT_FRAME( obj_type, prop_type, auto_texts, auto_lines, auto_fills, auto_effects ) \
{ obj_type, prop_type, auto_lines, auto_fills, auto_effects, auto_texts, true }
#define TYPEFORMAT_LINE( obj_type, prop_type, auto_texts, auto_lines ) \
{ obj_type, prop_type, auto_lines, 0, 0, auto_texts, false }
static const ObjectTypeFormatEntry spObjTypeFormatEntries[] =
{
// object type property info auto text auto line auto fill auto effect
TYPEFORMAT_FRAME( OBJECTTYPE_CHARTSPACE, &saCommonPropInfo, 0, spNoFormats, spNoFormats, 0 /* eq to Ch2 */ ),
TYPEFORMAT_FRAME( OBJECTTYPE_CHARTTITLE, &saCommonPropInfo, spChartTitleTexts, 0 /* eq to Ch2 */, 0 /* eq to Ch2 */, 0 /* eq to Ch2 */ ),
TYPEFORMAT_FRAME( OBJECTTYPE_LEGEND, &saCommonPropInfo, spOtherTexts, spNoFormats, spNoFormats, 0 /* eq to Ch2 */ ),
TYPEFORMAT_FRAME( OBJECTTYPE_PLOTAREA2D, &saCommonPropInfo, 0, 0 /* eq to Ch2 */, spPlotArea2dFills, 0 /* eq to Ch2 */ ),
TYPEFORMAT_FRAME( OBJECTTYPE_PLOTAREA3D, &saCommonPropInfo, 0, 0 /* eq to Ch2 */, 0 /* eq to Ch2 */, 0 /* eq to Ch2 */ ),
TYPEFORMAT_FRAME( OBJECTTYPE_WALL, &saCommonPropInfo, 0, 0 /* eq to Ch2 */, spWallFloorFills, 0 /* eq to Ch2 */ ),
TYPEFORMAT_FRAME( OBJECTTYPE_FLOOR, &saCommonPropInfo, 0, spFloorLines, spWallFloorFills, 0 /* eq to Ch2 */ ),
TYPEFORMAT_LINE( OBJECTTYPE_AXIS, &saCommonPropInfo, spOtherTexts, spAxisLines ),
TYPEFORMAT_FRAME( OBJECTTYPE_AXISTITLE, &saCommonPropInfo, spAxisTitleTexts, 0 /* eq to Ch2 */, 0 /* eq to Ch2 */, 0 /* eq to Ch2 */ ),
TYPEFORMAT_FRAME( OBJECTTYPE_AXISUNIT, &saCommonPropInfo, spAxisTitleTexts, 0 /* eq in Ch2 */, 0 /* eq in Ch2 */, 0 /* eq in Ch2 */ ),
TYPEFORMAT_LINE( OBJECTTYPE_MAJORGRIDLINE, &saCommonPropInfo, 0, spMajorGridLines ),
TYPEFORMAT_LINE( OBJECTTYPE_MINORGRIDLINE, &saCommonPropInfo, 0, spMinorGridLines ),
TYPEFORMAT_LINE( OBJECTTYPE_LINEARSERIES2D, &saLinearPropInfo, 0, spLinearSeriesLines ),
TYPEFORMAT_FRAME( OBJECTTYPE_FILLEDSERIES2D, &saFilledPropInfo, 0, spFilledSeriesLines, spFilledSeries2dFills, spFilledSeriesEffects ),
TYPEFORMAT_FRAME( OBJECTTYPE_FILLEDSERIES3D, &saFilledPropInfo, 0, spFilledSeriesLines, spFilledSeries3dFills, spFilledSeriesEffects ),
TYPEFORMAT_FRAME( OBJECTTYPE_DATALABEL, &saCommonPropInfo, spOtherTexts, 0 /* eq to Ch2 */, 0 /* eq to Ch2 */, 0 /* eq to Ch2 */ ),
TYPEFORMAT_LINE( OBJECTTYPE_TRENDLINE, &saCommonPropInfo, 0, spOtherLines ),
TYPEFORMAT_FRAME( OBJECTTYPE_TRENDLINELABEL, &saCommonPropInfo, spOtherTexts, 0 /* eq to Ch2 */, 0 /* eq to Ch2 */, 0 /* eq to Ch2 */ ),
TYPEFORMAT_LINE( OBJECTTYPE_ERRORBAR, &saCommonPropInfo, 0, spOtherLines ),
TYPEFORMAT_LINE( OBJECTTYPE_SERLINE, &saCommonPropInfo, 0, spOtherLines ),
TYPEFORMAT_LINE( OBJECTTYPE_LEADERLINE, &saCommonPropInfo, 0, spOtherLines ),
TYPEFORMAT_LINE( OBJECTTYPE_DROPLINE, &saCommonPropInfo, 0, spOtherLines ),
TYPEFORMAT_LINE( OBJECTTYPE_HILOLINE, &saLinearPropInfo, 0, spOtherLines ),
TYPEFORMAT_FRAME( OBJECTTYPE_UPBAR, &saCommonPropInfo, 0, spUpDownBarLines, spUpBarFills, spUpDownBarEffects ),
TYPEFORMAT_FRAME( OBJECTTYPE_DOWNBAR, &saCommonPropInfo, 0, spUpDownBarLines, spDownBarFills, spUpDownBarEffects ),
TYPEFORMAT_LINE( OBJECTTYPE_DATATABLE, &saCommonPropInfo, spOtherTexts, spDataTableLines )
};
#undef TYPEFORMAT_FRAME
#undef TYPEFORMAT_LINE
// ----------------------------------------------------------------------------
void lclConvertPictureOptions( FillProperties& orFillProps, const PictureOptionsModel& rPicOptions )
{
bool bStacked = (rPicOptions.mnPictureFormat == XML_stack) || (rPicOptions.mnPictureFormat == XML_stackScale);
orFillProps.maBlipProps.moBitmapMode = bStacked ? XML_tile : XML_stretch;
}
} // namespace
// ============================================================================
struct ObjectFormatterData;
// ----------------------------------------------------------------------------
class DetailFormatterBase
{
public:
explicit DetailFormatterBase(
ObjectFormatterData& rData,
const AutoFormatEntry* pAutoFormatEntry );
explicit DetailFormatterBase(
ObjectFormatterData& rData,
const AutoTextEntry* pAutoTextEntry );
protected:
/** Returns the placeholder color which may depend on the passed series index. */
sal_Int32 getPhColor( sal_Int32 nSeriesIdx ) const;
private:
/** Resolves and returns the scheme color with the passed transformation. */
sal_Int32 getSchemeColor( sal_Int32 nColorToken, sal_Int32 nModToken, sal_Int32 nModValue ) const;
protected:
typedef ::std::vector< sal_Int32 > ColorPatternVec;
ObjectFormatterData& mrData; /// Shared formatter data.
sal_Int32 mnPhClr; /// RGB placeholder color for themed style.
ColorPatternVec maColorPattern; /// Different cycling colors for data series.
};
// ----------------------------------------------------------------------------
class LineFormatter : public DetailFormatterBase
{
public:
explicit LineFormatter(
ObjectFormatterData& rData,
const AutoFormatEntry* pAutoFormatEntry );
/** Converts line formatting to the passed property set. */
void convertFormatting(
ShapePropertyMap& rPropMap,
const ModelRef< Shape >& rxShapeProp,
sal_Int32 nSeriesIdx );
private:
LinePropertiesPtr mxAutoLine; /// Automatic line properties.
};
// ----------------------------------------------------------------------------
class FillFormatter : public DetailFormatterBase
{
public:
explicit FillFormatter(
ObjectFormatterData& rData,
const AutoFormatEntry* pAutoFormatEntry );
/** Converts area formatting to the passed property set. */
void convertFormatting(
ShapePropertyMap& rPropMap,
const ModelRef< Shape >& rxShapeProp,
const PictureOptionsModel* pPicOptions,
sal_Int32 nSeriesIdx );
private:
FillPropertiesPtr mxAutoFill; /// Automatic fill properties.
};
// ----------------------------------------------------------------------------
class EffectFormatter : public DetailFormatterBase
{
public:
explicit EffectFormatter(
ObjectFormatterData& rData,
const AutoFormatEntry* pAutoFormatEntry );
/** Converts effect formatting to the passed property set. */
void convertFormatting(
ShapePropertyMap& rPropMap,
const ModelRef< Shape >& rxShapeProp,
sal_Int32 nSeriesIdx ) const;
};
// ----------------------------------------------------------------------------
class TextFormatter : public DetailFormatterBase
{
public:
explicit TextFormatter(
ObjectFormatterData& rData,
const AutoTextEntry* pAutoTextEntry,
const ModelRef< TextBody >& rxGlobalTextProp );
/** Converts text formatting to the passed property set. */
void convertFormatting(
PropertySet& rPropSet,
const TextCharacterProperties* pTextProps );
/** Converts text formatting to the passed property set. */
void convertFormatting(
PropertySet& rPropSet,
const ModelRef< TextBody >& rxTextProp );
private:
TextCharacterPropertiesPtr mxAutoText; /// Automatic text properties.
};
// ----------------------------------------------------------------------------
/** Formatter for a specific object type. */
class ObjectTypeFormatter
{
public:
explicit ObjectTypeFormatter(
ObjectFormatterData& rData,
const ObjectTypeFormatEntry& rEntry,
const ChartSpaceModel& rChartSpace );
/** Sets frame formatting properties to the passed property set. */
void convertFrameFormatting(
PropertySet& rPropSet,
const ModelRef< Shape >& rxShapeProp,
const PictureOptionsModel* pPicOptions,
sal_Int32 nSeriesIdx );
/** Sets text formatting properties to the passed property set. */
void convertTextFormatting(
PropertySet& rPropSet,
const ModelRef< TextBody >& rxTextProp );
/** Sets frame/text formatting properties to the passed property set. */
void convertFormatting(
PropertySet& rPropSet,
const ModelRef< Shape >& rxShapeProp,
const ModelRef< TextBody >& rxTextProp );
/** Sets text formatting properties to the passed property set. */
void convertTextFormatting(
PropertySet& rPropSet,
const TextCharacterProperties& rTextProps );
/** Sets automatic line properties to the passed property set. */
void convertAutomaticLine(
PropertySet& rPropSet,
sal_Int32 nSeriesIdx );
/** Sets automatic fill properties to the passed property set. */
void convertAutomaticFill(
PropertySet& rPropSet,
sal_Int32 nSeriesIdx );
private:
LineFormatter maLineFormatter; /// Converter for line formatting.
FillFormatter maFillFormatter; /// Converter for fill formatting.
EffectFormatter maEffectFormatter; /// Converter for effect formatting.
TextFormatter maTextFormatter; /// Converter for text formatting.
ModelObjectHelper& mrModelObjHelper; /// Helper for named drawing formatting.
const ObjectTypeFormatEntry& mrEntry; /// Additional settings.
};
// ----------------------------------------------------------------------------
struct ObjectFormatterData
{
typedef RefMap< ObjectType, ObjectTypeFormatter > ObjectTypeFormatterMap;
const XmlFilterBase& mrFilter; /// Base filter object.
ObjectTypeFormatterMap maTypeFormatters; /// Formatters for all types of objects in a chart.
ModelObjectHelper maModelObjHelper; /// Helper for named drawing formatting (dashes, gradients, bitmaps).
Reference< XNumberFormats > mxNumFmts; /// Number formats collection of container document.
Reference< XNumberFormatTypes > mxNumTypes; /// Number format types collection of container document.
Locale maEnUsLocale; /// Locale struct containing en-US.
Locale maFromLocale; /// Empty locale struct.
sal_Int32 mnMaxSeriesIdx; /// Maximum series index used for color cycling/fading.
explicit ObjectFormatterData(
const XmlFilterBase& rFilter,
const Reference< XChartDocument >& rxChartDoc,
const ChartSpaceModel& rChartSpace );
ObjectTypeFormatter* getTypeFormatter( ObjectType eObjType );
};
// ============================================================================
DetailFormatterBase::DetailFormatterBase( ObjectFormatterData& rData, const AutoFormatEntry* pAutoFormatEntry ) :
mrData( rData ),
mnPhClr( -1 )
{
if( pAutoFormatEntry )
{
if( pAutoFormatEntry->mpPattern )
{
// prepare multi-color pattern
for( const AutoFormatPatternEntry* pPatternEntry = pAutoFormatEntry->mpPattern; pPatternEntry->mnColorToken != XML_TOKEN_INVALID; ++pPatternEntry )
maColorPattern.push_back( getSchemeColor( pPatternEntry->mnColorToken, pPatternEntry->mnModToken, pPatternEntry->mnModValue ) );
}
else if( pAutoFormatEntry->mnColorToken != XML_TOKEN_INVALID )
{
// prepare color or single-color pattern (color fading)
mnPhClr = getSchemeColor( pAutoFormatEntry->mnColorToken, pAutoFormatEntry->mnModToken, pAutoFormatEntry->mnModValue );
if( pAutoFormatEntry->mbFadedColor )
maColorPattern.push_back( mnPhClr );
}
}
}
DetailFormatterBase::DetailFormatterBase( ObjectFormatterData& rData, const AutoTextEntry* pAutoTextEntry ) :
mrData( rData ),
mnPhClr( -1 )
{
if( pAutoTextEntry && (pAutoTextEntry->mnColorToken != XML_TOKEN_INVALID) )
mnPhClr = getSchemeColor( pAutoTextEntry->mnColorToken, XML_TOKEN_INVALID, 0 );
}
sal_Int32 DetailFormatterBase::getPhColor( sal_Int32 nSeriesIdx ) const
{
if( maColorPattern.empty() || (mrData.mnMaxSeriesIdx < 0) || (nSeriesIdx < 0) )
return mnPhClr;
/* Apply tint/shade depending on the cycle index. The colors of leading
series are darkened (color shade), the colors of trailing series are
lightened (color tint). Shade/tint is applied in an exclusive range of
-70% to 70%.
Example 1: 3 data series using single-color shading with accent color 1
(e.g. automatic chart style #3). Shade/tint is applied per series.
Shade/tint changes in steps of 140%/(<series_count+1) = 140%/4 = 35%,
starting at -70%:
Step 1: -70% -> Not used.
Step 2: -35% -> Series 1 has 35% shade of accent color 1.
Step 3: 0% -> Series 2 has pure accent color 1.
Step 4: 35% -> Series 3 has 35% tint of accent color 1.
Step 5: 70% -> Not used.
Example 2: 20 data series using accent color pattern (e.g. automatic
chart style #2). Each color cycle has a size of 6 series (accent colors
1 to 6). Shade/tint is applied per color cycle.
Cycle #1: Series 1...6 are based on accent colors 1 to 6.
Cycle #2: Series 7...12 are based on accent colors 1 to 6.
Cycle #3: Series 13...18 are based on accent colors 1 to 6.
Cycle #4: Series 19...20 are based on accent colors 1 to 2.
Shade/tint changes in steps of 140%/(cycle_count+1) = 140%/5 = 28%,
starting at -70%:
Step 1: -70% -> Not used.
Step 2: -42% -> Cycle #1 has 42% shade of accent colors 1...6
step 3: -14% -> Cycle #2 has 14% shade of accent colors 1...6
step 4: 14% -> Cycle #3 has 14% tint of accent colors 1...6
step 5: 42% -> Cycle #4 has 42% tint of accent colors 1...6
step 6: 70% -> Not used.
*/
sal_Int32 nPhClr = maColorPattern[ static_cast< size_t >( nSeriesIdx % maColorPattern.size() ) ];
size_t nCycleIdx = static_cast< size_t >( nSeriesIdx / maColorPattern.size() );
size_t nMaxCycleIdx = static_cast< size_t >( mrData.mnMaxSeriesIdx / maColorPattern.size() );
double fShadeTint = static_cast< double >( nCycleIdx + 1 ) / (nMaxCycleIdx + 2) * 1.4 - 0.7;
if( fShadeTint != 0.0 )
{
Color aColor;
aColor.setSrgbClr( nPhClr );
aColor.addChartTintTransformation( fShadeTint );
nPhClr = aColor.getColor( mrData.mrFilter.getGraphicHelper() );
}
return nPhClr;
}
sal_Int32 DetailFormatterBase::getSchemeColor( sal_Int32 nColorToken, sal_Int32 nModToken, sal_Int32 nModValue ) const
{
Color aColor;
aColor.setSchemeClr( nColorToken );
if( nModToken != XML_TOKEN_INVALID )
aColor.addTransformation( nModToken, nModValue );
return aColor.getColor( mrData.mrFilter.getGraphicHelper() );
}
// ============================================================================
LineFormatter::LineFormatter( ObjectFormatterData& rData, const AutoFormatEntry* pAutoFormatEntry ) :
DetailFormatterBase( rData, pAutoFormatEntry )
{
if( pAutoFormatEntry )
{
mxAutoLine.reset( new LineProperties );
mxAutoLine->maLineFill.moFillType = XML_noFill;
if( const Theme* pTheme = mrData.mrFilter.getCurrentTheme() )
if( const LineProperties* pLineProps = pTheme->getLineStyle( pAutoFormatEntry->mnThemedIdx ) )
*mxAutoLine = *pLineProps;
// change line width according to chart auto style
if( mxAutoLine->moLineWidth.has() )
mxAutoLine->moLineWidth = mxAutoLine->moLineWidth.get() * pAutoFormatEntry->mnRelLineWidth / 100;
}
}
void LineFormatter::convertFormatting( ShapePropertyMap& rPropMap, const ModelRef< Shape >& rxShapeProp, sal_Int32 nSeriesIdx )
{
LineProperties aLineProps;
if( mxAutoLine.get() )
aLineProps.assignUsed( *mxAutoLine );
if( rxShapeProp.is() )
aLineProps.assignUsed( rxShapeProp->getLineProperties() );
aLineProps.pushToPropMap( rPropMap, mrData.mrFilter.getGraphicHelper(), getPhColor( nSeriesIdx ) );
}
// ============================================================================
FillFormatter::FillFormatter( ObjectFormatterData& rData, const AutoFormatEntry* pAutoFormatEntry ) :
DetailFormatterBase( rData, pAutoFormatEntry )
{
if( pAutoFormatEntry )
{
mxAutoFill.reset( new FillProperties );
mxAutoFill->moFillType = XML_noFill;
if( const Theme* pTheme = mrData.mrFilter.getCurrentTheme() )
if( const FillProperties* pFillProps = pTheme->getFillStyle( pAutoFormatEntry->mnThemedIdx ) )
*mxAutoFill = *pFillProps;
}
}
void FillFormatter::convertFormatting( ShapePropertyMap& rPropMap, const ModelRef< Shape >& rxShapeProp, const PictureOptionsModel* pPicOptions, sal_Int32 nSeriesIdx )
{
FillProperties aFillProps;
if( mxAutoFill.get() )
aFillProps.assignUsed( *mxAutoFill );
if( rxShapeProp.is() )
aFillProps.assignUsed( rxShapeProp->getFillProperties() );
if( pPicOptions )
lclConvertPictureOptions( aFillProps, *pPicOptions );
aFillProps.pushToPropMap( rPropMap, mrData.mrFilter.getGraphicHelper(), 0, getPhColor( nSeriesIdx ) );
}
// ============================================================================
EffectFormatter::EffectFormatter( ObjectFormatterData& rData, const AutoFormatEntry* pAutoFormatEntry ) :
DetailFormatterBase( rData, pAutoFormatEntry )
{
}
//TODO :
void EffectFormatter::convertFormatting( ShapePropertyMap& /*rPropMap*/, const ModelRef< Shape >& /*rxShapeProp*/, sal_Int32 /*nSeriesIdx*/ ) const
{
}
// ============================================================================
namespace {
const TextCharacterProperties* lclGetTextProperties( const ModelRef< TextBody >& rxTextProp )
{
return (rxTextProp.is() && !rxTextProp->getParagraphs().empty()) ?
&rxTextProp->getParagraphs().front()->getProperties().getTextCharacterProperties() : 0;
}
} // namespace
TextFormatter::TextFormatter( ObjectFormatterData& rData, const AutoTextEntry* pAutoTextEntry, const ModelRef< TextBody >& rxGlobalTextProp ) :
DetailFormatterBase( rData, pAutoTextEntry )
{
if( pAutoTextEntry )
{
mxAutoText.reset( new TextCharacterProperties );
if( const Theme* pTheme = mrData.mrFilter.getCurrentTheme() )
if( const TextCharacterProperties* pTextProps = pTheme->getFontStyle( pAutoTextEntry->mnThemedFont ) )
*mxAutoText = *pTextProps;
sal_Int32 nTextColor = getPhColor( -1 );
if( nTextColor >= 0 )
mxAutoText->maCharColor.setSrgbClr( nTextColor );
mxAutoText->moHeight = pAutoTextEntry->mnDefFontSize;
mxAutoText->moBold = pAutoTextEntry->mbBold;
if( const TextCharacterProperties* pTextProps = lclGetTextProperties( rxGlobalTextProp ) )
{
mxAutoText->assignUsed( *pTextProps );
if( pTextProps->moHeight.has() )
mxAutoText->moHeight = pTextProps->moHeight.get() * pAutoTextEntry->mnRelFontSize / 100;
}
}
}
void TextFormatter::convertFormatting( PropertySet& rPropSet, const TextCharacterProperties* pTextProps )
{
TextCharacterProperties aTextProps;
if( mxAutoText.get() )
aTextProps.assignUsed( *mxAutoText );
if( pTextProps )
aTextProps.assignUsed( *pTextProps );
aTextProps.pushToPropSet( rPropSet, mrData.mrFilter );
}
void TextFormatter::convertFormatting( PropertySet& rPropSet, const ModelRef< TextBody >& rxTextProp )
{
convertFormatting( rPropSet, lclGetTextProperties( rxTextProp ) );
}
// ============================================================================
ObjectTypeFormatter::ObjectTypeFormatter( ObjectFormatterData& rData, const ObjectTypeFormatEntry& rEntry, const ChartSpaceModel& rChartSpace ) :
maLineFormatter( rData, lclGetAutoFormatEntry( rEntry.mpAutoLines, rChartSpace.mnStyle ) ),
maFillFormatter( rData, lclGetAutoFormatEntry( rEntry.mpAutoFills, rChartSpace.mnStyle ) ),
maEffectFormatter( rData, lclGetAutoFormatEntry( rEntry.mpAutoEffects, rChartSpace.mnStyle ) ),
maTextFormatter( rData, lclGetAutoTextEntry( rEntry.mpAutoTexts, rChartSpace.mnStyle ), rChartSpace.mxTextProp ),
mrModelObjHelper( rData.maModelObjHelper ),
mrEntry( rEntry )
{
}
void ObjectTypeFormatter::convertFrameFormatting( PropertySet& rPropSet, const ModelRef< Shape >& rxShapeProp, const PictureOptionsModel* pPicOptions, sal_Int32 nSeriesIdx )
{
ShapePropertyMap aPropMap( mrModelObjHelper, *mrEntry.mpPropInfo );
maLineFormatter.convertFormatting( aPropMap, rxShapeProp, nSeriesIdx );
if( mrEntry.mbIsFrame )
maFillFormatter.convertFormatting( aPropMap, rxShapeProp, pPicOptions, nSeriesIdx );
maEffectFormatter.convertFormatting( aPropMap, rxShapeProp, nSeriesIdx );
rPropSet.setProperties( aPropMap );
}
void ObjectTypeFormatter::convertTextFormatting( PropertySet& rPropSet, const ModelRef< TextBody >& rxTextProp )
{
maTextFormatter.convertFormatting( rPropSet, rxTextProp );
}
void ObjectTypeFormatter::convertFormatting( PropertySet& rPropSet, const ModelRef< Shape >& rxShapeProp, const ModelRef< TextBody >& rxTextProp )
{
convertFrameFormatting( rPropSet, rxShapeProp, 0, -1 );
convertTextFormatting( rPropSet, rxTextProp );
}
void ObjectTypeFormatter::convertTextFormatting( PropertySet& rPropSet, const TextCharacterProperties& rTextProps )
{
maTextFormatter.convertFormatting( rPropSet, &rTextProps );
}
void ObjectTypeFormatter::convertAutomaticLine( PropertySet& rPropSet, sal_Int32 nSeriesIdx )
{
ShapePropertyMap aPropMap( mrModelObjHelper, *mrEntry.mpPropInfo );
ModelRef< Shape > xShapeProp;
maLineFormatter.convertFormatting( aPropMap, xShapeProp, nSeriesIdx );
maEffectFormatter.convertFormatting( aPropMap, xShapeProp, nSeriesIdx );
rPropSet.setProperties( aPropMap );
}
void ObjectTypeFormatter::convertAutomaticFill( PropertySet& rPropSet, sal_Int32 nSeriesIdx )
{
ShapePropertyMap aPropMap( mrModelObjHelper, *mrEntry.mpPropInfo );
ModelRef< Shape > xShapeProp;
maFillFormatter.convertFormatting( aPropMap, xShapeProp, 0, nSeriesIdx );
maEffectFormatter.convertFormatting( aPropMap, xShapeProp, nSeriesIdx );
rPropSet.setProperties( aPropMap );
}
// ============================================================================
ObjectFormatterData::ObjectFormatterData( const XmlFilterBase& rFilter, const Reference< XChartDocument >& rxChartDoc, const ChartSpaceModel& rChartSpace ) :
mrFilter( rFilter ),
maModelObjHelper( Reference< XMultiServiceFactory >( rxChartDoc, UNO_QUERY ) ),
maEnUsLocale( CREATE_OUSTRING( "en" ), CREATE_OUSTRING( "US" ), OUString() ),
mnMaxSeriesIdx( -1 )
{
const ObjectTypeFormatEntry* pEntryEnd = STATIC_ARRAY_END( spObjTypeFormatEntries );
for( const ObjectTypeFormatEntry* pEntry = spObjTypeFormatEntries; pEntry != pEntryEnd; ++pEntry )
maTypeFormatters[ pEntry->meObjType ].reset( new ObjectTypeFormatter( *this, *pEntry, rChartSpace ) );
try
{
Reference< XNumberFormatsSupplier > xNumFmtsSupp( rxChartDoc, UNO_QUERY_THROW );
mxNumFmts = xNumFmtsSupp->getNumberFormats();
mxNumTypes.set( mxNumFmts, UNO_QUERY );
}
catch( Exception& )
{
}
OSL_ENSURE( mxNumFmts.is() && mxNumTypes.is(), "ObjectFormatterData::ObjectFormatterData - cannot get number formats" );
}
ObjectTypeFormatter* ObjectFormatterData::getTypeFormatter( ObjectType eObjType )
{
OSL_ENSURE( maTypeFormatters.has( eObjType ), "ObjectFormatterData::getTypeFormatter - unknown object type" );
return maTypeFormatters.get( eObjType ).get();
}
// ============================================================================
ObjectFormatter::ObjectFormatter( const XmlFilterBase& rFilter, const Reference< XChartDocument >& rxChartDoc, const ChartSpaceModel& rChartSpace ) :
mxData( new ObjectFormatterData( rFilter, rxChartDoc, rChartSpace ) )
{
}
ObjectFormatter::~ObjectFormatter()
{
}
void ObjectFormatter::setMaxSeriesIndex( sal_Int32 nMaxSeriesIdx )
{
mxData->mnMaxSeriesIdx = nMaxSeriesIdx;
}
sal_Int32 ObjectFormatter::getMaxSeriesIndex() const
{
return mxData->mnMaxSeriesIdx;
}
void ObjectFormatter::convertFrameFormatting( PropertySet& rPropSet, const ModelRef< Shape >& rxShapeProp, ObjectType eObjType, sal_Int32 nSeriesIdx )
{
if( ObjectTypeFormatter* pFormat = mxData->getTypeFormatter( eObjType ) )
pFormat->convertFrameFormatting( rPropSet, rxShapeProp, 0, nSeriesIdx );
}
void ObjectFormatter::convertFrameFormatting( PropertySet& rPropSet, const ModelRef< Shape >& rxShapeProp, const PictureOptionsModel& rPicOptions, ObjectType eObjType, sal_Int32 nSeriesIdx )
{
if( ObjectTypeFormatter* pFormat = mxData->getTypeFormatter( eObjType ) )
pFormat->convertFrameFormatting( rPropSet, rxShapeProp, &rPicOptions, nSeriesIdx );
}
void ObjectFormatter::convertTextFormatting( PropertySet& rPropSet, const ModelRef< TextBody >& rxTextProp, ObjectType eObjType )
{
if( ObjectTypeFormatter* pFormat = mxData->getTypeFormatter( eObjType ) )
pFormat->convertTextFormatting( rPropSet, rxTextProp );
}
void ObjectFormatter::convertFormatting( PropertySet& rPropSet, const ModelRef< Shape >& rxShapeProp, const ModelRef< TextBody >& rxTextProp, ObjectType eObjType )
{
if( ObjectTypeFormatter* pFormat = mxData->getTypeFormatter( eObjType ) )
pFormat->convertFormatting( rPropSet, rxShapeProp, rxTextProp );
}
void ObjectFormatter::convertTextFormatting( PropertySet& rPropSet, const TextCharacterProperties& rTextProps, ObjectType eObjType )
{
if( ObjectTypeFormatter* pFormat = mxData->getTypeFormatter( eObjType ) )
pFormat->convertTextFormatting( rPropSet, rTextProps );
}
void ObjectFormatter::convertTextRotation( PropertySet& rPropSet, const ModelRef< TextBody >& rxTextProp, bool bSupportsStacked )
{
if( rxTextProp.is() )
{
bool bStacked = false;
if( bSupportsStacked )
{
sal_Int32 nVert = rxTextProp->getTextProperties().moVert.get( XML_horz );
bStacked = (nVert == XML_wordArtVert) || (nVert == XML_wordArtVertRtl);
rPropSet.setProperty( PROP_StackCharacters, bStacked );
}
/* Chart2 expects rotation angle as double value in range of [0,360).
OOXML counts clockwise, Chart2 counts counterclockwise. */
double fAngle = static_cast< double >( bStacked ? 0 : rxTextProp->getTextProperties().moRotation.get( 0 ) );
fAngle = getDoubleIntervalValue< double >( -fAngle / 60000.0, 0.0, 360.0 );
rPropSet.setProperty( PROP_TextRotation, fAngle );
}
}
void ObjectFormatter::convertNumberFormat( PropertySet& rPropSet, const NumberFormat& rNumberFormat, bool bPercentFormat )
{
if( mxData->mxNumFmts.is() )
{
sal_Int32 nPropId = bPercentFormat ? PROP_PercentageNumberFormat : PROP_NumberFormat;
if( rNumberFormat.mbSourceLinked || rNumberFormat.maFormatCode.isEmpty() )
{
rPropSet.setAnyProperty( nPropId, Any() );
}
else try
{
sal_Int32 nIndex = rNumberFormat.maFormatCode.equalsIgnoreAsciiCaseAsciiL(RTL_CONSTASCII_STRINGPARAM("general")) ?
mxData->mxNumTypes->getStandardIndex( mxData->maFromLocale ) :
mxData->mxNumFmts->addNewConverted( rNumberFormat.maFormatCode, mxData->maEnUsLocale, mxData->maFromLocale );
if( nIndex >= 0 )
rPropSet.setProperty( nPropId, nIndex );
}
catch( Exception& )
{
OSL_FAIL( OStringBuffer( "ObjectFormatter::convertNumberFormat - cannot create number format '" ).
append( OUStringToOString( rNumberFormat.maFormatCode, osl_getThreadTextEncoding() ) ).append( '\'' ).getStr() );
}
}
}
void ObjectFormatter::convertAutomaticFill( PropertySet& rPropSet, ObjectType eObjType, sal_Int32 nSeriesIdx )
{
if( ObjectTypeFormatter* pFormat = mxData->getTypeFormatter( eObjType ) )
pFormat->convertAutomaticFill( rPropSet, nSeriesIdx );
}
/*static*/ bool ObjectFormatter::isAutomaticFill( const ModelRef< Shape >& rxShapeProp )
{
return !rxShapeProp || !rxShapeProp->getFillProperties().moFillType.has();
}
// ============================================================================
} // namespace chart
} // namespace drawingml
} // namespace oox
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */