diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-15 05:54:39 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-15 05:54:39 +0000 |
commit | 267c6f2ac71f92999e969232431ba04678e7437e (patch) | |
tree | 358c9467650e1d0a1d7227a21dac2e3d08b622b2 /oox/source/drawingml/misccontexts.cxx | |
parent | Initial commit. (diff) | |
download | libreoffice-267c6f2ac71f92999e969232431ba04678e7437e.tar.xz libreoffice-267c6f2ac71f92999e969232431ba04678e7437e.zip |
Adding upstream version 4:24.2.0.upstream/4%24.2.0
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'oox/source/drawingml/misccontexts.cxx')
-rw-r--r-- | oox/source/drawingml/misccontexts.cxx | 703 |
1 files changed, 703 insertions, 0 deletions
diff --git a/oox/source/drawingml/misccontexts.cxx b/oox/source/drawingml/misccontexts.cxx new file mode 100644 index 0000000000..93d2c8ee41 --- /dev/null +++ b/oox/source/drawingml/misccontexts.cxx @@ -0,0 +1,703 @@ +/* -*- 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/misccontexts.hxx> +#include <oox/helper/attributelist.hxx> +#include <oox/helper/graphichelper.hxx> +#include <oox/core/xmlfilterbase.hxx> +#include <oox/drawingml/drawingmltypes.hxx> +#include <drawingml/fillproperties.hxx> +#include <oox/token/namespaces.hxx> +#include <oox/token/tokens.hxx> +#include <vcl/GraphicExternalLink.hxx> +#include <vcl/graph.hxx> +#include <unordered_map> +#include <frozen/bits/defines.h> +#include <frozen/bits/elsa_std.h> +#include <frozen/unordered_map.h> + +using namespace ::com::sun::star; +using namespace ::com::sun::star::uno; +using namespace ::com::sun::star::xml::sax; +using ::oox::core::ContextHandler2; +using ::oox::core::ContextHandlerRef; + +namespace oox::drawingml { + +SolidFillContext::SolidFillContext(ContextHandler2Helper const & rParent, FillProperties& rFillProps, model::SolidFill* pSolidFill) + : ColorContext(rParent, rFillProps.maFillColor, pSolidFill ? &pSolidFill->maColor : nullptr) +{ +} + +SolidFillContext::~SolidFillContext() +{} + +GradientFillContext::GradientFillContext(ContextHandler2Helper const & rParent, + const AttributeList& rAttribs, GradientFillProperties& rGradientProps, model::GradientFill* pGradientFill) + : ContextHandler2(rParent) + , mpGradientFill(pGradientFill) + , mrGradientProps(rGradientProps) +{ + auto oRotateWithShape = rAttribs.getBool(XML_rotWithShape); + mrGradientProps.moShadeFlip = rAttribs.getToken( XML_flip ); + mrGradientProps.moRotateWithShape = oRotateWithShape; + if (mpGradientFill && oRotateWithShape.has_value()) + mpGradientFill->mbRotateWithShape = *oRotateWithShape; +} + +ContextHandlerRef GradientFillContext::onCreateContext( + sal_Int32 nElement, const AttributeList& rAttribs ) +{ + switch( nElement ) + { + case A_TOKEN( gsLst ): + return this; // for gs elements + + case A_TOKEN( gs ): + if (rAttribs.hasAttribute(XML_pos)) + { + double fPosition = getLimitedValue<double>(rAttribs.getDouble(XML_pos, 0.0) / 100000.0, 0.0, 1.0); + auto aElement = mrGradientProps.maGradientStops.emplace(fPosition, Color()); + + model::ComplexColor* pComplexColor = nullptr; + if (mpGradientFill) + { + model::GradientStop& rStop = mpGradientFill->maGradientStops.emplace_back(); + rStop.mfPosition = fPosition; + pComplexColor = &rStop.maColor; + } + + return new ColorContext(*this, aElement->second, pComplexColor); + } + break; + + case A_TOKEN( lin ): + { + mrGradientProps.moShadeAngle = rAttribs.getInteger(XML_ang); + mrGradientProps.moShadeScaled = rAttribs.getBool(XML_scaled); + + if (mpGradientFill) + { + mpGradientFill->meGradientType = model::GradientType::Linear; + mpGradientFill->maLinearGradient.mnAngle = rAttribs.getInteger(XML_ang, 0); + mpGradientFill->maLinearGradient.mbScaled = rAttribs.getBool(XML_scaled, false); + } + } + break; + + case A_TOKEN( path ): + { + // always set a path type, this disables linear gradient in conversion + sal_Int32 nToken = rAttribs.getToken(XML_path, XML_rect); + mrGradientProps.moGradientPath = nToken; + if (mpGradientFill) + { + switch (nToken) + { + case XML_rect: + mpGradientFill->meGradientType = model::GradientType::Rectangle; + break; + case XML_circle: + mpGradientFill->meGradientType = model::GradientType::Circle; + break; + case XML_shape: + mpGradientFill->meGradientType = model::GradientType::Shape; + break; + default: + break; + } + } + return this; // for fillToRect element + } + case A_TOKEN( fillToRect ): + { + mrGradientProps.moFillToRect = GetRelativeRect( rAttribs.getFastAttributeList() ); + if (mpGradientFill) + fillRelativeRectangle(mpGradientFill->maFillToRectangle, rAttribs.getFastAttributeList()); + } + break; + + case A_TOKEN( tileRect ): + mrGradientProps.moTileRect = GetRelativeRect(rAttribs.getFastAttributeList()); + if (mpGradientFill) + fillRelativeRectangle(mpGradientFill->maTileRectangle, rAttribs.getFastAttributeList()); + break; + } + return nullptr; +} + + +namespace +{ + +constexpr frozen::unordered_map<sal_Int32, model::PatternPreset, 54> constPatternPresetMap +{ + { XML_pct5, model::PatternPreset::Percent_5 }, + { XML_pct10, model::PatternPreset::Percent_10 }, + { XML_pct20, model::PatternPreset::Percent_20 }, + { XML_pct25, model::PatternPreset::Percent_25 }, + { XML_pct30, model::PatternPreset::Percent_30 }, + { XML_pct40, model::PatternPreset::Percent_40 }, + { XML_pct50, model::PatternPreset::Percent_50 }, + { XML_pct60, model::PatternPreset::Percent_60 }, + { XML_pct70, model::PatternPreset::Percent_70 }, + { XML_pct75, model::PatternPreset::Percent_75 }, + { XML_pct80, model::PatternPreset::Percent_80 }, + { XML_pct90, model::PatternPreset::Percent_90 }, + { XML_horz, model::PatternPreset::Horizontal }, + { XML_vert, model::PatternPreset::Vertical }, + { XML_ltHorz, model::PatternPreset::LightHorizontal }, + { XML_ltVert, model::PatternPreset::LightVertical }, + { XML_dkHorz, model::PatternPreset::DarkHorizontal }, + { XML_dkVert, model::PatternPreset::DarkVertical }, + { XML_narHorz, model::PatternPreset::NarrowHorizontal }, + { XML_narVert, model::PatternPreset::NarrowVertical }, + { XML_dashHorz, model::PatternPreset::DashedHorizontal }, + { XML_dashVert, model::PatternPreset::DashedVertical }, + { XML_cross, model::PatternPreset::Cross }, + { XML_dnDiag, model::PatternPreset::DownwardDiagonal }, + { XML_upDiag, model::PatternPreset::UpwardDiagonal }, + { XML_ltDnDiag, model::PatternPreset::LightDownwardDiagonal }, + { XML_ltUpDiag, model::PatternPreset::LightUpwardDiagonal }, + { XML_dkDnDiag, model::PatternPreset::DarkDownwardDiagonal }, + { XML_dkUpDiag, model::PatternPreset::DarkUpwardDiagonal }, + { XML_wdDnDiag, model::PatternPreset::WideDownwardDiagonal }, + { XML_wdUpDiag, model::PatternPreset::WideUpwardDiagonal }, + { XML_dashDnDiag, model::PatternPreset::DashedDownwardDiagonal }, + { XML_dashUpDiag, model::PatternPreset::DashedUpwardDiagonal }, + { XML_diagCross, model::PatternPreset::DiagonalCross }, + { XML_smCheck, model::PatternPreset::SmallCheckerBoard }, + { XML_lgCheck, model::PatternPreset::LargeCheckerBoard }, + { XML_smGrid, model::PatternPreset::SmallGrid }, + { XML_lgGrid, model::PatternPreset::LargeGrid }, + { XML_dotGrid, model::PatternPreset::DottedGrid }, + { XML_smConfetti, model::PatternPreset::SmallConfetti }, + { XML_lgConfetti, model::PatternPreset::LargeConfetti }, + { XML_horzBrick, model::PatternPreset::HorizontalBrick }, + { XML_diagBrick, model::PatternPreset::DiagonalBrick }, + { XML_solidDmnd, model::PatternPreset::SolidDiamond }, + { XML_openDmnd, model::PatternPreset::OpenDiamond }, + { XML_dotDmnd, model::PatternPreset::DottedDiamond }, + { XML_plaid, model::PatternPreset::Plaid }, + { XML_sphere, model::PatternPreset::Sphere }, + { XML_weave, model::PatternPreset::Weave }, + { XML_divot, model::PatternPreset::Divot }, + { XML_shingle, model::PatternPreset::Shingle }, + { XML_wave, model::PatternPreset::Wave }, + { XML_trellis, model::PatternPreset::Trellis }, + { XML_zigZag, model::PatternPreset::ZigZag } +}; + +} // end anonymous namespace +PatternFillContext::PatternFillContext(ContextHandler2Helper const & rParent, + const AttributeList& rAttribs, PatternFillProperties& rPatternProps, model::PatternFill* pPatternFill) + : ContextHandler2(rParent) + , mpPatternFill(pPatternFill) + , mrPatternProps(rPatternProps) +{ + mrPatternProps.moPattPreset = rAttribs.getToken(XML_prst); + + if (mpPatternFill) + { + sal_Int32 nToken = rAttribs.getToken(XML_prst, XML_TOKEN_INVALID); + + auto aIterator = constPatternPresetMap.find(nToken); + if (aIterator != constPatternPresetMap.end()) + { + auto const& aPair = *aIterator; + model::PatternPreset ePatternPreset = aPair.second; + mpPatternFill->mePatternPreset = ePatternPreset; + } + } +} + +ContextHandlerRef PatternFillContext::onCreateContext( + sal_Int32 nElement, const AttributeList& ) +{ + model::ComplexColor* pComplexColor = nullptr; + switch( nElement ) + { + case A_TOKEN( bgClr ): + if (mpPatternFill) + pComplexColor = &mpPatternFill->maBackgroundColor; + return new ColorContext(*this, mrPatternProps.maPattBgColor, pComplexColor); + case A_TOKEN( fgClr ): + if (mpPatternFill) + pComplexColor = &mpPatternFill->maForegroundColor; + return new ColorContext(*this, mrPatternProps.maPattFgColor, pComplexColor); + } + return nullptr; +} + +ColorChangeContext::ColorChangeContext( ContextHandler2Helper const & rParent, + const AttributeList& rAttribs, BlipFillProperties& rBlipProps, model::BlipFill* pBlipFill) + : ContextHandler2(rParent) + , mpBlipFill(pBlipFill) + , mrBlipProps(rBlipProps) +{ + mrBlipProps.maColorChangeFrom.setUnused(); + mrBlipProps.maColorChangeTo.setUnused(); + mbUseAlpha = rAttribs.getBool( XML_useA, true ); + if (mpBlipFill) + { + auto& rEffect = mpBlipFill->maBlipEffects.emplace_back(); + rEffect.meType = model::BlipEffectType::ColorChange; + rEffect.mbUseAlpha = mbUseAlpha; + } +} + +ColorChangeContext::~ColorChangeContext() +{ + if( !mbUseAlpha ) + mrBlipProps.maColorChangeTo.clearTransparence(); +} + +ContextHandlerRef ColorChangeContext::onCreateContext( + sal_Int32 nElement, const AttributeList& ) +{ + model::ComplexColor* pComplexColor = nullptr; + switch (nElement) + { + case A_TOKEN(clrFrom): + if (mpBlipFill) + { + auto& rEffect = mpBlipFill->maBlipEffects.back(); + pComplexColor = &rEffect.getColorFrom(); + } + return new ColorContext(*this, mrBlipProps.maColorChangeFrom, pComplexColor); + case A_TOKEN(clrTo): + if (mpBlipFill) + { + auto& rEffect = mpBlipFill->maBlipEffects.back(); + pComplexColor = &rEffect.getColorTo(); + } + return new ColorContext(*this, mrBlipProps.maColorChangeTo, pComplexColor); + } + return nullptr; +} + +BlipContext::BlipContext(ContextHandler2Helper const & rParent, const AttributeList& rAttribs, + BlipFillProperties& rBlipProps, model::BlipFill* pBlipFill) + : ContextHandler2(rParent) + , mpBlipFill(pBlipFill) + , mrBlipProps(rBlipProps) +{ + if( rAttribs.hasAttribute( R_TOKEN( embed ) ) ) + { + // internal picture URL + OUString aFragmentPath = getFragmentPathFromRelId( rAttribs.getStringDefaulted( R_TOKEN( embed )) ); + if (!aFragmentPath.isEmpty()) + { + auto xGraphic = getFilter().getGraphicHelper().importEmbeddedGraphic(aFragmentPath); + mrBlipProps.mxFillGraphic = xGraphic; + if (mpBlipFill) + mpBlipFill->mxGraphic = xGraphic; + } + } + else if( rAttribs.hasAttribute( R_TOKEN( link ) ) ) + { + // external URL + + // we will embed this link, this is better than just doing nothing... + // TODO: import this graphic as real link, but this requires some + // code rework. + OUString aRelId = rAttribs.getStringDefaulted( R_TOKEN( link )); + OUString aTargetLink = getFilter().getAbsoluteUrl( getRelations().getExternalTargetFromRelId( aRelId ) ); + GraphicExternalLink aLink(aTargetLink); + Graphic aGraphic(aLink); + auto xGraphic = aGraphic.GetXGraphic(); + mrBlipProps.mxFillGraphic = xGraphic; + if (mpBlipFill) + mpBlipFill->mxGraphic = xGraphic; + } +} + +ContextHandlerRef BlipContext::onCreateContext( + sal_Int32 nElement, const AttributeList& rAttribs ) +{ + switch( nElement ) + { + case A_TOKEN( biLevel ): + { + sal_Int32 nTreshold = rAttribs.getInteger(XML_thresh, 0); + + mrBlipProps.moBiLevelThreshold = nTreshold; + mrBlipProps.moColorEffect = getBaseToken(nElement); + + if (mpBlipFill) + { + auto& rEffect = mpBlipFill->maBlipEffects.emplace_back(); + rEffect.meType = model::BlipEffectType::BiLevel; + rEffect.mnThreshold = nTreshold; + } + } + break; + + case A_TOKEN( grayscl ): + { + mrBlipProps.moColorEffect = getBaseToken( nElement ); + if (mpBlipFill) + { + auto& rEffect = mpBlipFill->maBlipEffects.emplace_back(); + rEffect.meType = model::BlipEffectType::Grayscale; + } + } + break; + + case A_TOKEN( clrChange ): + { + return new ColorChangeContext(*this, rAttribs, mrBlipProps, mpBlipFill); + } + break; + case A_TOKEN( duotone ): + return new DuotoneContext( *this, mrBlipProps ); + + case A_TOKEN( extLst ): + return new BlipExtensionContext(*this, mrBlipProps, mpBlipFill); + + case A_TOKEN( lum ): + { + mrBlipProps.moBrightness = rAttribs.getInteger( XML_bright ); + mrBlipProps.moContrast = rAttribs.getInteger( XML_contrast ); + + if (mpBlipFill) + { + auto& rEffect = mpBlipFill->maBlipEffects.emplace_back(); + rEffect.meType = model::BlipEffectType::Luminance; + rEffect.mnBrightness = rAttribs.getInteger(XML_bright, 0); + rEffect.mnContrast = rAttribs.getInteger(XML_contrast, 0); + } + } + break; + case A_TOKEN( alphaModFix ): + { + mrBlipProps.moAlphaModFix = rAttribs.getInteger(XML_amt); + if (mpBlipFill) + { + auto& rEffect = mpBlipFill->maBlipEffects.emplace_back(); + rEffect.meType = model::BlipEffectType::AlphaModulateFixed; + rEffect.mnAmount = rAttribs.getInteger(XML_amt, 100 * 1000); + } + } + break; + } + return nullptr; +} + +DuotoneContext::DuotoneContext( ContextHandler2Helper const & rParent, + BlipFillProperties& rBlipProps ) : + ContextHandler2( rParent ), + mrBlipProps( rBlipProps ), + mnColorIndex( 0 ) +{ + mrBlipProps.maDuotoneColors[0].setUnused(); + mrBlipProps.maDuotoneColors[1].setUnused(); +} + +DuotoneContext::~DuotoneContext() +{ +} + +::oox::core::ContextHandlerRef DuotoneContext::onCreateContext( + sal_Int32 /*nElement*/, const AttributeList& /*rAttribs*/ ) +{ + if( mnColorIndex < 2 ) + return new ColorValueContext(*this, mrBlipProps.maDuotoneColors[mnColorIndex++], nullptr); + return nullptr; +} + +BlipFillContext::BlipFillContext(ContextHandler2Helper const & rParent, const AttributeList& rAttribs, + BlipFillProperties& rBlipProps, model::BlipFill* pBlipFill) + : ContextHandler2( rParent ) + , mpBlipFill(pBlipFill) + , mrBlipProps(rBlipProps) +{ + mrBlipProps.moRotateWithShape = rAttribs.getBool( XML_rotWithShape ); + if (mpBlipFill) + mpBlipFill->mbRotateWithShape = rAttribs.getBool(XML_rotWithShape, false); +} + +ContextHandlerRef BlipFillContext::onCreateContext( + sal_Int32 nElement, const AttributeList& rAttribs ) +{ + switch( nElement ) + { + case A_TOKEN( blip ): + return new BlipContext(*this, rAttribs, mrBlipProps, mpBlipFill); + + case A_TOKEN( srcRect ): + { + mrBlipProps.moClipRect = GetRelativeRect( rAttribs.getFastAttributeList() ); + + if (mpBlipFill) + fillRelativeRectangle(mpBlipFill->maClipRectangle, rAttribs.getFastAttributeList()); + } + break; + + case A_TOKEN( tile ): + { + mrBlipProps.moBitmapMode = getBaseToken( nElement ); + mrBlipProps.moTileOffsetX = rAttribs.getInteger( XML_tx ); + mrBlipProps.moTileOffsetY = rAttribs.getInteger( XML_ty ); + mrBlipProps.moTileScaleX = rAttribs.getInteger( XML_sx ); + mrBlipProps.moTileScaleY = rAttribs.getInteger( XML_sy ); + mrBlipProps.moTileAlign = rAttribs.getToken( XML_algn ); + mrBlipProps.moTileFlip = rAttribs.getToken( XML_flip ); + + if (mpBlipFill) + { + mpBlipFill->meMode = model::BitmapMode::Tile; + mpBlipFill->mnTileOffsetX = rAttribs.getInteger(XML_tx, 0); + mpBlipFill->mnTileOffsetY = rAttribs.getInteger(XML_ty, 0); + mpBlipFill->mnTileScaleX = rAttribs.getInteger(XML_sx, 0); + mpBlipFill->mnTileScaleY = rAttribs.getInteger(XML_sy, 0); + + switch (rAttribs.getToken(XML_flip, XML_none)) + { + case XML_x: mpBlipFill->meTileFlipMode = model::FlipMode::X; break; + case XML_y: mpBlipFill->meTileFlipMode = model::FlipMode::Y; break; + case XML_xy: mpBlipFill->meTileFlipMode = model::FlipMode::XY; break; + default: + case XML_none: mpBlipFill->meTileFlipMode = model::FlipMode::None; break; + } + mpBlipFill->meTileAlignment = convertToRectangleAlignment(rAttribs.getToken(XML_algn, XML_TOKEN_INVALID)); + } + } + break; + + case A_TOKEN( stretch ): + { + mrBlipProps.moBitmapMode = getBaseToken( nElement ); + if (mpBlipFill) + { + mpBlipFill->meMode = model::BitmapMode::Stretch; + } + return this; // for fillRect element + } + break; + + case A_TOKEN( fillRect ): + { + mrBlipProps.moFillRect = GetRelativeRect( rAttribs.getFastAttributeList() ); + + if (mpBlipFill) + fillRelativeRectangle(mpBlipFill->maFillRectangle, rAttribs.getFastAttributeList()); + } + break; + } + return nullptr; +} + +FillPropertiesContext::FillPropertiesContext( ContextHandler2Helper const & rParent, FillProperties& rFillProps ) : + ContextHandler2( rParent ), + mrFillProps( rFillProps ) +{ +} + +ContextHandlerRef FillPropertiesContext::onCreateContext( + sal_Int32 nElement, const AttributeList& rAttribs ) +{ + return createFillContext(*this, nElement, rAttribs, mrFillProps, &maFillStyle); +} + +ContextHandlerRef FillPropertiesContext::createFillContext( + ContextHandler2Helper const & rParent, sal_Int32 nElement, + const AttributeList& rAttribs, FillProperties& rFillProps, + model::FillStyle* pFillStyle) +{ + switch( nElement ) + { + case A_TOKEN( noFill ): + { + rFillProps.moFillType = getBaseToken(nElement); + if (pFillStyle) + { + pFillStyle->mpFill = std::make_shared<model::NoFill>(); + } + return nullptr; + } + case A_TOKEN( solidFill ): + { + rFillProps.moFillType = getBaseToken(nElement); + model::SolidFill* pSolidFill = nullptr; + if (pFillStyle) + { + pFillStyle->mpFill = std::make_shared<model::SolidFill>(); + pSolidFill = static_cast<model::SolidFill*>(pFillStyle->mpFill.get()); + } + return new SolidFillContext(rParent, rFillProps, pSolidFill); + } + case A_TOKEN( gradFill ): + { + rFillProps.moFillType = getBaseToken(nElement); + model::GradientFill* pGradientFill = nullptr; + if (pFillStyle) + { + pFillStyle->mpFill = std::make_shared<model::GradientFill>(); + pGradientFill = static_cast<model::GradientFill*>(pFillStyle->mpFill.get()); + } + return new GradientFillContext(rParent, rAttribs, rFillProps.maGradientProps, pGradientFill); + } + case A_TOKEN( pattFill ): + { + rFillProps.moFillType = getBaseToken( nElement ); + model::PatternFill* pPatternFill = nullptr; + if (pFillStyle) + { + auto pFill = std::make_shared<model::PatternFill>(); + pPatternFill = pFill.get(); + pFillStyle->mpFill = pFill; + } + return new PatternFillContext(rParent, rAttribs, rFillProps.maPatternProps, pPatternFill); + } + case A_TOKEN( blipFill ): + { + rFillProps.moFillType = getBaseToken( nElement ); + model::BlipFill* pBlipFill = nullptr; + if (pFillStyle) + { + pFillStyle->mpFill = std::make_unique<model::BlipFill>(); + pBlipFill = static_cast<model::BlipFill*>(pFillStyle->mpFill.get()); + } + return new BlipFillContext( rParent, rAttribs, rFillProps.maBlipProps, pBlipFill); + } + case A_TOKEN( grpFill ): + { + // TODO + rFillProps.moFillType = getBaseToken( nElement ); + return nullptr; + }; + } + return nullptr; +} + +SimpleFillPropertiesContext::SimpleFillPropertiesContext( ContextHandler2Helper const & rParent, Color& rColor ) : + FillPropertiesContext( rParent, *this ), + mrColor( rColor ) +{ +} + +SimpleFillPropertiesContext::~SimpleFillPropertiesContext() +{ + mrColor = getBestSolidColor(); +} + +BlipExtensionContext::BlipExtensionContext(ContextHandler2Helper const & rParent, BlipFillProperties& rBlipProps, model::BlipFill* pBlipFill) + : ContextHandler2(rParent) + , mrBlipProps(rBlipProps) + , mpBlipFill(pBlipFill) +{ +} + +BlipExtensionContext::~BlipExtensionContext() +{ +} + +ContextHandlerRef BlipExtensionContext::onCreateContext(sal_Int32 nElement, const AttributeList& rAttribs) +{ + switch( nElement ) + { + case A_TOKEN(ext): + return new BlipExtensionContext(*this, mrBlipProps, mpBlipFill); + + case OOX_TOKEN(a14, imgProps): + return new ArtisticEffectContext(*this, mrBlipProps.maEffect); + + // Import the SVG Blip + case OOX_TOKEN(asvg, svgBlip): + { + if (rAttribs.hasAttribute(R_TOKEN(embed))) + { + OUString aFragmentPath = getFragmentPathFromRelId(rAttribs.getStringDefaulted(R_TOKEN(embed))); + if (!aFragmentPath.isEmpty()) + { + // Read the graphic from the fragment path + auto xGraphic = getFilter().getGraphicHelper().importEmbeddedGraphic(aFragmentPath); + + // Overwrite the fill graphic with the one contining SVG + mrBlipProps.mxFillGraphic = xGraphic; + if (mpBlipFill) + mpBlipFill->mxGraphic = xGraphic; + } + } + // TODO - link + } + break; + } + return nullptr; +} + +ArtisticEffectContext::ArtisticEffectContext( ContextHandler2Helper const & rParent, ArtisticEffectProperties& rEffect ) : + ContextHandler2( rParent ), + maEffect( rEffect ) +{ +} + +ArtisticEffectContext::~ArtisticEffectContext() +{ +} + +ContextHandlerRef ArtisticEffectContext::onCreateContext( + sal_Int32 nElement, const AttributeList& rAttribs ) +{ + // containers + if( nElement == OOX_TOKEN( a14, imgLayer ) ) + { + if( rAttribs.hasAttribute( R_TOKEN( embed ) ) ) + { + OUString aFragmentPath = getFragmentPathFromRelId( rAttribs.getStringDefaulted( R_TOKEN( embed )) ); + if( !aFragmentPath.isEmpty() ) + { + getFilter().importBinaryData( maEffect.mrOleObjectInfo.maEmbeddedData, aFragmentPath ); + maEffect.mrOleObjectInfo.maProgId = aFragmentPath; + } + } + return new ArtisticEffectContext( *this, maEffect ); + } + if( nElement == OOX_TOKEN( a14, imgEffect ) ) + return new ArtisticEffectContext( *this, maEffect ); + + // effects + maEffect.msName = ArtisticEffectProperties::getEffectString( nElement ); + if( maEffect.isEmpty() ) + return nullptr; + + // effect attributes + sal_Int32 const aAttribs[19] = { + XML_visible, XML_trans, XML_crackSpacing, XML_pressure, XML_numberOfShades, + XML_grainSize, XML_intensity, XML_smoothness, XML_gridSize, XML_pencilSize, + XML_size, XML_brushSize, XML_scaling, XML_detail, XML_bright, XML_contrast, + XML_colorTemp, XML_sat, XML_amount + }; + for(sal_Int32 nAttrib : aAttribs) + { + if( rAttribs.hasAttribute( nAttrib ) ) + { + OUString sName = ArtisticEffectProperties::getEffectString( nAttrib ); + if( !sName.isEmpty() ) + maEffect.maAttribs[sName] <<= rAttribs.getInteger( nAttrib, 0 ); + } + } + + return nullptr; +} + +} // namespace oox::drawingml + +/* vim:set shiftwidth=4 softtabstop=4 expandtab: */ |