/* -*- 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 "shapeimpl.hxx" #include "svx/unoshprp.hxx" #include #include "svx/svdviter.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::osl; using namespace ::cppu; using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::container; #define QUERYINT( xint ) \ if( rType == cppu::UnoType::get() ) \ aAny <<= Reference< xint >(this) SvxShapeGroup::SvxShapeGroup( SdrObject* pObj, SvxDrawPage* pDrawPage ) throw() : SvxShape( pObj, getSvxMapProvider().GetMap(SVXMAP_GROUP), getSvxMapProvider().GetPropertySet(SVXMAP_GROUP, SdrObject::GetGlobalDrawObjectItemPool()) ), mxPage( pDrawPage ) { } SvxShapeGroup::~SvxShapeGroup() throw() { } void SvxShapeGroup::Create( SdrObject* pNewObj, SvxDrawPage* pNewPage ) { SvxShape::Create( pNewObj, pNewPage ); mxPage = pNewPage; } uno::Any SAL_CALL SvxShapeGroup::queryInterface( const uno::Type & rType ) { return SvxShape::queryInterface( rType ); } uno::Any SAL_CALL SvxShapeGroup::queryAggregation( const uno::Type & rType ) { uno::Any aAny; QUERYINT( drawing::XShapeGroup ); else QUERYINT( drawing::XShapes ); else QUERYINT( drawing::XShapes2 ); else QUERYINT( container::XIndexAccess ); else QUERYINT( container::XElementAccess ); else return SvxShape::queryAggregation( rType ); return aAny; } void SAL_CALL SvxShapeGroup::acquire() throw ( ) { SvxShape::acquire(); } void SAL_CALL SvxShapeGroup::release() throw ( ) { SvxShape::release(); } uno::Sequence< sal_Int8 > SAL_CALL SvxShapeGroup::getImplementationId() { return css::uno::Sequence(); } // css::drawing::XShape OUString SAL_CALL SvxShapeGroup::getShapeType() { return SvxShape::getShapeType(); } awt::Point SAL_CALL SvxShapeGroup::getPosition() { return SvxShape::getPosition(); } void SAL_CALL SvxShapeGroup::setPosition( const awt::Point& Position ) { SvxShape::setPosition(Position); } awt::Size SAL_CALL SvxShapeGroup::getSize() { return SvxShape::getSize(); } void SAL_CALL SvxShapeGroup::setSize( const awt::Size& rSize ) { SvxShape::setSize( rSize ); } // drawing::XShapeGroup void SAL_CALL SvxShapeGroup::enterGroup( ) { // Todo // pDrView->EnterMarkedGroup(); } void SAL_CALL SvxShapeGroup::leaveGroup( ) { // Todo // pDrView->LeaveOneGroup(); } void SvxShapeGroup::addUnoShape( const uno::Reference< drawing::XShape >& xShape, size_t nPos ) { if (!mpObj.is() || !mxPage.is()) { OSL_FAIL("could not add XShape to group shape!"); return; } SvxShape* pShape = SvxShape::getImplementation( xShape ); if (!pShape) { OSL_FAIL("could not add XShape to group shape!"); return; } SdrObject* pSdrShape = pShape->GetSdrObject(); if( pSdrShape == nullptr ) pSdrShape = mxPage->CreateSdrObject_( xShape ); if( pSdrShape->IsInserted() ) pSdrShape->GetObjList()->RemoveObject( pSdrShape->GetOrdNum() ); mpObj->GetSubList()->InsertObject(pSdrShape, nPos); pSdrShape->SetModel(mpObj->GetModel()); // #85922# It makes no sense to set the layer asked // from the group object since these is an iteration // over the contained objects. In consequence, this // statement erases all layer information from the draw // objects. Layers need to be set at draw objects directly // and have nothing to do with grouping at all. // pSdrShape->SetLayer(pObject->GetLayer()); // Establish connection between new SdrObject and its wrapper before // inserting the new shape into the group. There a new wrapper // would be created when this connection would not already exist. pShape->Create( pSdrShape, mxPage.get() ); if( mpModel ) mpModel->SetChanged(); } // XShapes void SAL_CALL SvxShapeGroup::add( const uno::Reference< drawing::XShape >& xShape ) { ::SolarMutexGuard aGuard; // Add to the top of the stack (i.e. bottom of the list) by default. addUnoShape(xShape, SAL_MAX_SIZE); } void SAL_CALL SvxShapeGroup::remove( const uno::Reference< drawing::XShape >& xShape ) { ::SolarMutexGuard aGuard; SdrObject* pSdrShape = nullptr; SvxShape* pShape = SvxShape::getImplementation( xShape ); if( pShape ) pSdrShape = pShape->GetSdrObject(); if( !mpObj.is() || pSdrShape == nullptr || pSdrShape->GetObjList()->GetOwnerObj() != mpObj.get() ) throw uno::RuntimeException(); SdrObjList& rList = *pSdrShape->GetObjList(); const size_t nObjCount = rList.GetObjCount(); size_t nObjNum = 0; while( nObjNum < nObjCount ) { if(rList.GetObj( nObjNum ) == pSdrShape ) break; nObjNum++; } if( nObjNum < nObjCount ) { // #i29181# // If the SdrObject which is about to be deleted is in any selection, // deselect it first. SdrViewIter aIter( pSdrShape ); for ( SdrView* pView = aIter.FirstView(); pView; pView = aIter.NextView() ) { if(SAL_MAX_SIZE != pView->TryToFindMarkedObject(pSdrShape)) { pView->MarkObj(pSdrShape, pView->GetSdrPageView(), true); } } SdrObject* pObject = rList.NbcRemoveObject( nObjNum ); SdrObject::Free( pObject ); } else { SAL_WARN( "svx", "Fatality! SdrObject is not belonging to its SdrObjList! [CL]" ); } if( mpModel ) mpModel->SetChanged(); } void SAL_CALL SvxShapeGroup::addTop( const uno::Reference< drawing::XShape >& xShape ) { SolarMutexGuard aGuard; // Add to the top of the stack (i.e. bottom of the list). addUnoShape(xShape, SAL_MAX_SIZE); } void SAL_CALL SvxShapeGroup::addBottom( const uno::Reference< drawing::XShape >& xShape ) { SolarMutexGuard aGuard; // Add to the bottom of the stack (i.e. top of the list). addUnoShape(xShape, 0); } // XIndexAccess sal_Int32 SAL_CALL SvxShapeGroup::getCount() { ::SolarMutexGuard aGuard; sal_Int32 nRetval = 0; if(mpObj.is() && mpObj->GetSubList()) nRetval = mpObj->GetSubList()->GetObjCount(); else throw uno::RuntimeException(); return nRetval; } uno::Any SAL_CALL SvxShapeGroup::getByIndex( sal_Int32 Index ) { ::SolarMutexGuard aGuard; if( !mpObj.is() || mpObj->GetSubList() == nullptr ) throw uno::RuntimeException(); if( Index<0 || mpObj->GetSubList()->GetObjCount() <= static_cast(Index) ) throw lang::IndexOutOfBoundsException(); SdrObject* pDestObj = mpObj->GetSubList()->GetObj( Index ); if(pDestObj == nullptr) throw lang::IndexOutOfBoundsException(); Reference< drawing::XShape > xShape( pDestObj->getUnoShape(), uno::UNO_QUERY ); return uno::makeAny( xShape ); } // css::container::XElementAccess uno::Type SAL_CALL SvxShapeGroup::getElementType() { return cppu::UnoType::get(); } sal_Bool SAL_CALL SvxShapeGroup::hasElements() { ::SolarMutexGuard aGuard; return mpObj.is() && mpObj->GetSubList() && (mpObj->GetSubList()->GetObjCount() > 0); } SvxShapeConnector::SvxShapeConnector( SdrObject* pObj ) throw() : SvxShapeText( pObj, getSvxMapProvider().GetMap(SVXMAP_CONNECTOR), getSvxMapProvider().GetPropertySet(SVXMAP_CONNECTOR, SdrObject::GetGlobalDrawObjectItemPool()) ) { } SvxShapeConnector::~SvxShapeConnector() throw() { } uno::Any SAL_CALL SvxShapeConnector::queryInterface( const uno::Type & rType ) { return SvxShapeText::queryInterface( rType ); } uno::Any SAL_CALL SvxShapeConnector::queryAggregation( const uno::Type & rType ) { uno::Any aAny; QUERYINT( drawing::XConnectorShape ); else return SvxShapeText::queryAggregation( rType ); return aAny; } void SAL_CALL SvxShapeConnector::acquire() throw ( ) { SvxShapeText::acquire(); } void SAL_CALL SvxShapeConnector::release() throw ( ) { SvxShapeText::release(); } // XTypeProvider uno::Sequence< uno::Type > SAL_CALL SvxShapeConnector::getTypes() { return SvxShape::getTypes(); } uno::Sequence< sal_Int8 > SAL_CALL SvxShapeConnector::getImplementationId() { return css::uno::Sequence(); } // css::drawing::XShape OUString SAL_CALL SvxShapeConnector::getShapeType() { return SvxShapeText::getShapeType(); } awt::Point SAL_CALL SvxShapeConnector::getPosition() { return SvxShapeText::getPosition(); } void SAL_CALL SvxShapeConnector::setPosition( const awt::Point& Position ) { SvxShapeText::setPosition(Position); } awt::Size SAL_CALL SvxShapeConnector::getSize() { return SvxShapeText::getSize(); } void SAL_CALL SvxShapeConnector::setSize( const awt::Size& rSize ) { SvxShapeText::setSize( rSize ); } // XConnectorShape void SAL_CALL SvxShapeConnector::connectStart( const uno::Reference< drawing::XConnectableShape >& xShape, drawing::ConnectionType ) { ::SolarMutexGuard aGuard; Reference< drawing::XShape > xRef( xShape, UNO_QUERY ); SvxShape* pShape = SvxShape::getImplementation( xRef ); if( pShape ) mpObj->ConnectToNode( true, pShape->mpObj.get() ); if( mpModel ) mpModel->SetChanged(); } void SAL_CALL SvxShapeConnector::connectEnd( const uno::Reference< drawing::XConnectableShape >& xShape, drawing::ConnectionType ) { ::SolarMutexGuard aGuard; Reference< drawing::XShape > xRef( xShape, UNO_QUERY ); SvxShape* pShape = SvxShape::getImplementation( xRef ); if( mpObj.is() && pShape ) mpObj->ConnectToNode( false, pShape->mpObj.get() ); if( mpModel ) mpModel->SetChanged(); } void SAL_CALL SvxShapeConnector::disconnectBegin( const uno::Reference< drawing::XConnectableShape >& ) { ::SolarMutexGuard aGuard; if(mpObj.is()) mpObj->DisconnectFromNode( true ); if( mpModel ) mpModel->SetChanged(); } void SAL_CALL SvxShapeConnector::disconnectEnd( const uno::Reference< drawing::XConnectableShape >& ) { ::SolarMutexGuard aGuard; if(mpObj.is()) mpObj->DisconnectFromNode( false ); if( mpModel ) mpModel->SetChanged(); } SvxShapeControl::SvxShapeControl( SdrObject* pObj ) throw() : SvxShapeText( pObj, getSvxMapProvider().GetMap(SVXMAP_CONTROL), getSvxMapProvider().GetPropertySet(SVXMAP_CONTROL, SdrObject::GetGlobalDrawObjectItemPool()) ) { setShapeKind( OBJ_UNO ); } SvxShapeControl::~SvxShapeControl() throw() { } uno::Any SAL_CALL SvxShapeControl::queryInterface( const uno::Type & rType ) { return SvxShapeText::queryInterface( rType ); } uno::Any SAL_CALL SvxShapeControl::queryAggregation( const uno::Type & rType ) { uno::Any aAny; QUERYINT( drawing::XControlShape ); else return SvxShapeText::queryAggregation( rType ); return aAny; } void SAL_CALL SvxShapeControl::acquire() throw ( ) { SvxShapeText::acquire(); } void SAL_CALL SvxShapeControl::release() throw ( ) { SvxShapeText::release(); } // XTypeProvider uno::Sequence< uno::Type > SAL_CALL SvxShapeControl::getTypes() { return SvxShape::getTypes(); } uno::Sequence< sal_Int8 > SAL_CALL SvxShapeControl::getImplementationId() { return css::uno::Sequence(); } // css::drawing::XShape OUString SAL_CALL SvxShapeControl::getShapeType() { return SvxShapeText::getShapeType(); } awt::Point SAL_CALL SvxShapeControl::getPosition() { return SvxShapeText::getPosition(); } void SAL_CALL SvxShapeControl::setPosition( const awt::Point& Position ) { SvxShapeText::setPosition(Position); } awt::Size SAL_CALL SvxShapeControl::getSize() { return SvxShapeText::getSize(); } void SAL_CALL SvxShapeControl::setSize( const awt::Size& rSize ) { SvxShapeText::setSize( rSize ); } // XControlShape Reference< awt::XControlModel > SAL_CALL SvxShapeControl::getControl() { ::SolarMutexGuard aGuard; Reference< awt::XControlModel > xModel; SdrUnoObj* pUnoObj = dynamic_cast< SdrUnoObj * >(mpObj.get()); if( pUnoObj ) xModel = pUnoObj->GetUnoControlModel(); return xModel; } void SAL_CALL SvxShapeControl::setControl( const Reference< awt::XControlModel >& xControl ) { ::SolarMutexGuard aGuard; SdrUnoObj* pUnoObj = dynamic_cast< SdrUnoObj * >(mpObj.get()); if( pUnoObj ) pUnoObj->SetUnoControlModel( xControl ); if( mpModel ) mpModel->SetChanged(); } static struct { const sal_Char* mpAPIName; sal_uInt16 mnAPINameLen; const sal_Char* mpFormName; sal_uInt16 mnFormNameLen; } SvxShapeControlPropertyMapping[] = { // Warning: The first entry must be FontSlant because the any needs to be converted { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_POSTURE), RTL_CONSTASCII_STRINGPARAM("FontSlant") }, // const sal_Int16 => css::awt::FontSlant { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_FONTNAME), RTL_CONSTASCII_STRINGPARAM("FontName") }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_FONTSTYLENAME), RTL_CONSTASCII_STRINGPARAM("FontStyleName") }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_FONTFAMILY), RTL_CONSTASCII_STRINGPARAM("FontFamily") }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_FONTCHARSET), RTL_CONSTASCII_STRINGPARAM("FontCharset") }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_HEIGHT), RTL_CONSTASCII_STRINGPARAM("FontHeight") }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_FONTPITCH), RTL_CONSTASCII_STRINGPARAM("FontPitch" ) }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_WEIGHT), RTL_CONSTASCII_STRINGPARAM("FontWeight" ) }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_UNDERLINE), RTL_CONSTASCII_STRINGPARAM("FontUnderline") }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_STRIKEOUT), RTL_CONSTASCII_STRINGPARAM("FontStrikeout") }, { RTL_CONSTASCII_STRINGPARAM("CharKerning"), RTL_CONSTASCII_STRINGPARAM("FontKerning") }, { RTL_CONSTASCII_STRINGPARAM("CharWordMode"), RTL_CONSTASCII_STRINGPARAM("FontWordLineMode" ) }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_CHAR_COLOR), RTL_CONSTASCII_STRINGPARAM("TextColor") }, { RTL_CONSTASCII_STRINGPARAM("CharBackColor"), RTL_CONSTASCII_STRINGPARAM("CharBackColor") }, { RTL_CONSTASCII_STRINGPARAM("CharBackTransparent"), RTL_CONSTASCII_STRINGPARAM("CharBackTransparent") }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_TEXT_CHAINNEXTNAME), RTL_CONSTASCII_STRINGPARAM(UNO_NAME_TEXT_CHAINNEXTNAME) }, { RTL_CONSTASCII_STRINGPARAM("CharRelief"), RTL_CONSTASCII_STRINGPARAM("FontRelief") }, { RTL_CONSTASCII_STRINGPARAM("CharUnderlineColor"), RTL_CONSTASCII_STRINGPARAM("TextLineColor") }, { RTL_CONSTASCII_STRINGPARAM(UNO_NAME_EDIT_PARA_ADJUST), RTL_CONSTASCII_STRINGPARAM("Align") }, { RTL_CONSTASCII_STRINGPARAM("TextVerticalAdjust"), RTL_CONSTASCII_STRINGPARAM("VerticalAlign") }, { RTL_CONSTASCII_STRINGPARAM("ControlBackground"), RTL_CONSTASCII_STRINGPARAM("BackgroundColor") }, { RTL_CONSTASCII_STRINGPARAM("ControlSymbolColor"), RTL_CONSTASCII_STRINGPARAM("SymbolColor") }, { RTL_CONSTASCII_STRINGPARAM("ControlBorder"), RTL_CONSTASCII_STRINGPARAM("Border") }, { RTL_CONSTASCII_STRINGPARAM("ControlBorderColor"), RTL_CONSTASCII_STRINGPARAM("BorderColor") }, { RTL_CONSTASCII_STRINGPARAM("ControlTextEmphasis"), RTL_CONSTASCII_STRINGPARAM("FontEmphasisMark") }, { RTL_CONSTASCII_STRINGPARAM("ImageScaleMode"), RTL_CONSTASCII_STRINGPARAM("ScaleMode") }, { RTL_CONSTASCII_STRINGPARAM("ControlWritingMode"), RTL_CONSTASCII_STRINGPARAM("WritingMode") }, //added for exporting OCX control { RTL_CONSTASCII_STRINGPARAM("ControlTypeinMSO"), RTL_CONSTASCII_STRINGPARAM("ControlTypeinMSO") }, { RTL_CONSTASCII_STRINGPARAM("ObjIDinMSO"), RTL_CONSTASCII_STRINGPARAM("ObjIDinMSO") }, { RTL_CONSTASCII_STRINGPARAM("CharCaseMap"), RTL_CONSTASCII_STRINGPARAM("CharCaseMap") }, { nullptr,0, nullptr, 0 } }; namespace { bool lcl_convertPropertyName( const OUString& rApiName, OUString& rInternalName ) { sal_uInt16 i = 0; while( SvxShapeControlPropertyMapping[i].mpAPIName ) { if( rApiName.reverseCompareToAsciiL( SvxShapeControlPropertyMapping[i].mpAPIName, SvxShapeControlPropertyMapping[i].mnAPINameLen ) == 0 ) { rInternalName = OUString( SvxShapeControlPropertyMapping[i].mpFormName, SvxShapeControlPropertyMapping[i].mnFormNameLen, RTL_TEXTENCODING_ASCII_US ); } ++i; } return !rInternalName.isEmpty(); } struct EnumConversionMap { style::ParagraphAdjust nAPIValue; sal_Int16 nFormValue; }; EnumConversionMap aMapAdjustToAlign[] = { // note that order matters: // lcl_convertTextAlignmentToParaAdjustment and lcl_convertParaAdjustmentToTextAlignment search this map from the _beginning_ // and use the first matching entry {style::ParagraphAdjust_LEFT, (sal_Int16)awt::TextAlign::LEFT}, {style::ParagraphAdjust_CENTER, (sal_Int16)awt::TextAlign::CENTER}, {style::ParagraphAdjust_RIGHT, (sal_Int16)awt::TextAlign::RIGHT}, {style::ParagraphAdjust_BLOCK, (sal_Int16)awt::TextAlign::RIGHT}, {style::ParagraphAdjust_STRETCH, (sal_Int16)awt::TextAlign::LEFT}, {(style::ParagraphAdjust)-1,-1} }; void lcl_convertTextAlignmentToParaAdjustment( Any& _rValue ) { sal_Int16 nValue = sal_Int16(); OSL_VERIFY( _rValue >>= nValue ); for ( auto const & rEntry : aMapAdjustToAlign ) if ( nValue == rEntry.nFormValue ) { _rValue <<= rEntry.nAPIValue; return; } } void lcl_convertParaAdjustmentToTextAlignment( Any& _rValue ) { sal_Int32 nValue = 0; OSL_VERIFY( _rValue >>= nValue ); for ( auto const & rEntry : aMapAdjustToAlign ) if ( (style::ParagraphAdjust)nValue == rEntry.nAPIValue ) { _rValue <<= rEntry.nFormValue; return; } } void convertVerticalAdjustToVerticalAlign( Any& _rValue ) { if ( !_rValue.hasValue() ) return; drawing::TextVerticalAdjust eAdjust = drawing::TextVerticalAdjust_TOP; style::VerticalAlignment eAlign = style::VerticalAlignment_TOP; if ( !( _rValue >>= eAdjust ) ) throw lang::IllegalArgumentException(); switch ( eAdjust ) { case drawing::TextVerticalAdjust_TOP: eAlign = style::VerticalAlignment_TOP; break; case drawing::TextVerticalAdjust_BOTTOM: eAlign = style::VerticalAlignment_BOTTOM; break; default: eAlign = style::VerticalAlignment_MIDDLE; break; } _rValue <<= eAlign; } void convertVerticalAlignToVerticalAdjust( Any& _rValue ) { if ( !_rValue.hasValue() ) return; style::VerticalAlignment eAlign = style::VerticalAlignment_TOP; drawing::TextVerticalAdjust eAdjust = drawing::TextVerticalAdjust_TOP; OSL_VERIFY( _rValue >>= eAlign ); switch ( eAlign ) { case style::VerticalAlignment_TOP: eAdjust = drawing::TextVerticalAdjust_TOP; break; case style::VerticalAlignment_BOTTOM: eAdjust = drawing::TextVerticalAdjust_BOTTOM; break; default: eAdjust = drawing::TextVerticalAdjust_CENTER; break; } _rValue <<= eAdjust; } } void SAL_CALL SvxShapeControl::setPropertyValue( const OUString& aPropertyName, const uno::Any& aValue ) { OUString aFormsName; if ( lcl_convertPropertyName( aPropertyName, aFormsName ) ) { uno::Reference< beans::XPropertySet > xControl( getControl(), uno::UNO_QUERY ); if( xControl.is() ) { uno::Reference< beans::XPropertySetInfo > xInfo( xControl->getPropertySetInfo() ); if( xInfo.is() && xInfo->hasPropertyByName( aFormsName ) ) { uno::Any aConvertedValue( aValue ); if ( aFormsName == "FontSlant" ) { awt::FontSlant nSlant; if( !(aValue >>= nSlant ) ) throw lang::IllegalArgumentException(); aConvertedValue <<= (sal_Int16)nSlant; } else if ( aFormsName == "Align" ) { lcl_convertParaAdjustmentToTextAlignment( aConvertedValue ); } else if ( aFormsName == "VerticalAlign" ) { convertVerticalAdjustToVerticalAlign( aConvertedValue ); } xControl->setPropertyValue( aFormsName, aConvertedValue ); } } } else { SvxShape::setPropertyValue( aPropertyName, aValue ); } } uno::Any SAL_CALL SvxShapeControl::getPropertyValue( const OUString& aPropertyName ) { OUString aFormsName; if ( lcl_convertPropertyName( aPropertyName, aFormsName ) ) { uno::Reference< beans::XPropertySet > xControl( getControl(), uno::UNO_QUERY ); uno::Any aValue; if( xControl.is() ) { uno::Reference< beans::XPropertySetInfo > xInfo( xControl->getPropertySetInfo() ); if( xInfo.is() && xInfo->hasPropertyByName( aFormsName ) ) { aValue = xControl->getPropertyValue( aFormsName ); if ( aFormsName == "FontSlant" ) { awt::FontSlant eSlant = awt::FontSlant_NONE; sal_Int16 nSlant = sal_Int16(); if ( aValue >>= nSlant ) { eSlant = (awt::FontSlant)nSlant; } else { OSL_VERIFY( aValue >>= eSlant ); } aValue <<= eSlant; } else if ( aFormsName == "Align" ) { lcl_convertTextAlignmentToParaAdjustment( aValue ); } else if ( aFormsName == "VerticalAlign" ) { convertVerticalAlignToVerticalAdjust( aValue ); } } } return aValue; } else { return SvxShape::getPropertyValue( aPropertyName ); } } // XPropertyState beans::PropertyState SAL_CALL SvxShapeControl::getPropertyState( const OUString& PropertyName ) { OUString aFormsName; if ( lcl_convertPropertyName( PropertyName, aFormsName ) ) { uno::Reference< beans::XPropertyState > xControl( getControl(), uno::UNO_QUERY ); uno::Reference< beans::XPropertySet > xPropSet( getControl(), uno::UNO_QUERY ); if( xControl.is() && xPropSet.is() ) { uno::Reference< beans::XPropertySetInfo > xInfo( xPropSet->getPropertySetInfo() ); if( xInfo.is() && xInfo->hasPropertyByName( aFormsName ) ) { return xControl->getPropertyState( aFormsName ); } } return beans::PropertyState_DEFAULT_VALUE; } else { return SvxShape::getPropertyState( PropertyName ); } } void SAL_CALL SvxShapeControl::setPropertyToDefault( const OUString& PropertyName ) { OUString aFormsName; if ( lcl_convertPropertyName( PropertyName, aFormsName ) ) { uno::Reference< beans::XPropertyState > xControl( getControl(), uno::UNO_QUERY ); uno::Reference< beans::XPropertySet > xPropSet( getControl(), uno::UNO_QUERY ); if( xControl.is() && xPropSet.is() ) { uno::Reference< beans::XPropertySetInfo > xInfo( xPropSet->getPropertySetInfo() ); if( xInfo.is() && xInfo->hasPropertyByName( aFormsName ) ) { xControl->setPropertyToDefault( aFormsName ); } } } else { SvxShape::setPropertyToDefault( PropertyName ); } } uno::Any SAL_CALL SvxShapeControl::getPropertyDefault( const OUString& aPropertyName ) { OUString aFormsName; if ( lcl_convertPropertyName( aPropertyName, aFormsName ) ) { uno::Reference< beans::XPropertyState > xControl( getControl(), uno::UNO_QUERY ); if( xControl.is() ) { Any aDefault( xControl->getPropertyDefault( aFormsName ) ); if ( aFormsName == "FontSlant" ) { sal_Int16 nSlant( 0 ); aDefault >>= nSlant; aDefault <<= (awt::FontSlant)nSlant; } else if ( aFormsName == "Align" ) { lcl_convertTextAlignmentToParaAdjustment( aDefault ); } else if ( aFormsName == "VerticalAlign" ) { convertVerticalAlignToVerticalAdjust( aDefault ); } return aDefault; } throw beans::UnknownPropertyException( aPropertyName, static_cast(this)); } else { return SvxShape::getPropertyDefault( aPropertyName ); } } SvxShapeDimensioning::SvxShapeDimensioning( SdrObject* pObj ) throw() : SvxShapeText( pObj, getSvxMapProvider().GetMap(SVXMAP_DIMENSIONING), getSvxMapProvider().GetPropertySet(SVXMAP_DIMENSIONING, SdrObject::GetGlobalDrawObjectItemPool()) ) { } SvxShapeDimensioning::~SvxShapeDimensioning() throw() { } SvxShapeCircle::SvxShapeCircle( SdrObject* pObj ) throw() : SvxShapeText( pObj, getSvxMapProvider().GetMap(SVXMAP_CIRCLE), getSvxMapProvider().GetPropertySet(SVXMAP_CIRCLE, SdrObject::GetGlobalDrawObjectItemPool()) ) { } SvxShapeCircle::~SvxShapeCircle() throw() { } SvxShapePolyPolygon::SvxShapePolyPolygon( SdrObject* pObj , drawing::PolygonKind eNew ) : SvxShapeText( pObj, getSvxMapProvider().GetMap(SVXMAP_POLYPOLYGON), getSvxMapProvider().GetPropertySet(SVXMAP_POLYPOLYGON, SdrObject::GetGlobalDrawObjectItemPool()) ) , mePolygonKind( eNew ) { } SvxShapePolyPolygon::~SvxShapePolyPolygon() throw() { } basegfx::B2DPolyPolygon SAL_CALL ImplSvxPointSequenceSequenceToB2DPolyPolygon( const drawing::PointSequenceSequence* pOuterSequence) throw() { basegfx::B2DPolyPolygon aRetval; // get pointer to internal sequences const drawing::PointSequence* pInnerSequence = pOuterSequence->getConstArray(); const drawing::PointSequence* pInnerSeqEnd = pInnerSequence + pOuterSequence->getLength(); for(;pInnerSequence != pInnerSeqEnd; ++pInnerSequence) { // prepare new polygon basegfx::B2DPolygon aNewPolygon; // get pointer to arrays const awt::Point* pArray = pInnerSequence->getConstArray(); const awt::Point* pArrayEnd = pArray + pInnerSequence->getLength(); for(;pArray != pArrayEnd;++pArray) { aNewPolygon.append(basegfx::B2DPoint(pArray->X, pArray->Y)); } // check for closed state flag basegfx::tools::checkClosed(aNewPolygon); // add new subpolygon aRetval.append(aNewPolygon); } return aRetval; } bool SvxShapePolyPolygon::setPropertyValueImpl( const OUString& rName, const SfxItemPropertySimpleEntry* pProperty, const css::uno::Any& rValue ) { switch( pProperty->nWID ) { case OWN_ATTR_VALUE_POLYPOLYGON: { if( auto s = o3tl::tryAccess(rValue) ) { basegfx::B2DPolyPolygon aNewPolyPolygon(ImplSvxPointSequenceSequenceToB2DPolyPolygon(s)); SetPolygon(aNewPolyPolygon); return true; } break; } case OWN_ATTR_BASE_GEOMETRY: { if( auto s = o3tl::tryAccess(rValue) ) { if( mpObj.is() ) { basegfx::B2DPolyPolygon aNewPolyPolygon; basegfx::B2DHomMatrix aNewHomogenMatrix; mpObj->TRGetBaseGeometry(aNewHomogenMatrix, aNewPolyPolygon); aNewPolyPolygon = ImplSvxPointSequenceSequenceToB2DPolyPolygon(s); mpObj->TRSetBaseGeometry(aNewHomogenMatrix, aNewPolyPolygon); } return true; } break; } case OWN_ATTR_VALUE_POLYGON: { if( auto pSequence = o3tl::tryAccess(rValue) ) { // prepare new polygon basegfx::B2DPolygon aNewPolygon; // get pointer to arrays const awt::Point* pArray = pSequence->getConstArray(); const awt::Point* pArrayEnd = pArray + pSequence->getLength(); for(;pArray != pArrayEnd;++pArray) { aNewPolygon.append(basegfx::B2DPoint(pArray->X, pArray->Y)); } // check for closed state flag basegfx::tools::checkClosed(aNewPolygon); // set polygon SetPolygon(basegfx::B2DPolyPolygon(aNewPolygon)); return true; } break; } default: return SvxShapeText::setPropertyValueImpl( rName, pProperty, rValue ); } throw lang::IllegalArgumentException(); } void SAL_CALL B2DPolyPolygonToSvxPointSequenceSequence( const basegfx::B2DPolyPolygon& rPolyPoly, drawing::PointSequenceSequence& rRetval ) { if( (sal_uInt32)rRetval.getLength() != rPolyPoly.count() ) rRetval.realloc( rPolyPoly.count() ); // get pointer to external arrays drawing::PointSequence* pOuterSequence = rRetval.getArray(); for(sal_uInt32 a(0L); a < rPolyPoly.count(); a++) { // get single polygon const basegfx::B2DPolygon aPoly(rPolyPoly.getB2DPolygon(a)); // #i75974# take closed state into account, the API polygon still uses the old closed definition // with last/first point are identical (cannot hold information about open polygons with identical // first and last point, though) const sal_uInt32 nPointCount(aPoly.count()); const bool bIsClosed(aPoly.isClosed()); // create space in arrays pOuterSequence->realloc(bIsClosed ? nPointCount + 1 : nPointCount); // get pointer to arrays awt::Point* pInnerSequence = pOuterSequence->getArray(); for(sal_uInt32 b(0L); b < nPointCount; b++) { const basegfx::B2DPoint aPoint(aPoly.getB2DPoint(b)); *pInnerSequence = awt::Point( basegfx::fround(aPoint.getX()), basegfx::fround(aPoint.getY()) ); pInnerSequence++; } // #i75974# copy first point if(bIsClosed) { *pInnerSequence = *pOuterSequence->getArray(); } pOuterSequence++; } } bool SvxShapePolyPolygon::getPropertyValueImpl( const OUString& rName, const SfxItemPropertySimpleEntry* pProperty, css::uno::Any& rValue ) { switch( pProperty->nWID ) { case OWN_ATTR_VALUE_POLYPOLYGON: { // pack a tools::PolyPolygon in a struct tools::PolyPolygon const basegfx::B2DPolyPolygon& rPolyPoly = GetPolygon(); drawing::PointSequenceSequence aRetval( rPolyPoly.count() ); B2DPolyPolygonToSvxPointSequenceSequence( rPolyPoly, aRetval ); rValue <<= aRetval; break; } case OWN_ATTR_BASE_GEOMETRY: { // pack a tools::PolyPolygon in struct PolyPolygon basegfx::B2DPolyPolygon aNewPolyPolygon; basegfx::B2DHomMatrix aNewHomogenMatrix; if(mpObj.is()) mpObj->TRGetBaseGeometry(aNewHomogenMatrix, aNewPolyPolygon); drawing::PointSequenceSequence aRetval(aNewPolyPolygon.count()); B2DPolyPolygonToSvxPointSequenceSequence(aNewPolyPolygon, aRetval); rValue <<= aRetval; break; } case OWN_ATTR_VALUE_POLYGON: { // pack a tools::PolyPolygon in a struct tools::PolyPolygon const basegfx::B2DPolyPolygon& rPolyPoly = GetPolygon(); sal_Int32 nCount = 0; if( rPolyPoly.count() > 0 ) nCount = rPolyPoly.getB2DPolygon(0L).count(); drawing::PointSequence aRetval( nCount ); if( nCount > 0 ) { // get single polygon const basegfx::B2DPolygon aPoly(rPolyPoly.getB2DPolygon(0L)); // get pointer to arrays awt::Point* pSequence = aRetval.getArray(); for(sal_Int32 b=0;b(mpObj.get())->SetPathPoly(rNew); } basegfx::B2DPolyPolygon SvxShapePolyPolygon::GetPolygon() const throw() { ::SolarMutexGuard aGuard; if(mpObj.is()) { return static_cast(mpObj.get())->GetPathPoly(); } else { return basegfx::B2DPolyPolygon(); } } SvxShapePolyPolygonBezier::SvxShapePolyPolygonBezier( SdrObject* pObj , drawing::PolygonKind eNew ) throw() : SvxShapeText( pObj, getSvxMapProvider().GetMap(SVXMAP_POLYPOLYGONBEZIER), getSvxMapProvider().GetPropertySet(SVXMAP_POLYPOLYGONBEZIER, SdrObject::GetGlobalDrawObjectItemPool()) ) , mePolygonKind( eNew ) { } SvxShapePolyPolygonBezier::~SvxShapePolyPolygonBezier() throw() { } bool SvxShapePolyPolygonBezier::setPropertyValueImpl( const OUString& rName, const SfxItemPropertySimpleEntry* pProperty, const css::uno::Any& rValue ) { switch( pProperty->nWID ) { case OWN_ATTR_VALUE_POLYPOLYGONBEZIER: { if( auto s = o3tl::tryAccess(rValue) ) { basegfx::B2DPolyPolygon aNewPolyPolygon( basegfx::unotools::polyPolygonBezierToB2DPolyPolygon(*s)); SetPolygon(aNewPolyPolygon); return true; } break; } case OWN_ATTR_BASE_GEOMETRY: { if( auto s = o3tl::tryAccess(rValue) ) { if( mpObj.is() ) { basegfx::B2DPolyPolygon aNewPolyPolygon; basegfx::B2DHomMatrix aNewHomogenMatrix; mpObj->TRGetBaseGeometry(aNewHomogenMatrix, aNewPolyPolygon); aNewPolyPolygon = basegfx::unotools::polyPolygonBezierToB2DPolyPolygon( *s); mpObj->TRSetBaseGeometry(aNewHomogenMatrix, aNewPolyPolygon); } return true; } break; } default: return SvxShapeText::setPropertyValueImpl( rName, pProperty, rValue ); } throw IllegalArgumentException(); } bool SvxShapePolyPolygonBezier::getPropertyValueImpl( const OUString& rName, const SfxItemPropertySimpleEntry* pProperty, css::uno::Any& rValue ) { switch( pProperty->nWID ) { case OWN_ATTR_VALUE_POLYPOLYGONBEZIER: { // pack a tools::PolyPolygon in a struct tools::PolyPolygon const basegfx::B2DPolyPolygon& rPolyPoly = GetPolygon(); drawing::PolyPolygonBezierCoords aRetval; basegfx::unotools::b2DPolyPolygonToPolyPolygonBezier(rPolyPoly, aRetval); rValue <<= aRetval; break; } case OWN_ATTR_BASE_GEOMETRY: { // pack a tools::PolyPolygon in a struct tools::PolyPolygon basegfx::B2DPolyPolygon aNewPolyPolygon; basegfx::B2DHomMatrix aNewHomogenMatrix; mpObj.get()->TRGetBaseGeometry(aNewHomogenMatrix, aNewPolyPolygon); drawing::PolyPolygonBezierCoords aRetval; basegfx::unotools::b2DPolyPolygonToPolyPolygonBezier(aNewPolyPolygon, aRetval); rValue <<= aRetval; break; } case OWN_ATTR_VALUE_POLYGONKIND: { rValue <<= mePolygonKind; break; } default: return SvxShapeText::getPropertyValueImpl( rName, pProperty, rValue ); } return true; } void SvxShapePolyPolygonBezier::SetPolygon(const basegfx::B2DPolyPolygon& rNew) { ::SolarMutexGuard aGuard; if(mpObj.is()) static_cast(mpObj.get())->SetPathPoly(rNew); } basegfx::B2DPolyPolygon SvxShapePolyPolygonBezier::GetPolygon() const throw() { ::SolarMutexGuard aGuard; if(mpObj.is()) { return static_cast(mpObj.get())->GetPathPoly(); } else { return basegfx::B2DPolyPolygon(); } } SvxGraphicObject::SvxGraphicObject( SdrObject* pObj, OUString const & referer ) throw() : SvxShapeText( pObj, getSvxMapProvider().GetMap(SVXMAP_GRAPHICOBJECT), getSvxMapProvider().GetPropertySet(SVXMAP_GRAPHICOBJECT, SdrObject::GetGlobalDrawObjectItemPool()) ), referer_(referer) { } SvxGraphicObject::~SvxGraphicObject() throw() { } bool SvxGraphicObject::setPropertyValueImpl( const OUString& rName, const SfxItemPropertySimpleEntry* pProperty, const css::uno::Any& rValue ) { bool bOk = false; switch( pProperty->nWID ) { case OWN_ATTR_VALUE_FILLBITMAP: { if( auto pSeq = o3tl::tryAccess>(rValue) ) { SvMemoryStream aMemStm; Graphic aGraphic; aMemStm.SetBuffer( const_cast *>(pSeq)->getArray(), pSeq->getLength(), pSeq->getLength() ); if( GraphicConverter::Import( aMemStm, aGraphic ) == ERRCODE_NONE ) { static_cast(mpObj.get())->SetGraphic(aGraphic); bOk = true; } } else if( (rValue.getValueType() == cppu::UnoType::get()) || (rValue.getValueType() == cppu::UnoType::get())) { Reference< graphic::XGraphic> xGraphic( rValue, UNO_QUERY ); if( xGraphic.is() ) { static_cast(mpObj.get())->SetGraphic(Graphic(xGraphic)); bOk = true; } else { // pack bitmap in the object Reference< awt::XBitmap > xBmp( rValue, UNO_QUERY ); if( xBmp.is() ) { // apply bitmap Graphic aGraphic(VCLUnoHelper::GetBitmap( xBmp )); static_cast(mpObj.get())->SetGraphic(aGraphic); bOk = true; } } } break; } case OWN_ATTR_GRAFURL: { OUString aURL; if( rValue >>= aURL ) { if( aURL.startsWith( UNO_NAME_GRAPHOBJ_URLPREFIX ) ) { // graphic manager url aURL = aURL.copy( sizeof( UNO_NAME_GRAPHOBJ_URLPREFIX ) - 1 ); OUString aTmpStr(aURL); OString aUniqueID(OUStringToOString(aTmpStr, RTL_TEXTENCODING_UTF8)); GraphicObject aGrafObj(aUniqueID); // #101808# since loading a graphic can cause a reschedule of the office // it is possible that our shape is removed while where in this // method. if( mpObj.is() ) { static_cast(mpObj.get())->ReleaseGraphicLink(); static_cast(mpObj.get())->SetGraphicObject( aGrafObj ); } } else if( !aURL.startsWith( UNO_NAME_GRAPHOBJ_URLPKGPREFIX ) ) { // normal link OUString aFilterName; std::shared_ptr pSfxFilter; SfxMedium aSfxMedium( aURL, referer_, StreamMode::READ | StreamMode::SHARE_DENYNONE ); SfxGetpApp()->GetFilterMatcher().GuessFilter( aSfxMedium, pSfxFilter ); if( !pSfxFilter ) { INetURLObject aURLObj( aURL ); if( aURLObj.GetProtocol() == INetProtocol::NotValid ) { OUString aValidURL; if( osl::FileBase::getFileURLFromSystemPath( aURL, aValidURL ) == osl::FileBase::E_None ) aURLObj = INetURLObject( aValidURL ); } if( aURLObj.GetProtocol() != INetProtocol::NotValid ) { GraphicFilter &rGrfFilter = GraphicFilter::GetGraphicFilter(); aFilterName = rGrfFilter.GetImportFormatName( rGrfFilter.GetImportFormatNumberForShortName( aURLObj.getExtension() ) ); } } else aFilterName = pSfxFilter->GetFilterName(); // #101808# since loading a graphic can cause a reschedule of the office // it is possible that our shape is removed while where in this // method. if( mpObj.is() ) static_cast(mpObj.get())->SetGraphicLink( aURL, referer_, aFilterName ); } bOk = true; } break; } case OWN_ATTR_GRAFSTREAMURL: { OUString aStreamURL; if( rValue >>= aStreamURL ) { if( !aStreamURL.startsWith( UNO_NAME_GRAPHOBJ_URLPKGPREFIX ) ) aStreamURL.clear(); if( mpObj.is() ) { static_cast(mpObj.get())->SetGrafStreamURL( aStreamURL ); } bOk = true; } break; } case OWN_ATTR_VALUE_GRAPHIC: { Reference< graphic::XGraphic > xGraphic( rValue, uno::UNO_QUERY ); if( xGraphic.is() ) { static_cast< SdrGrafObj*>( mpObj.get() )->SetGraphic( xGraphic ); bOk = true; } break; } default: return SvxShapeText::setPropertyValueImpl( rName, pProperty, rValue ); } if( !bOk ) throw lang::IllegalArgumentException(); if( mpModel ) mpModel->SetChanged(); return true; } bool SvxGraphicObject::getPropertyValueImpl( const OUString& rName, const SfxItemPropertySimpleEntry* pProperty, css::uno::Any& rValue ) { switch( pProperty->nWID ) { case OWN_ATTR_VALUE_FILLBITMAP: { const Graphic& rGraphic = static_cast< SdrGrafObj*>( mpObj.get() )->GetGraphic(); if(rGraphic.GetType() != GraphicType::GdiMetafile) { // pack object in a bitmap Reference< css::awt::XBitmap > xBitmap( VCLUnoHelper::CreateBitmap(static_cast< SdrGrafObj*>( mpObj.get() )->GetGraphic().GetBitmapEx()) ); rValue <<= xBitmap; } else { SvMemoryStream aDestStrm( 65535, 65535 ); ConvertGDIMetaFileToWMF( rGraphic.GetGDIMetaFile(), aDestStrm, nullptr, false ); const uno::Sequence aSeq( static_cast< const sal_Int8* >(aDestStrm.GetData()), aDestStrm.GetEndOfData()); rValue <<= aSeq; } break; } case OWN_ATTR_GRAFURL: { if( static_cast< SdrGrafObj*>( mpObj.get() )->IsLinkedGraphic() ) { rValue <<= OUString( static_cast< SdrGrafObj*>( mpObj.get() )->GetFileName() ); } else { const GraphicObject& rGrafObj = static_cast< SdrGrafObj*>( mpObj.get() )->GetGraphicObject(true); OUString aURL( UNO_NAME_GRAPHOBJ_URLPREFIX); aURL += OStringToOUString(rGrafObj.GetUniqueID(), RTL_TEXTENCODING_ASCII_US); rValue <<= aURL; } break; } case OWN_ATTR_REPLACEMENTGRAFURL: { const GraphicObject* pGrafObj = static_cast< SdrGrafObj* >(mpObj.get())->GetReplacementGraphicObject(); if(pGrafObj) { OUString aURL(UNO_NAME_GRAPHOBJ_URLPREFIX); aURL += OStringToOUString(pGrafObj->GetUniqueID(), RTL_TEXTENCODING_ASCII_US); rValue <<= aURL; } break; } case OWN_ATTR_GRAFSTREAMURL: { const OUString aStreamURL( static_cast( mpObj.get() )->GetGrafStreamURL() ); if( !aStreamURL.isEmpty() ) rValue <<= aStreamURL; break; } case OWN_ATTR_VALUE_GRAPHIC: { Reference< graphic::XGraphic > xGraphic( static_cast< SdrGrafObj* >( mpObj.get() )->GetGraphic().GetXGraphic() ); rValue <<= xGraphic; break; } case OWN_ATTR_GRAPHIC_STREAM: { rValue <<= static_cast< SdrGrafObj* >( mpObj.get() )->getInputStream(); break; } default: return SvxShapeText::getPropertyValueImpl(rName, pProperty,rValue); } return true; } SvxShapeCaption::SvxShapeCaption( SdrObject* pObj ) throw() : SvxShapeText( pObj, getSvxMapProvider().GetMap(SVXMAP_CAPTION), getSvxMapProvider().GetPropertySet(SVXMAP_CAPTION, SdrObject::GetGlobalDrawObjectItemPool()) ) { } SvxShapeCaption::~SvxShapeCaption() throw() { } SvxCustomShape::SvxCustomShape( SdrObject* pObj ) throw() : SvxShapeText( pObj, getSvxMapProvider().GetMap( SVXMAP_CUSTOMSHAPE ), getSvxMapProvider().GetPropertySet(SVXMAP_CUSTOMSHAPE, SdrObject::GetGlobalDrawObjectItemPool()) ) { } SvxCustomShape::~SvxCustomShape() throw() { } uno::Any SAL_CALL SvxCustomShape::queryInterface( const uno::Type & rType ) { return SvxShapeText::queryInterface( rType ); } uno::Any SAL_CALL SvxCustomShape::queryAggregation( const uno::Type & rType ) { css::uno::Any aReturn = SvxShapeText::queryAggregation( rType ); if ( !aReturn.hasValue() ) aReturn = ::cppu::queryInterface(rType, static_cast(this) ); return aReturn; } void SAL_CALL SvxCustomShape::acquire() throw ( ) { SvxShapeText::acquire(); } void SAL_CALL SvxCustomShape::release() throw ( ) { SvxShapeText::release(); } uno::Sequence< uno::Type > SAL_CALL SvxCustomShape::getTypes() { return SvxShapeText::getTypes(); } uno::Sequence< sal_Int8 > SAL_CALL SvxCustomShape::getImplementationId() { return css::uno::Sequence(); } // css::drawing::XShape awt::Point SAL_CALL SvxCustomShape::getPosition() { ::SolarMutexGuard aGuard; if ( mpModel && mpObj.is() ) { SdrAShapeObjGeoData aCustomShapeGeoData; static_cast(mpObj.get())->SaveGeoData( aCustomShapeGeoData ); bool bMirroredX = false; bool bMirroredY = false; if ( mpObj.is() ) { bMirroredX = static_cast(mpObj.get())->IsMirroredX(); bMirroredY = static_cast(mpObj.get())->IsMirroredY(); } // get aRect, this is the unrotated snaprect tools::Rectangle aRect(static_cast(mpObj.get())->GetLogicRect()); tools::Rectangle aRectangle( aRect ); if ( bMirroredX || bMirroredY ) { // we have to retrieve the unmirrored rect GeoStat aNewGeo( aCustomShapeGeoData.aGeo ); if ( bMirroredX ) { tools::Polygon aPol( Rect2Poly( aRect, aNewGeo ) ); tools::Rectangle aBoundRect( aPol.GetBoundRect() ); Point aRef1( ( aBoundRect.Left() + aBoundRect.Right() ) >> 1, aBoundRect.Top() ); Point aRef2( aRef1.X(), aRef1.Y() + 1000 ); sal_uInt16 i; sal_uInt16 nPointCount=aPol.GetSize(); for (i=0; i> 1 ); Point aRef2( aRef1.X() + 1000, aRef1.Y() ); sal_uInt16 i; sal_uInt16 nPointCount=aPol.GetSize(); for (i=0; iIsWriter() ) aPt -= mpObj->GetAnchorPos(); ForceMetricTo100th_mm(aPt); return css::awt::Point( aPt.X(), aPt.Y() ); } else return SvxShape::getPosition(); } void SAL_CALL SvxCustomShape::setPropertyValue( const OUString& aPropertyName, const uno::Any& aValue ) { ::SolarMutexGuard aGuard; SdrObject* pObject = mpObj.get(); // tdf#98163 Use a custom slot to have filter code flush the UNO // API implementations of SdrObjCustomShape. Used e.g. by // ~SdXMLCustomShapeContext, see there for more information const OUString sFlushCustomShapeUnoApiObjects("FlushCustomShapeUnoApiObjects"); if(sFlushCustomShapeUnoApiObjects == aPropertyName) { SdrObjCustomShape* pTarget = dynamic_cast< SdrObjCustomShape* >(pObject); if(pTarget) { // Luckily, the object causing problems in tdf#93994 is not the // UNO API object, but the XCustomShapeEngine involved. This // object is on-demand replaceable and can be reset here. This // will free the involved EditEngine and VirtualDevice. pTarget->mxCustomShapeEngine.set(nullptr); } } bool bCustomShapeGeometry = pObject && aPropertyName == "CustomShapeGeometry"; bool bMirroredX = false; bool bMirroredY = false; if ( bCustomShapeGeometry ) { bMirroredX = static_cast(pObject)->IsMirroredX(); bMirroredY = static_cast(pObject)->IsMirroredY(); } SvxShape::setPropertyValue( aPropertyName, aValue ); if ( bCustomShapeGeometry ) { static_cast(pObject)->MergeDefaultAttributes(); tools::Rectangle aRect( pObject->GetSnapRect() ); // #i38892# bool bNeedsMirrorX = static_cast(pObject)->IsMirroredX() != bMirroredX; bool bNeedsMirrorY = static_cast(pObject)->IsMirroredY() != bMirroredY; std::unique_ptr< SdrGluePointList > pListCopy; if( bNeedsMirrorX || bNeedsMirrorY ) { const SdrGluePointList* pList = pObject->GetGluePointList(); if( pList ) pListCopy.reset( new SdrGluePointList(*pList) ); } if ( bNeedsMirrorX ) { Point aTop( ( aRect.Left() + aRect.Right() ) >> 1, aRect.Top() ); Point aBottom( aTop.X(), aTop.Y() + 1000 ); pObject->NbcMirror( aTop, aBottom ); // NbcMirroring is flipping the current mirror state, // so we have to set the correct state again static_cast(pObject)->SetMirroredX( !bMirroredX ); } if ( bNeedsMirrorY ) { Point aLeft( aRect.Left(), ( aRect.Top() + aRect.Bottom() ) >> 1 ); Point aRight( aLeft.X() + 1000, aLeft.Y() ); pObject->NbcMirror( aLeft, aRight ); // NbcMirroring is flipping the current mirror state, // so we have to set the correct state again static_cast(pObject)->SetMirroredY( !bMirroredY ); } if( pListCopy ) { SdrGluePointList* pNewList = const_cast< SdrGluePointList* >( pObject->GetGluePointList() ); if(pNewList) *pNewList = *pListCopy; } } } bool SvxCustomShape::getPropertyValueImpl( const OUString& rName, const SfxItemPropertySimpleEntry* pProperty, css::uno::Any& rValue ) { switch( pProperty->nWID ) { case SDRATTR_ROTATEANGLE: { double fAngle = static_cast(mpObj.get())->GetObjectRotation(); fAngle *= 100; rValue <<= (sal_Int32)fAngle; return true; } default: return SvxShape::getPropertyValueImpl( rName, pProperty, rValue ); } } void SvxCustomShape::createCustomShapeDefaults( const OUString& rValueType ) { static_cast(mpObj.get())->MergeDefaultAttributes( &rValueType ); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */