/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "DomainMapper.hxx" #include #include #include "ConversionHelper.hxx" #include "GraphicHelpers.hxx" #include "GraphicImport.hxx" #include "PropertyMap.hxx" #include "WrapPolygonHandler.hxx" #include "util.hxx" using namespace css; namespace { bool isTopGroupObj(const uno::Reference& xShape) { SdrObject* pObject = GetSdrObjectFromXShape(xShape); if (!pObject) return false; if (pObject->getParentSdrObjectFromSdrObject()) return false; return pObject->IsGroupObject(); } } namespace writerfilter { namespace dmapper { class XInputStreamHelper : public cppu::WeakImplHelper { const sal_uInt8* m_pBuffer; const sal_Int32 m_nLength; sal_Int32 m_nPosition; public: XInputStreamHelper(const sal_uInt8* buf, size_t len); virtual ::sal_Int32 SAL_CALL readBytes( uno::Sequence< ::sal_Int8 >& aData, ::sal_Int32 nBytesToRead ) override; virtual ::sal_Int32 SAL_CALL readSomeBytes( uno::Sequence< ::sal_Int8 >& aData, ::sal_Int32 nMaxBytesToRead ) override; virtual void SAL_CALL skipBytes( ::sal_Int32 nBytesToSkip ) override; virtual ::sal_Int32 SAL_CALL available( ) override; virtual void SAL_CALL closeInput( ) override; }; XInputStreamHelper::XInputStreamHelper(const sal_uInt8* buf, size_t len) : m_pBuffer( buf ), m_nLength( len ), m_nPosition( 0 ) { } sal_Int32 XInputStreamHelper::readBytes( uno::Sequence& aData, sal_Int32 nBytesToRead ) { return readSomeBytes( aData, nBytesToRead ); } sal_Int32 XInputStreamHelper::readSomeBytes( uno::Sequence& aData, sal_Int32 nMaxBytesToRead ) { sal_Int32 nRet = 0; if( nMaxBytesToRead > 0 ) { if( nMaxBytesToRead > m_nLength - m_nPosition ) nRet = m_nLength - m_nPosition; else nRet = nMaxBytesToRead; aData.realloc( nRet ); sal_Int8* pData = aData.getArray(); if( nRet ) { memcpy( pData, m_pBuffer + m_nPosition, nRet ); m_nPosition += nRet; } } return nRet; } void XInputStreamHelper::skipBytes( sal_Int32 nBytesToSkip ) { if( nBytesToSkip < 0 || m_nPosition + nBytesToSkip > m_nLength) throw io::BufferSizeExceededException(); m_nPosition += nBytesToSkip; } sal_Int32 XInputStreamHelper::available( ) { return m_nLength - m_nPosition; } void XInputStreamHelper::closeInput( ) { } struct GraphicBorderLine { sal_Int32 nLineWidth; bool bHasShadow; GraphicBorderLine() : nLineWidth(0) ,bHasShadow(false) {} bool isEmpty() { return nLineWidth == 0 && !bHasShadow; } }; class GraphicImport_Impl { private: sal_Int32 nXSize; bool bXSizeValid; sal_Int32 nYSize; bool bYSizeValid; public: GraphicImportType const eGraphicImportType; DomainMapper& rDomainMapper; sal_Int32 nLeftPosition; sal_Int32 nTopPosition; bool bUseSimplePos; sal_Int32 zOrder; sal_Int16 nHoriOrient; sal_Int16 nHoriRelation; sal_Int16 nVertOrient; sal_Int16 nVertRelation; text::WrapTextMode nWrap; bool bLayoutInCell; bool bOpaque; bool bContour; bool bContourOutside; WrapPolygon::Pointer_t mpWrapPolygon; sal_Int32 nLeftMargin; sal_Int32 nRightMargin; sal_Int32 nTopMargin; sal_Int32 nBottomMargin; bool bShadow; sal_Int32 nShadowXDistance; sal_Int32 nShadowYDistance; sal_Int32 nShadowColor; sal_Int32 nShadowTransparence; sal_Int32 nContrast; sal_Int32 nBrightness; static constexpr sal_Int32 nFillColor = 0xffffffff; drawing::ColorMode eColorMode; GraphicBorderLine aBorders[4]; bool bIsGraphic; bool bSizeProtected; bool bPositionProtected; bool bHidden; sal_Int32 nShapeOptionType; OUString sName; OUString sAlternativeText; OUString title; OUString sHyperlinkURL; std::pair& m_rPositionOffsets; std::pair& m_rAligns; std::queue& m_rPositivePercentages; OUString sAnchorId; comphelper::SequenceAsHashMap m_aInteropGrabBag; boost::optional m_oEffectExtentLeft; boost::optional m_oEffectExtentTop; boost::optional m_oEffectExtentRight; boost::optional m_oEffectExtentBottom; GraphicImport_Impl(GraphicImportType eImportType, DomainMapper& rDMapper, std::pair& rPositionOffsets, std::pair& rAligns, std::queue& rPositivePercentages) : nXSize(0) ,bXSizeValid(false) ,nYSize(0) ,bYSizeValid(false) ,eGraphicImportType( eImportType ) ,rDomainMapper( rDMapper ) ,nLeftPosition(0) ,nTopPosition(0) ,bUseSimplePos(false) ,zOrder(-1) ,nHoriOrient( text::HoriOrientation::NONE ) ,nHoriRelation( text::RelOrientation::FRAME ) ,nVertOrient( text::VertOrientation::NONE ) ,nVertRelation( text::RelOrientation::FRAME ) ,nWrap(text::WrapTextMode_NONE) ,bLayoutInCell(false) ,bOpaque( !rDMapper.IsInHeaderFooter() ) ,bContour(false) ,bContourOutside(true) ,nLeftMargin(319) ,nRightMargin(319) ,nTopMargin(0) ,nBottomMargin(0) ,bShadow(false) ,nShadowXDistance(0) ,nShadowYDistance(0) ,nShadowColor(0) ,nShadowTransparence(0) ,nContrast(0) ,nBrightness(0) ,eColorMode( drawing::ColorMode_STANDARD ) ,bIsGraphic(false) ,bSizeProtected(false) ,bPositionProtected(false) ,bHidden(false) ,nShapeOptionType(0) ,m_rPositionOffsets(rPositionOffsets) ,m_rAligns(rAligns) ,m_rPositivePercentages(rPositivePercentages) {} void setXSize(sal_Int32 _nXSize) { nXSize = _nXSize; bXSizeValid = true; } sal_uInt32 getXSize() const { return nXSize; } bool isXSizeValid() const { return bXSizeValid; } void setYSize(sal_Int32 _nYSize) { nYSize = _nYSize; bYSizeValid = true; } sal_uInt32 getYSize() const { return nYSize; } bool isYSizeValis () const { return bYSizeValid; } void applyMargins(const uno::Reference< beans::XPropertySet >& xGraphicObjectProperties) const { xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_LEFT_MARGIN ), uno::makeAny(nLeftMargin)); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_RIGHT_MARGIN ), uno::makeAny(nRightMargin)); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_TOP_MARGIN ), uno::makeAny(nTopMargin)); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_BOTTOM_MARGIN ), uno::makeAny(nBottomMargin)); } void applyPosition(const uno::Reference< beans::XPropertySet >& xGraphicObjectProperties) const { xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_HORI_ORIENT ), uno::makeAny(nHoriOrient)); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_VERT_ORIENT ), uno::makeAny(nVertOrient)); } void applyRelativePosition(const uno::Reference< beans::XPropertySet >& xGraphicObjectProperties, bool bRelativeOnly = false) const { if (!bRelativeOnly) xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_HORI_ORIENT_POSITION), uno::makeAny(nLeftPosition)); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_HORI_ORIENT_RELATION ), uno::makeAny(nHoriRelation)); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_PAGE_TOGGLE ), uno::makeAny(false)); if (!bRelativeOnly) xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_VERT_ORIENT_POSITION), uno::makeAny(nTopPosition)); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_VERT_ORIENT_RELATION ), uno::makeAny(nVertRelation)); } void applyZOrder(uno::Reference const & xGraphicObjectProperties) const { if (zOrder >= 0) { GraphicZOrderHelper* pZOrderHelper = rDomainMapper.graphicZOrderHelper(); xGraphicObjectProperties->setPropertyValue(getPropertyName(PROP_Z_ORDER), uno::makeAny(pZOrderHelper->findZOrder(zOrder))); pZOrderHelper->addItem(xGraphicObjectProperties, zOrder); } } void applyName(uno::Reference const & xGraphicObjectProperties) const { try { // Ask the graphic naming helper to find out the name for this // object: It's around till the end of the import, so it remembers // what's the first free name. uno::Reference< container::XNamed > xNamed( xGraphicObjectProperties, uno::UNO_QUERY_THROW ); xNamed->setName(rDomainMapper.GetGraphicNamingHelper().NameGraphic(sName)); if ( sHyperlinkURL.getLength() > 0 ) xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_HYPER_LINK_U_R_L ), uno::makeAny ( sHyperlinkURL )); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_DESCRIPTION ), uno::makeAny( sAlternativeText )); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_TITLE ), uno::makeAny( title )); } catch( const uno::Exception& e ) { SAL_WARN("writerfilter", "failed. Message :" << e); } } /// Getter for m_aInteropGrabBag, but also merges in the values from other members if they are set. comphelper::SequenceAsHashMap const & getInteropGrabBag() { comphelper::SequenceAsHashMap aEffectExtent; if (m_oEffectExtentLeft) aEffectExtent["l"] <<= *m_oEffectExtentLeft; if (m_oEffectExtentTop) aEffectExtent["t"] <<= *m_oEffectExtentTop; if (m_oEffectExtentRight) aEffectExtent["r"] <<= *m_oEffectExtentRight; if (m_oEffectExtentBottom) aEffectExtent["b"] <<= *m_oEffectExtentBottom; if (!aEffectExtent.empty()) m_aInteropGrabBag["CT_EffectExtent"] <<= aEffectExtent.getAsConstPropertyValueList(); return m_aInteropGrabBag; } }; GraphicImport::GraphicImport(uno::Reference const& xComponentContext, uno::Reference const& xTextFactory, DomainMapper& rDMapper, GraphicImportType eImportType, std::pair& rPositionOffsets, std::pair& rAligns, std::queue& rPositivePercentages) : LoggedProperties("GraphicImport") , LoggedTable("GraphicImport") , LoggedStream("GraphicImport") , m_pImpl(new GraphicImport_Impl(eImportType, rDMapper, rPositionOffsets, rAligns, rPositivePercentages)) , m_xComponentContext(xComponentContext) , m_xTextFactory(xTextFactory) { } GraphicImport::~GraphicImport() { } void GraphicImport::handleWrapTextValue(sal_uInt32 nVal) { switch (nVal) { case NS_ooxml::LN_Value_wordprocessingDrawing_ST_WrapText_bothSides: // 90920; m_pImpl->nWrap = text::WrapTextMode_PARALLEL; break; case NS_ooxml::LN_Value_wordprocessingDrawing_ST_WrapText_left: // 90921; m_pImpl->nWrap = text::WrapTextMode_LEFT; break; case NS_ooxml::LN_Value_wordprocessingDrawing_ST_WrapText_right: // 90922; m_pImpl->nWrap = text::WrapTextMode_RIGHT; break; case NS_ooxml::LN_Value_wordprocessingDrawing_ST_WrapText_largest: // 90923; m_pImpl->nWrap = text::WrapTextMode_DYNAMIC; break; default:; } } void GraphicImport::putPropertyToFrameGrabBag( const OUString& sPropertyName, const uno::Any& aPropertyValue ) { beans::PropertyValue aProperty; aProperty.Name = sPropertyName; aProperty.Value = aPropertyValue; if (!m_xShape.is()) return; uno::Reference< beans::XPropertySet > xSet(m_xShape, uno::UNO_QUERY_THROW); uno::Reference< beans::XPropertySetInfo > xSetInfo(xSet->getPropertySetInfo()); if (!xSetInfo.is()) return; OUString aGrabBagPropName; uno::Reference xServiceInfo(m_xShape, uno::UNO_QUERY_THROW); if (xServiceInfo->supportsService("com.sun.star.text.TextFrame")) aGrabBagPropName = "FrameInteropGrabBag"; else aGrabBagPropName = "InteropGrabBag"; if (xSetInfo->hasPropertyByName(aGrabBagPropName)) { //Add pProperty to the end of the Sequence for aGrabBagPropName uno::Sequence aTmp; xSet->getPropertyValue(aGrabBagPropName) >>= aTmp; std::vector aGrabBag(comphelper::sequenceToContainer >(aTmp)); aGrabBag.push_back(aProperty); xSet->setPropertyValue(aGrabBagPropName, uno::makeAny(comphelper::containerToSequence(aGrabBag))); } } void GraphicImport::lcl_attribute(Id nName, Value& rValue) { sal_Int32 nIntValue = rValue.getInt(); switch( nName ) { case NS_ooxml::LN_CT_Hyperlink_URL://90682; m_pImpl->sHyperlinkURL = rValue.getString(); break; case NS_ooxml::LN_blip: //the binary graphic data in a shape { writerfilter::Reference::Pointer_t pProperties = rValue.getProperties(); if( pProperties.get()) { pProperties->resolve(*this); } } break; case NS_ooxml::LN_payload : { writerfilter::Reference::Pointer_t pPictureData = rValue.getBinary(); if( pPictureData.get()) pPictureData->resolve(*this); } break; //border properties case NS_ooxml::LN_CT_Border_sz: m_pImpl->aBorders[BORDER_TOP].nLineWidth = nIntValue; break; case NS_ooxml::LN_CT_Border_val: //graphic borders don't support different line types break; case NS_ooxml::LN_CT_Border_space: break; case NS_ooxml::LN_CT_Border_shadow: m_pImpl->aBorders[BORDER_TOP].bHasShadow = nIntValue != 0; break; case NS_ooxml::LN_CT_Border_frame: break; case NS_ooxml::LN_CT_PositiveSize2D_cx: case NS_ooxml::LN_CT_PositiveSize2D_cy: { sal_Int32 nDim = oox::drawingml::convertEmuToHmm(nIntValue); // drawingML equivalent of oox::vml::ShapeType::getAbsRectangle(): // make sure a shape isn't hidden implicitly just because it has // zero height or width. if (nDim == 0) nDim = 1; if( nName == NS_ooxml::LN_CT_PositiveSize2D_cx ) m_pImpl->setXSize(nDim); else m_pImpl->setYSize(nDim); } break; case NS_ooxml::LN_CT_EffectExtent_l: m_pImpl->m_oEffectExtentLeft = nIntValue; m_pImpl->nLeftMargin += oox::drawingml::convertEmuToHmm(nIntValue); break; case NS_ooxml::LN_CT_EffectExtent_t: m_pImpl->m_oEffectExtentTop = nIntValue; m_pImpl->nTopMargin += oox::drawingml::convertEmuToHmm(nIntValue); break; case NS_ooxml::LN_CT_EffectExtent_r: m_pImpl->m_oEffectExtentRight = nIntValue; m_pImpl->nRightMargin += oox::drawingml::convertEmuToHmm(nIntValue); break; case NS_ooxml::LN_CT_EffectExtent_b: m_pImpl->m_oEffectExtentBottom = nIntValue; m_pImpl->nBottomMargin += oox::drawingml::convertEmuToHmm(nIntValue); break; case NS_ooxml::LN_CT_NonVisualDrawingProps_id:// 90650; //id of the object - ignored break; case NS_ooxml::LN_CT_NonVisualDrawingProps_name:// 90651; //name of the object m_pImpl->sName = rValue.getString(); break; case NS_ooxml::LN_CT_NonVisualDrawingProps_descr:// 90652; //alternative text m_pImpl->sAlternativeText = rValue.getString(); break; case NS_ooxml::LN_CT_NonVisualDrawingProps_title: //alternative text m_pImpl->title = rValue.getString(); break; case NS_ooxml::LN_CT_NonVisualDrawingProps_hidden: m_pImpl->bHidden = (nIntValue == 1); break; case NS_ooxml::LN_CT_GraphicalObjectFrameLocking_noChangeAspect://90644; //disallow aspect ratio change - ignored break; case NS_ooxml::LN_CT_GraphicalObjectFrameLocking_noMove:// 90645; m_pImpl->bPositionProtected = true; break; case NS_ooxml::LN_CT_GraphicalObjectFrameLocking_noResize: // 90646; m_pImpl->bSizeProtected = true; break; case NS_ooxml::LN_CT_Anchor_distT: // 90983; case NS_ooxml::LN_CT_Anchor_distB: // 90984; case NS_ooxml::LN_CT_Anchor_distL: // 90985; case NS_ooxml::LN_CT_Anchor_distR: // 90986; { m_pImpl->nShapeOptionType = nName; ProcessShapeOptions(rValue); } break; case NS_ooxml::LN_CT_Anchor_simplePos_attr: // 90987; m_pImpl->bUseSimplePos = nIntValue > 0; break; case NS_ooxml::LN_CT_Anchor_relativeHeight: // 90988; m_pImpl->zOrder = nIntValue; break; case NS_ooxml::LN_CT_Anchor_behindDoc: // 90989; - in background if( nIntValue > 0 ) m_pImpl->bOpaque = false; break; case NS_ooxml::LN_CT_Anchor_locked: // 90990; - ignored break; case NS_ooxml::LN_CT_Anchor_layoutInCell: // 90991; - ignored m_pImpl->bLayoutInCell = nIntValue != 0; break; case NS_ooxml::LN_CT_Anchor_hidden: // 90992; - ignored break; case NS_ooxml::LN_CT_Anchor_allowOverlap: // 90993; //enable overlapping - ignored break; case NS_ooxml::LN_CT_Anchor_wp14_anchorId: case NS_ooxml::LN_CT_Inline_wp14_anchorId: { OUStringBuffer aBuffer = OUString::number(nIntValue, 16); OUStringBuffer aString; comphelper::string::padToLength(aString, 8 - aBuffer.getLength(), '0'); aString.append(aBuffer.getStr()); m_pImpl->sAnchorId = aString.makeStringAndClear().toAsciiUpperCase(); } break; case NS_ooxml::LN_CT_Point2D_x: // 90405; m_pImpl->nLeftPosition = ConversionHelper::convertTwipToMM100(nIntValue); m_pImpl->nHoriRelation = text::RelOrientation::PAGE_FRAME; m_pImpl->nHoriOrient = text::HoriOrientation::NONE; break; case NS_ooxml::LN_CT_Point2D_y: // 90406; m_pImpl->nTopPosition = ConversionHelper::convertTwipToMM100(nIntValue); m_pImpl->nVertRelation = text::RelOrientation::PAGE_FRAME; m_pImpl->nVertOrient = text::VertOrientation::NONE; break; case NS_ooxml::LN_CT_WrapTight_wrapText: // 90934; m_pImpl->bContour = true; m_pImpl->bContourOutside = true; handleWrapTextValue(rValue.getInt()); break; case NS_ooxml::LN_CT_WrapThrough_wrapText: m_pImpl->bContour = true; m_pImpl->bContourOutside = false; handleWrapTextValue(rValue.getInt()); break; case NS_ooxml::LN_CT_WrapSquare_wrapText: //90928; handleWrapTextValue(rValue.getInt()); break; case NS_ooxml::LN_shape: { uno::Reference< drawing::XShape> xShape; rValue.getAny( ) >>= xShape; if ( xShape.is( ) ) { // Is it a graphic image bool bUseShape = true; try { uno::Reference< beans::XPropertySet > xShapeProps ( xShape, uno::UNO_QUERY_THROW ); uno::Reference xGraphic; xShapeProps->getPropertyValue("Graphic") >>= xGraphic; sal_Int32 nRotation = 0; xShapeProps->getPropertyValue("RotateAngle") >>= nRotation; css::beans::PropertyValues aGrabBag; bool bContainsEffects = false; xShapeProps->getPropertyValue("InteropGrabBag") >>= aGrabBag; for( sal_Int32 i = 0; i < aGrabBag.getLength(); ++i ) { // if the shape contains effects in the grab bag, we should not transform it // in a XTextContent so those effects can be preserved if( aGrabBag[i].Name == "EffectProperties" || aGrabBag[i].Name == "3DEffectProperties" || aGrabBag[i].Name == "ArtisticEffectProperties" ) bContainsEffects = true; } xShapeProps->getPropertyValue("Shadow") >>= m_pImpl->bShadow; if (m_pImpl->bShadow) { xShapeProps->getPropertyValue("ShadowXDistance") >>= m_pImpl->nShadowXDistance; xShapeProps->getPropertyValue("ShadowYDistance") >>= m_pImpl->nShadowYDistance; xShapeProps->getPropertyValue("ShadowColor") >>= m_pImpl->nShadowColor; xShapeProps->getPropertyValue("ShadowTransparence") >>= m_pImpl->nShadowTransparence; } xShapeProps->getPropertyValue("GraphicColorMode") >>= m_pImpl->eColorMode; xShapeProps->getPropertyValue("AdjustLuminance") >>= m_pImpl->nBrightness; xShapeProps->getPropertyValue("AdjustContrast") >>= m_pImpl->nContrast; // fdo#70457: transform XShape into a SwXTextGraphicObject only if there's no rotation if ( nRotation == 0 && !bContainsEffects ) m_xGraphicObject = createGraphicObject( xGraphic, xShapeProps ); bUseShape = !m_xGraphicObject.is( ); if ( !bUseShape ) { // Define the object size uno::Reference< beans::XPropertySet > xGraphProps( m_xGraphicObject, uno::UNO_QUERY ); awt::Size aSize = xShape->getSize( ); xGraphProps->setPropertyValue("Height", uno::makeAny( aSize.Height ) ); xGraphProps->setPropertyValue("Width", uno::makeAny( aSize.Width ) ); text::GraphicCrop aGraphicCrop( 0, 0, 0, 0 ); uno::Reference< beans::XPropertySet > xSourceGraphProps( xShape, uno::UNO_QUERY ); uno::Any aAny = xSourceGraphProps->getPropertyValue("GraphicCrop"); if(aAny >>= aGraphicCrop) { xGraphProps->setPropertyValue("GraphicCrop", uno::makeAny( aGraphicCrop ) ); } // We need to drop the shape here somehow uno::Reference< lang::XComponent > xShapeComponent( xShape, uno::UNO_QUERY ); xShapeComponent->dispose( ); } } catch( const beans::UnknownPropertyException & ) { // It isn't a graphic image } if ( bUseShape ) m_xShape = xShape; if ( m_xShape.is( ) ) { uno::Reference< beans::XPropertySet > xShapeProps (m_xShape, uno::UNO_QUERY_THROW); xShapeProps->setPropertyValue (getPropertyName(PROP_ANCHOR_TYPE), uno::makeAny (text::TextContentAnchorType_AS_CHARACTER)); // In Word, if a shape is anchored inline, that // excludes being in the background. xShapeProps->setPropertyValue("Opaque", uno::makeAny(true)); uno::Reference xServiceInfo(m_xShape, uno::UNO_QUERY_THROW); // TextFrames can't be rotated. But for anything else, // make sure that setting size doesn't affect rotation, // that would not match Word's definition of rotation. bool bKeepRotation = false; if (!xServiceInfo->supportsService("com.sun.star.text.TextFrame")) { bKeepRotation = true; xShapeProps->setPropertyValue (getPropertyName(PROP_TEXT_RANGE), uno::makeAny (m_pImpl->rDomainMapper.GetCurrentTextRange())); } awt::Size aSize(m_xShape->getSize()); if (m_pImpl->isXSizeValid()) aSize.Width = m_pImpl->getXSize(); if (m_pImpl->isYSizeValis()) aSize.Height = m_pImpl->getYSize(); sal_Int32 nRotation = 0; if (bKeepRotation) { // Use internal API, getPropertyValue(RotateAngle) // would use GetObjectRotation(), which is not what // we want. if (SdrObject* pShape = GetSdrObjectFromXShape(m_xShape)) nRotation = pShape->GetRotateAngle(); } m_xShape->setSize(aSize); if (bKeepRotation) xShapeProps->setPropertyValue("RotateAngle", uno::makeAny(nRotation)); m_pImpl->bIsGraphic = true; if (!m_pImpl->sAnchorId.isEmpty()) { putPropertyToFrameGrabBag("AnchorId", uno::makeAny(m_pImpl->sAnchorId)); } } if (bUseShape && m_pImpl->eGraphicImportType == IMPORT_AS_DETECTED_ANCHOR) { // If we are here, this is a drawingML shape. For those, only dmapper (and not oox) knows the anchoring infos (just like for Writer pictures). // But they aren't Writer pictures, either (which are already handled above). uno::Reference< beans::XPropertySet > xShapeProps(m_xShape, uno::UNO_QUERY_THROW); // This needs to be AT_PARAGRAPH by default and not AT_CHARACTER, otherwise shape will move when the user inserts a new paragraph. text::TextContentAnchorType eAnchorType = text::TextContentAnchorType_AT_PARAGRAPH; if (m_pImpl->bHidden) { xShapeProps->setPropertyValue("Visible", uno::makeAny(false)); xShapeProps->setPropertyValue("Printable", uno::makeAny(false)); } // Avoid setting AnchorType for TextBoxes till SwTextBoxHelper::syncProperty() doesn't handle transition. bool bTextBox = false; xShapeProps->getPropertyValue("TextBox") >>= bTextBox; if (m_pImpl->nVertRelation == text::RelOrientation::TEXT_LINE && !bTextBox) eAnchorType = text::TextContentAnchorType_AT_CHARACTER; xShapeProps->setPropertyValue("AnchorType", uno::makeAny(eAnchorType)); //only the position orientation is handled in applyPosition() m_pImpl->applyPosition(xShapeProps); uno::Reference xServiceInfo(m_xShape, uno::UNO_QUERY_THROW); if (xServiceInfo->supportsService("com.sun.star.drawing.GroupShape") || xServiceInfo->supportsService("com.sun.star.drawing.GraphicObjectShape")) { // You would expect that position and rotation are // independent, but they are not. Till we are not // there yet to handle all scaling, translation and // rotation with a single transformation matrix, // make sure there is no rotation set when we set // the position. sal_Int32 nRotation = 0; xShapeProps->getPropertyValue("RotateAngle") >>= nRotation; if (nRotation) xShapeProps->setPropertyValue("RotateAngle", uno::makeAny(sal_Int32(0))); // Position of the groupshape should be set after children have been added. // Long-term we should get rid of positioning group // shapes, though. Do it for top-level ones with // absolute page position as a start. // fdo#80555: also set position for graphic shapes here if (!isTopGroupObj(m_xShape) || m_pImpl->nHoriRelation != text::RelOrientation::PAGE_FRAME || m_pImpl->nVertRelation != text::RelOrientation::PAGE_FRAME) m_xShape->setPosition( awt::Point(m_pImpl->nLeftPosition, m_pImpl->nTopPosition)); if (nRotation) xShapeProps->setPropertyValue("RotateAngle", uno::makeAny(nRotation)); } m_pImpl->applyRelativePosition(xShapeProps, /*bRelativeOnly=*/true); xShapeProps->setPropertyValue("SurroundContour", uno::makeAny(m_pImpl->bContour)); m_pImpl->applyMargins(xShapeProps); xShapeProps->setPropertyValue("Opaque", uno::makeAny(m_pImpl->bOpaque)); xShapeProps->setPropertyValue("Surround", uno::makeAny(static_cast(m_pImpl->nWrap))); m_pImpl->applyZOrder(xShapeProps); m_pImpl->applyName(xShapeProps); // Get the grab-bag set by oox, merge with our one and then put it back. comphelper::SequenceAsHashMap aInteropGrabBag(xShapeProps->getPropertyValue("InteropGrabBag")); aInteropGrabBag.update(m_pImpl->getInteropGrabBag()); xShapeProps->setPropertyValue("InteropGrabBag", uno::makeAny(aInteropGrabBag.getAsConstPropertyValueList())); } else if (bUseShape && m_pImpl->eGraphicImportType == IMPORT_AS_DETECTED_INLINE) { uno::Reference< beans::XPropertySet > xShapeProps(m_xShape, uno::UNO_QUERY_THROW); comphelper::SequenceAsHashMap aInteropGrabBag(xShapeProps->getPropertyValue("InteropGrabBag")); aInteropGrabBag.update(m_pImpl->getInteropGrabBag()); xShapeProps->setPropertyValue("InteropGrabBag", uno::makeAny(aInteropGrabBag.getAsConstPropertyValueList())); } } } break; case NS_ooxml::LN_CT_Inline_distT: m_pImpl->nTopMargin = 0; break; case NS_ooxml::LN_CT_Inline_distB: m_pImpl->nBottomMargin = 0; break; case NS_ooxml::LN_CT_Inline_distL: m_pImpl->nLeftMargin = 0; break; case NS_ooxml::LN_CT_Inline_distR: m_pImpl->nRightMargin = 0; break; case NS_ooxml::LN_CT_GraphicalObjectData_uri: rValue.getString(); //TODO: does it need to be handled? break; case NS_ooxml::LN_CT_SizeRelH_relativeFrom: { switch (nIntValue) { case NS_ooxml::LN_ST_SizeRelFromH_margin: if (m_xShape.is()) { uno::Reference xPropertySet(m_xShape, uno::UNO_QUERY); xPropertySet->setPropertyValue("RelativeWidthRelation", uno::makeAny(text::RelOrientation::FRAME)); } break; case NS_ooxml::LN_ST_SizeRelFromH_page: if (m_xShape.is()) { uno::Reference xPropertySet(m_xShape, uno::UNO_QUERY); xPropertySet->setPropertyValue("RelativeWidthRelation", uno::makeAny(text::RelOrientation::PAGE_FRAME)); } break; default: SAL_WARN("writerfilter", "GraphicImport::lcl_attribute: unhandled NS_ooxml::LN_CT_SizeRelH_relativeFrom value: " << nIntValue); break; } } break; case NS_ooxml::LN_CT_SizeRelV_relativeFrom: { switch (nIntValue) { case NS_ooxml::LN_ST_SizeRelFromV_margin: if (m_xShape.is()) { uno::Reference xPropertySet(m_xShape, uno::UNO_QUERY); xPropertySet->setPropertyValue("RelativeHeightRelation", uno::makeAny(text::RelOrientation::FRAME)); } break; case NS_ooxml::LN_ST_SizeRelFromV_page: if (m_xShape.is()) { uno::Reference xPropertySet(m_xShape, uno::UNO_QUERY); xPropertySet->setPropertyValue("RelativeHeightRelation", uno::makeAny(text::RelOrientation::PAGE_FRAME)); } break; default: SAL_WARN("writerfilter", "GraphicImport::lcl_attribute: unhandled NS_ooxml::LN_CT_SizeRelV_relativeFrom value: " << nIntValue); break; } } break; default: #ifdef DBG_UTIL TagLogger::getInstance().element("unhandled"); #endif break; } } uno::Reference GraphicImport::GetGraphicObject() { uno::Reference xResult; if (m_xGraphicObject.is()) xResult = m_xGraphicObject; else if (m_xShape.is()) { xResult.set(m_xShape, uno::UNO_QUERY_THROW); } return xResult; } void GraphicImport::ProcessShapeOptions(Value const & rValue) { sal_Int32 nIntValue = rValue.getInt(); switch( m_pImpl->nShapeOptionType ) { case NS_ooxml::LN_CT_Anchor_distL: //todo: changes have to be applied depending on the orientation, see SwWW8ImplReader::AdjustLRWrapForWordMargins() m_pImpl->nLeftMargin = nIntValue / 360; break; case NS_ooxml::LN_CT_Anchor_distT: //todo: changes have to be applied depending on the orientation, see SwWW8ImplReader::AdjustULWrapForWordMargins() m_pImpl->nTopMargin = nIntValue / 360; break; case NS_ooxml::LN_CT_Anchor_distR: //todo: changes have to be applied depending on the orientation, see SwWW8ImplReader::AdjustLRWrapForWordMargins() m_pImpl->nRightMargin = nIntValue / 360; break; case NS_ooxml::LN_CT_Anchor_distB: //todo: changes have to be applied depending on the orientation, see SwWW8ImplReader::AdjustULWrapForWordMargins() m_pImpl->nBottomMargin = nIntValue / 360; break; default: OSL_FAIL( "shape option unsupported?"); } } void GraphicImport::lcl_sprm(Sprm& rSprm) { sal_uInt32 nSprmId = rSprm.getId(); Value::Pointer_t pValue = rSprm.getValue(); switch(nSprmId) { case NS_ooxml::LN_CT_Inline_extent: // 90911; case NS_ooxml::LN_CT_Inline_effectExtent: // 90912; case NS_ooxml::LN_CT_Inline_docPr: // 90913; case NS_ooxml::LN_CT_Inline_cNvGraphicFramePr: // 90914; case NS_ooxml::LN_CT_NonVisualGraphicFrameProperties_graphicFrameLocks:// 90657 case NS_ooxml::LN_CT_Inline_a_graphic:// 90915 case NS_ooxml::LN_CT_Anchor_simplePos_elem: // 90975; case NS_ooxml::LN_CT_Anchor_extent: // 90978; case NS_ooxml::LN_CT_Anchor_effectExtent: // 90979; case NS_ooxml::LN_EG_WrapType_wrapSquare: // 90945; case NS_ooxml::LN_EG_WrapType_wrapTight: // 90946; case NS_ooxml::LN_EG_WrapType_wrapThrough: case NS_ooxml::LN_CT_Anchor_docPr: // 90980; case NS_ooxml::LN_CT_Anchor_cNvGraphicFramePr: // 90981; case NS_ooxml::LN_CT_Anchor_a_graphic: // 90982; case NS_ooxml::LN_CT_WrapPath_start: // 90924; case NS_ooxml::LN_CT_WrapPath_lineTo: // 90925; case NS_ooxml::LN_graphic_graphic: case NS_ooxml::LN_pic_pic: case NS_ooxml::LN_dgm_relIds: case NS_ooxml::LN_lc_lockedCanvas: case NS_ooxml::LN_c_chart: case NS_ooxml::LN_wps_wsp: case NS_ooxml::LN_wpg_wgp: case NS_ooxml::LN_sizeRelH_sizeRelH: case NS_ooxml::LN_sizeRelV_sizeRelV: case NS_ooxml::LN_hlinkClick_hlinkClick: { writerfilter::Reference::Pointer_t pProperties = rSprm.getProps(); if( pProperties.get()) { pProperties->resolve(*this); } // We'll map these to PARALLEL, save the original wrap type. if (nSprmId == NS_ooxml::LN_EG_WrapType_wrapTight) m_pImpl->m_aInteropGrabBag["EG_WrapType"] <<= OUString("wrapTight"); else if (nSprmId == NS_ooxml::LN_EG_WrapType_wrapThrough) m_pImpl->m_aInteropGrabBag["EG_WrapType"] <<= OUString("wrapThrough"); } break; case NS_ooxml::LN_CT_WrapTight_wrapPolygon: case NS_ooxml::LN_CT_WrapThrough_wrapPolygon: { WrapPolygonHandler aHandler; resolveSprmProps(aHandler, rSprm); m_pImpl->mpWrapPolygon = aHandler.getPolygon(); // Save the wrap path in case we can't handle it natively: drawinglayer shapes, TextFrames. m_pImpl->m_aInteropGrabBag["CT_WrapPath"] <<= m_pImpl->mpWrapPolygon->getPointSequenceSequence(); } break; case NS_ooxml::LN_CT_Anchor_positionH: // 90976; { // Use a special handler for the positioning std::shared_ptr pHandler( new PositionHandler( m_pImpl->m_rPositionOffsets, m_pImpl->m_rAligns )); writerfilter::Reference::Pointer_t pProperties = rSprm.getProps(); if( pProperties.get( ) ) { pProperties->resolve( *pHandler ); if( !m_pImpl->bUseSimplePos ) { m_pImpl->nHoriRelation = pHandler->relation(); m_pImpl->nHoriOrient = pHandler->orientation(); m_pImpl->nLeftPosition = pHandler->position(); } } } break; case NS_ooxml::LN_CT_Anchor_positionV: // 90977; { // Use a special handler for the positioning std::shared_ptr pHandler( new PositionHandler( m_pImpl->m_rPositionOffsets, m_pImpl->m_rAligns)); writerfilter::Reference::Pointer_t pProperties = rSprm.getProps(); if( pProperties.get( ) ) { pProperties->resolve( *pHandler ); if( !m_pImpl->bUseSimplePos ) { m_pImpl->nVertRelation = pHandler->relation(); m_pImpl->nVertOrient = pHandler->orientation(); m_pImpl->nTopPosition = pHandler->position(); } } } break; case NS_ooxml::LN_CT_SizeRelH_pctWidth: case NS_ooxml::LN_CT_SizeRelV_pctHeight: if (m_pImpl->m_rPositivePercentages.empty()) break; if (m_xShape.is()) { sal_Int16 nPositivePercentage = rtl::math::round(m_pImpl->m_rPositivePercentages.front().toDouble() / oox::drawingml::PER_PERCENT); if (nPositivePercentage) { uno::Reference xPropertySet(m_xShape, uno::UNO_QUERY); OUString aProperty = nSprmId == NS_ooxml::LN_CT_SizeRelH_pctWidth ? OUString("RelativeWidth") : OUString("RelativeHeight"); xPropertySet->setPropertyValue(aProperty, uno::makeAny(nPositivePercentage)); } } // Make sure the token is consumed even if xShape is an empty // reference. m_pImpl->m_rPositivePercentages.pop(); break; case NS_ooxml::LN_EG_WrapType_wrapNone: // 90944; - doesn't contain attributes //depending on the behindDoc attribute text wraps through behind or in front of the object m_pImpl->nWrap = text::WrapTextMode_THROUGH; // Wrap though means the margins defined earlier should not be // respected. m_pImpl->nLeftMargin = 0; m_pImpl->nTopMargin = 0; m_pImpl->nRightMargin = 0; m_pImpl->nBottomMargin = 0; break; case NS_ooxml::LN_EG_WrapType_wrapTopAndBottom: // 90948; m_pImpl->nWrap = text::WrapTextMode_NONE; break; case NS_ooxml::LN_CT_GraphicalObject_graphicData:// 90660; { m_pImpl->bIsGraphic = true; writerfilter::Reference::Pointer_t pProperties = rSprm.getProps(); if( pProperties.get()) pProperties->resolve(*this); } break; case NS_ooxml::LN_CT_NonVisualDrawingProps_a_hlinkClick: // 90689; { writerfilter::Reference::Pointer_t pProperties = rSprm.getProps(); if( pProperties.get( ) ) pProperties->resolve( *this ); } break; default: SAL_WARN("writerfilter", "GraphicImport::lcl_sprm: unhandled token: " << nSprmId); break; } } void GraphicImport::lcl_entry(int /*pos*/, writerfilter::Reference::Pointer_t /*ref*/) { } uno::Reference GraphicImport::createGraphicObject(uno::Reference const & rxGraphic, uno::Reference const & xShapeProps) { uno::Reference xGraphicObject; try { if (rxGraphic.is()) { uno::Reference< beans::XPropertySet > xGraphicObjectProperties( m_xTextFactory->createInstance("com.sun.star.text.TextGraphicObject"), uno::UNO_QUERY_THROW); xGraphicObjectProperties->setPropertyValue(getPropertyName(PROP_GRAPHIC), uno::makeAny(rxGraphic)); xGraphicObjectProperties->setPropertyValue(getPropertyName(PROP_ANCHOR_TYPE), uno::makeAny( m_pImpl->eGraphicImportType == IMPORT_AS_DETECTED_ANCHOR ? text::TextContentAnchorType_AT_CHARACTER : text::TextContentAnchorType_AS_CHARACTER )); xGraphicObject.set( xGraphicObjectProperties, uno::UNO_QUERY_THROW ); //shapes have only one border table::BorderLine2 aBorderLine; GraphicBorderLine& rBorderLine = m_pImpl->aBorders[0]; if (rBorderLine.isEmpty() && xShapeProps.is() && xShapeProps->getPropertyValue("LineStyle").get() != drawing::LineStyle_NONE) { // In case we got no border tokens and we have the // original shape, then use its line properties as the // border. aBorderLine.Color = xShapeProps->getPropertyValue("LineColor").get(); aBorderLine.LineWidth = xShapeProps->getPropertyValue("LineWidth").get(); } else { aBorderLine.Color = 0; aBorderLine.InnerLineWidth = 0; aBorderLine.OuterLineWidth = static_cast(rBorderLine.nLineWidth); aBorderLine.LineDistance = 0; } PropertyIds const aBorderProps[] = { PROP_LEFT_BORDER, PROP_RIGHT_BORDER, PROP_TOP_BORDER, PROP_BOTTOM_BORDER }; for(PropertyIds const & rBorderProp : aBorderProps) xGraphicObjectProperties->setPropertyValue(getPropertyName(rBorderProp), uno::makeAny(aBorderLine)); // setting graphic object shadow properties if (m_pImpl->bShadow) { // Shadow width is approximated by average of X and Y table::ShadowFormat aShadow; sal_uInt32 nShadowColor = m_pImpl->nShadowColor & 0x00FFFFFF; // The shadow color we get is RGB only. sal_Int32 nShadowWidth = (abs(m_pImpl->nShadowXDistance) + abs(m_pImpl->nShadowYDistance)) / 2; aShadow.ShadowWidth = nShadowWidth; sal_uInt8 nShadowTransparence = float(m_pImpl->nShadowTransparence) * 2.55; nShadowColor |= (nShadowTransparence << 24); // Add transparence to the color. aShadow.Color = nShadowColor; // Distances -ve for top and right, +ve for bottom and left if (m_pImpl->nShadowXDistance > 0) { if (m_pImpl->nShadowYDistance > 0) aShadow.Location = table::ShadowLocation_BOTTOM_RIGHT; else aShadow.Location = table::ShadowLocation_TOP_RIGHT; } else { if (m_pImpl->nShadowYDistance > 0) aShadow.Location = table::ShadowLocation_BOTTOM_LEFT; else aShadow.Location = table::ShadowLocation_TOP_LEFT; } xGraphicObjectProperties->setPropertyValue(getPropertyName(PROP_SHADOW_FORMAT), uno::makeAny(aShadow)); } // setting properties for all types if( m_pImpl->bPositionProtected ) xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_POSITION_PROTECTED ), uno::makeAny(true)); if( m_pImpl->bSizeProtected ) xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_SIZE_PROTECTED ), uno::makeAny(true)); sal_Int32 nWidth = - m_pImpl->nLeftPosition; if (m_pImpl->eGraphicImportType == IMPORT_AS_DETECTED_ANCHOR) { //adjust margins if( (m_pImpl->nHoriOrient == text::HoriOrientation::LEFT && (m_pImpl->nHoriRelation == text::RelOrientation::PAGE_PRINT_AREA || m_pImpl->nHoriRelation == text::RelOrientation::FRAME) ) || (m_pImpl->nHoriOrient == text::HoriOrientation::INSIDE && m_pImpl->nHoriRelation == text::RelOrientation::PAGE_PRINT_AREA )) m_pImpl->nLeftMargin = 0; if((m_pImpl->nHoriOrient == text::HoriOrientation::RIGHT && (m_pImpl->nHoriRelation == text::RelOrientation::PAGE_PRINT_AREA || m_pImpl->nHoriRelation == text::RelOrientation::FRAME) ) || (m_pImpl->nHoriOrient == text::HoriOrientation::INSIDE && m_pImpl->nHoriRelation == text::RelOrientation::PAGE_PRINT_AREA )) m_pImpl->nRightMargin = 0; // adjust top/bottom margins if( m_pImpl->nVertOrient == text::VertOrientation::TOP && ( m_pImpl->nVertRelation == text::RelOrientation::PAGE_PRINT_AREA || m_pImpl->nVertRelation == text::RelOrientation::PAGE_FRAME)) m_pImpl->nTopMargin = 0; if( m_pImpl->nVertOrient == text::VertOrientation::BOTTOM && ( m_pImpl->nVertRelation == text::RelOrientation::PAGE_PRINT_AREA || m_pImpl->nVertRelation == text::RelOrientation::PAGE_FRAME)) m_pImpl->nBottomMargin = 0; if( m_pImpl->nVertOrient == text::VertOrientation::BOTTOM && m_pImpl->nVertRelation == text::RelOrientation::PAGE_PRINT_AREA ) m_pImpl->nBottomMargin = 0; //adjust alignment if( m_pImpl->nHoriOrient == text::HoriOrientation::INSIDE && m_pImpl->nHoriRelation == text::RelOrientation::PAGE_FRAME ) { // convert 'left to page' to 'from left - to page text area' m_pImpl->nHoriOrient = text::HoriOrientation::NONE; m_pImpl->nHoriRelation = text::RelOrientation::PAGE_PRINT_AREA; m_pImpl->nLeftPosition = - nWidth; } else if( m_pImpl->nHoriOrient == text::HoriOrientation::OUTSIDE && m_pImpl->nHoriRelation == text::RelOrientation::PAGE_FRAME ) { // convert 'right to page' to 'from left 0 to right page border' m_pImpl->nHoriOrient = text::HoriOrientation::NONE; m_pImpl->nHoriRelation = text::RelOrientation::PAGE_RIGHT; m_pImpl->nLeftPosition = 0; } m_pImpl->applyPosition(xGraphicObjectProperties); m_pImpl->applyRelativePosition(xGraphicObjectProperties); if( !m_pImpl->bOpaque ) { xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_OPAQUE ), uno::makeAny(m_pImpl->bOpaque)); } xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_SURROUND ), uno::makeAny(static_cast(m_pImpl->nWrap))); if( m_pImpl->rDomainMapper.IsInTable() && m_pImpl->bLayoutInCell && m_pImpl->nWrap != text::WrapTextMode_THROUGH ) xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_FOLLOW_TEXT_FLOW ), uno::makeAny(true)); if( m_pImpl->rDomainMapper.IsInTable() && m_pImpl->bLayoutInCell ) { xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_LAYOUT_IN_CELL ), uno::makeAny(true)); } xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_SURROUND_CONTOUR ), uno::makeAny(m_pImpl->bContour)); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_CONTOUR_OUTSIDE ), uno::makeAny(m_pImpl->bContourOutside)); m_pImpl->applyMargins(xGraphicObjectProperties); } if( m_pImpl->eColorMode == drawing::ColorMode_WATERMARK && m_pImpl->nContrast == -70 && m_pImpl->nBrightness == 70 ) { // watermark filter is already applied at this point, so reset Contrast and Brightness m_pImpl->nContrast = 0; m_pImpl->nBrightness = 0; } xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_ADJUST_CONTRAST ), uno::makeAny(static_cast(m_pImpl->nContrast))); xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_ADJUST_LUMINANCE ), uno::makeAny(static_cast(m_pImpl->nBrightness))); if(m_pImpl->eColorMode != drawing::ColorMode_STANDARD) { xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_GRAPHIC_COLOR_MODE ), uno::makeAny(m_pImpl->eColorMode)); } xGraphicObjectProperties->setPropertyValue(getPropertyName( PROP_BACK_COLOR ), uno::makeAny( GraphicImport_Impl::nFillColor )); m_pImpl->applyZOrder(xGraphicObjectProperties); //there seems to be no way to detect the original size via _real_ API uno::Reference< beans::XPropertySet > xGraphicProperties(rxGraphic, uno::UNO_QUERY_THROW); if (m_pImpl->mpWrapPolygon.get() != nullptr) { uno::Any aContourPolyPolygon; awt::Size aGraphicSize; WrapPolygon::Pointer_t pCorrected; xGraphicProperties->getPropertyValue(getPropertyName(PROP_SIZE100th_M_M)) >>= aGraphicSize; if (aGraphicSize.Width && aGraphicSize.Height) { pCorrected = m_pImpl->mpWrapPolygon->correctWordWrapPolygon(aGraphicSize); } else { xGraphicProperties->getPropertyValue(getPropertyName(PROP_SIZE_PIXEL)) >>= aGraphicSize; if (aGraphicSize.Width && aGraphicSize.Height) { pCorrected = m_pImpl->mpWrapPolygon->correctWordWrapPolygonPixel(aGraphicSize); } } if (pCorrected) { aContourPolyPolygon <<= pCorrected->getPointSequenceSequence(); xGraphicObjectProperties->setPropertyValue(getPropertyName(PROP_CONTOUR_POLY_POLYGON), aContourPolyPolygon); // We should bring it to front, even if wp:anchor's behindDoc="1", // because otherwise paragraph background (if set) overlaps the graphic // TODO: if paragraph's background becomes bottommost, then remove this hack xGraphicObjectProperties->setPropertyValue("Opaque", uno::makeAny(true)); } } if(m_pImpl->eGraphicImportType == IMPORT_AS_DETECTED_INLINE || m_pImpl->eGraphicImportType == IMPORT_AS_DETECTED_ANCHOR) { if( m_pImpl->getXSize() && m_pImpl->getYSize() ) xGraphicObjectProperties->setPropertyValue(getPropertyName(PROP_SIZE), uno::makeAny( awt::Size( m_pImpl->getXSize(), m_pImpl->getYSize() ))); m_pImpl->applyMargins(xGraphicObjectProperties); m_pImpl->applyName(xGraphicObjectProperties); } // Handle horizontal flip. bool bMirrored = false; xShapeProps->getPropertyValue("IsMirrored") >>= bMirrored; if (bMirrored) { xGraphicObjectProperties->setPropertyValue("HoriMirroredOnEvenPages", uno::makeAny(true)); xGraphicObjectProperties->setPropertyValue("HoriMirroredOnOddPages", uno::makeAny(true)); } } } catch( const uno::Exception& e ) { SAL_WARN("writerfilter", "failed. Message :" << e); } return xGraphicObject; } void GraphicImport::data(const sal_uInt8* buf, size_t len, writerfilter::Reference::Pointer_t /*ref*/) { beans::PropertyValues aMediaProperties( 1 ); aMediaProperties[0].Name = getPropertyName(PROP_INPUT_STREAM); uno::Reference< io::XInputStream > xIStream = new XInputStreamHelper( buf, len ); aMediaProperties[0].Value <<= xIStream; uno::Reference xPropertySet; uno::Reference xGraphicProvider(graphic::GraphicProvider::create(m_xComponentContext)); uno::Reference xGraphic = xGraphicProvider->queryGraphic(aMediaProperties); m_xGraphicObject = createGraphicObject(xGraphic, xPropertySet); } void GraphicImport::lcl_startSectionGroup() { } void GraphicImport::lcl_endSectionGroup() { } void GraphicImport::lcl_startParagraphGroup() { } void GraphicImport::lcl_endParagraphGroup() { } void GraphicImport::lcl_startCharacterGroup() { } void GraphicImport::lcl_endCharacterGroup() { } void GraphicImport::lcl_text(const sal_uInt8 * /*_data*/, size_t /*len*/) { } void GraphicImport::lcl_utext(const sal_uInt8 * /*_data*/, size_t /*len*/) { } void GraphicImport::lcl_props(writerfilter::Reference::Pointer_t /*ref*/) { } void GraphicImport::lcl_table(Id /*name*/, writerfilter::Reference::Pointer_t /*ref*/) { } void GraphicImport::lcl_substream(Id /*name*/, ::writerfilter::Reference::Pointer_t /*ref*/) { } void GraphicImport::lcl_info(const std::string& /*info*/) { } void GraphicImport::lcl_startShape(uno::Reference const&) { } void GraphicImport::lcl_endShape( ) { } bool GraphicImport::IsGraphic() const { return m_pImpl->bIsGraphic; } } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */