/* -*- 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 "shapeimpl.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include 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; using namespace ::com::sun::star::drawing; UNO3_GETIMPLEMENTATION_IMPL( SvxDrawPage ); SvxDrawPage::SvxDrawPage(SdrPage* pInPage) // TTTT should be reference : mpPage(pInPage) ,mpModel(&pInPage->getSdrModelFromSdrPage()) // register at broadcaster ,mpView(new SdrView(pInPage->getSdrModelFromSdrPage())) // create (hidden) view { mpView->SetDesignMode(); } SvxDrawPage::~SvxDrawPage() noexcept { if( !m_bDisposed ) { assert(!"SvxDrawPage must be disposed!"); acquire(); dispose(); } } // XComponent void SvxDrawPage::disposing() noexcept { if( mpModel ) { mpModel = nullptr; } mpView.reset(); mpPage = nullptr; } // XComponent void SvxDrawPage::dispose() { SolarMutexGuard aSolarGuard; // An frequently programming error is to release the last // reference to this object in the disposing message. // Make it robust, hold a self Reference. uno::Reference< lang::XComponent > xSelf( this ); // Guard dispose against multiple threading // Remark: It is an error to call dispose more than once { std::unique_lock aGuard( m_aMutex ); if( m_bDisposed ) return; m_bDisposed = true; } // Create an event with this as sender css::document::EventObject aEvt; aEvt.Source.set(uno::Reference::query( static_cast(this) )); // inform all listeners to release this object // The listener container are automatically cleared { std::unique_lock aGuard( m_aMutex ); maEventListeners.disposeAndClear( aGuard, aEvt ); } // notify subclasses to do their dispose disposing(); } void SAL_CALL SvxDrawPage::addEventListener( const css::uno::Reference< css::lang::XEventListener >& aListener ) { std::unique_lock aGuard( m_aMutex ); if( mpModel == nullptr ) throw lang::DisposedException(); maEventListeners.addInterface( aGuard, aListener ); } void SAL_CALL SvxDrawPage::removeEventListener( const css::uno::Reference< css::lang::XEventListener >& aListener ) { std::unique_lock aGuard( m_aMutex ); if( mpModel == nullptr ) throw lang::DisposedException(); maEventListeners.removeInterface( aGuard, aListener ); } void SAL_CALL SvxDrawPage::add( const uno::Reference< drawing::XShape >& xShape ) { SolarMutexGuard aGuard; if ( ( mpModel == nullptr ) || ( mpPage == nullptr ) ) throw lang::DisposedException(); SvxShape* pShape = comphelper::getFromUnoTunnel( xShape ); if( nullptr == pShape ) { assert(false && "adding a non-SvxShape to a page?"); return; } rtl::Reference pObj = pShape->GetSdrObject(); bool bNeededToClone(false); if(pObj && &pObj->getSdrModelFromSdrObject() != &mpPage->getSdrModelFromSdrPage()) { // TTTT UNO API tries to add an existing SvxShape to this SvxDrawPage, // but these use different SdrModels. It was possible before to completely // 'change' a SdrObject to another SdrModel (including dangerous MigrateItemPool // stuff), but is no longer. We need to Clone the SdrObject to the target model // and ::Create a new SvxShape (set SdrObject there, take over values, ...) rtl::Reference pClonedSdrShape(pObj->CloneSdrObject(mpPage->getSdrModelFromSdrPage())); pObj->setUnoShape(nullptr); pClonedSdrShape->setUnoShape(pShape); // pShape->InvalidateSdrObject(); // pShape->Create(pClonedSdrShape, this); pObj = pClonedSdrShape; bNeededToClone = true; } if(!pObj) { pObj = CreateSdrObject( xShape ); ENSURE_OR_RETURN_VOID( pObj != nullptr, "SvxDrawPage::add: no SdrObject was created!" ); } else if ( !pObj->IsInserted() ) { mpPage->InsertObject( pObj.get() ); if(bNeededToClone) { // TTTT Unfortunately in SdrObject::SetPage (see there) the // xShape/UnoShape at the newly cloned SDrObject is *removed* again, // so re-set it here, the caller *may need it* (e.g. Writer) uno::Reference< drawing::XShape > xShapeCheck(pObj->getWeakUnoShape()); if( !xShapeCheck.is() ) { pObj->setUnoShape(pShape); } } } pShape->Create( pObj.get(), this ); OSL_ENSURE( pShape->GetSdrObject() == pObj.get(), "SvxDrawPage::add: shape does not know about its newly created SdrObject!" ); if ( !pObj->IsInserted() ) { mpPage->InsertObject( pObj.get() ); } mpModel->SetChanged(); } void SAL_CALL SvxDrawPage::addTop( const uno::Reference< drawing::XShape >& xShape ) { add(xShape); } void SAL_CALL SvxDrawPage::addBottom( const uno::Reference< drawing::XShape >& xShape ) { SolarMutexGuard aGuard; if ( ( mpModel == nullptr ) || ( mpPage == nullptr ) ) throw lang::DisposedException(); SvxShape* pShape = comphelper::getFromUnoTunnel( xShape ); if( nullptr == pShape ) { assert(false && "adding a non-SvxShape to a page?"); return; } rtl::Reference pObj = pShape->GetSdrObject(); if(!pObj) { pObj = CreateSdrObject( xShape, true ); ENSURE_OR_RETURN_VOID( pObj != nullptr, "SvxDrawPage::add: no SdrObject was created!" ); } else if ( !pObj->IsInserted() ) { mpPage->InsertObject( pObj.get(), 0 ); } pShape->Create( pObj.get(), this ); OSL_ENSURE( pShape->GetSdrObject() == pObj.get(), "SvxDrawPage::add: shape does not know about its newly created SdrObject!" ); if ( !pObj->IsInserted() ) { mpPage->InsertObject( pObj.get(), 0 ); } mpModel->SetChanged(); } void SAL_CALL SvxDrawPage::remove( const Reference< drawing::XShape >& xShape ) { SolarMutexGuard aGuard; if( (mpModel == nullptr) || (mpPage == nullptr) ) throw lang::DisposedException(); SdrObject* pObj = SdrObject::getSdrObjectFromXShape( xShape ); if (!pObj) return; // remove SdrObject from page const size_t nCount = mpPage->GetObjCount(); for( size_t nNum = 0; nNum < nCount; ++nNum ) { if(mpPage->GetObj(nNum) == pObj) { const bool bUndoEnabled = mpModel->IsUndoEnabled(); if (bUndoEnabled) { mpModel->BegUndo(SvxResId(STR_EditDelete), pObj->TakeObjNameSingul(), SdrRepeatFunc::Delete); mpModel->AddUndo(mpModel->GetSdrUndoFactory().CreateUndoDeleteObject(*pObj)); } OSL_VERIFY( mpPage->RemoveObject( nNum ) == pObj ); if (bUndoEnabled) mpModel->EndUndo(); break; } } mpModel->SetChanged(); } void SvxDrawPage::sort( const css::uno::Sequence< sal_Int32 >& sortOrder ) { SolarMutexGuard aGuard; if ((mpModel == nullptr) || (mpPage == nullptr)) throw lang::DisposedException(); auto newOrder = comphelper::sequenceToContainer>(sortOrder); mpPage->sort(newOrder); } // css::container::XIndexAccess sal_Int32 SAL_CALL SvxDrawPage::getCount() { SolarMutexGuard aGuard; if( (mpModel == nullptr) || (mpPage == nullptr) ) throw lang::DisposedException(); return static_cast( mpPage->GetObjCount() ); } uno::Any SAL_CALL SvxDrawPage::getByIndex( sal_Int32 Index ) { SolarMutexGuard aGuard; if( (mpModel == nullptr) || (mpPage == nullptr) ) throw lang::DisposedException(u"Model or Page was already disposed!"_ustr); if ( Index < 0 || o3tl::make_unsigned(Index) >= mpPage->GetObjCount() ) throw lang::IndexOutOfBoundsException("Index (" + OUString::number(Index) + ") needs to be a positive integer smaller than the shape count (" + OUString::number(mpPage->GetObjCount()) + ")!"); SdrObject* pObj = mpPage->GetObj( Index ); if( pObj == nullptr ) throw uno::RuntimeException("Runtime exception thrown while getting a ref to the SdrObject at index: " + OUString::number(Index)); return Any(Reference< drawing::XShape >( pObj->getUnoShape(), uno::UNO_QUERY )); } // css::container::XElementAccess uno::Type SAL_CALL SvxDrawPage::getElementType() { return cppu::UnoType::get(); } sal_Bool SAL_CALL SvxDrawPage::hasElements() { SolarMutexGuard aGuard; if( (mpModel == nullptr) || (mpPage == nullptr) ) throw lang::DisposedException(); return mpPage && mpPage->GetObjCount()>0; } namespace { void lcl_markSdrObjectOfShape( const Reference< drawing::XShape >& _rxShape, SdrView& _rView, SdrPageView& _rPageView ) { SdrObject* pObj = SdrObject::getSdrObjectFromXShape( _rxShape ); if ( !pObj ) return; _rView.MarkObj( pObj, &_rPageView ); } } // ATTENTION: SelectObjectsInView selects the css::drawing::Shapes // only in the given SdrPageView. It hasn't to be the visible SdrPageView. void SvxDrawPage::SelectObjectsInView( const Reference< drawing::XShapes > & aShapes, SdrPageView* pPageView ) noexcept { SAL_WARN_IF(!pPageView, "svx", "SdrPageView is NULL!"); SAL_WARN_IF(!mpView, "svx", "SdrView is NULL!"); if(pPageView==nullptr || mpView==nullptr) return; mpView->UnmarkAllObj( pPageView ); tools::Long nCount = aShapes->getCount(); for( tools::Long i = 0; i < nCount; i++ ) { uno::Any aAny( aShapes->getByIndex(i) ); Reference< drawing::XShape > xShape; if( aAny >>= xShape ) lcl_markSdrObjectOfShape( xShape, *mpView, *pPageView ); } } // ATTENTION: SelectObjectInView selects the shape only in the given SdrPageView. // It hasn't to be the visible SdrPageView. void SvxDrawPage::SelectObjectInView( const Reference< drawing::XShape > & xShape, SdrPageView* pPageView ) noexcept { SAL_WARN_IF(!pPageView, "svx", "SdrPageView is NULL!"); SAL_WARN_IF(!mpView, "svx", "SdrView is NULL!"); if(pPageView!=nullptr && mpView != nullptr) { mpView->UnmarkAllObj( pPageView ); lcl_markSdrObjectOfShape( xShape, *mpView, *pPageView ); } } Reference< drawing::XShapeGroup > SAL_CALL SvxDrawPage::group( const Reference< drawing::XShapes >& xShapes ) { SolarMutexGuard aGuard; if( (mpModel == nullptr) || (mpPage == nullptr) ) throw lang::DisposedException(); SAL_WARN_IF(!mpPage , "svx", "SdrPage is NULL!"); SAL_WARN_IF(!mpView, "svx", "SdrView is NULL!"); Reference< css::drawing::XShapeGroup > xShapeGroup; if(mpPage==nullptr||mpView==nullptr||!xShapes.is()) return xShapeGroup; SdrPageView* pPageView = mpView->ShowSdrPage( mpPage ); SelectObjectsInView( xShapes, pPageView ); mpView->GroupMarked(); mpView->AdjustMarkHdl(); const SdrMarkList& rMarkList = mpView->GetMarkedObjectList(); if( rMarkList.GetMarkCount() == 1 ) { SdrObject* pObj = rMarkList.GetMark(0)->GetMarkedSdrObj(); if( pObj ) xShapeGroup.set( pObj->getUnoShape(), UNO_QUERY ); } mpView->HideSdrPage(); if( mpModel ) mpModel->SetChanged(); return xShapeGroup; } void SAL_CALL SvxDrawPage::ungroup( const Reference< drawing::XShapeGroup >& aGroup ) { SolarMutexGuard aGuard; if( (mpModel == nullptr) || (mpPage == nullptr) ) throw lang::DisposedException(); SAL_WARN_IF(!mpPage, "svx", "SdrPage is NULL!"); SAL_WARN_IF(!mpView, "svx", "SdrView is NULL!"); if(mpPage==nullptr||mpView==nullptr||!aGroup.is()) return; SdrPageView* pPageView = mpView->ShowSdrPage( mpPage ); SelectObjectInView( aGroup, pPageView ); mpView->UnGroupMarked(); mpView->HideSdrPage(); if( mpModel ) mpModel->SetChanged(); } rtl::Reference SvxDrawPage::CreateSdrObject_(const Reference< drawing::XShape > & xShape) { OUString aShapeType( xShape->getShapeType() ); if ( aShapeType == "com.sun.star.drawing.ShapeControl" // compatibility || aShapeType == "com.sun.star.drawing.ControlShape" ) { return new FmFormObj(GetSdrPage()->getSdrModelFromSdrPage()); } SdrObjKind nType = SdrObjKind::NONE; SdrInventor nInventor; GetTypeAndInventor( nType, nInventor, xShape->getShapeType() ); if (nType == SdrObjKind::NONE) return nullptr; awt::Size aSize = xShape->getSize(); aSize.Width += 1; aSize.Height += 1; awt::Point aPos = xShape->getPosition(); tools::Rectangle aRect( Point( aPos.X, aPos.Y ), Size( aSize.Width, aSize.Height ) ); rtl::Reference pNewObj = SdrObjFactory::MakeNewObject( *mpModel, nInventor, nType, &aRect); if (!pNewObj) return nullptr; if( nType == SdrObjKind::E3D_Scene ) { auto pScene = static_cast(pNewObj.get()); // initialise scene double fW = static_cast(aSize.Width); double fH = static_cast(aSize.Height); Camera3D aCam(pScene->GetCamera()); aCam.SetAutoAdjustProjection(false); aCam.SetViewWindow(- fW / 2, - fH / 2, fW, fH); basegfx::B3DPoint aLookAt; basegfx::B3DPoint aCamPos(0.0, 0.0, 10000.0); aCam.SetPosAndLookAt(aCamPos, aLookAt); aCam.SetFocalLength(100.0); pScene->SetCamera(aCam); pScene->SetBoundAndSnapRectsDirty(); } else if(nType == SdrObjKind::E3D_Extrusion) { auto pObj = static_cast(pNewObj.get()); basegfx::B2DPolygon aNewPolygon; aNewPolygon.append(basegfx::B2DPoint(0.0, 0.0)); aNewPolygon.append(basegfx::B2DPoint(0.0, 1.0)); aNewPolygon.append(basegfx::B2DPoint(1.0, 0.0)); aNewPolygon.setClosed(true); pObj->SetExtrudePolygon(basegfx::B2DPolyPolygon(aNewPolygon)); // #107245# pObj->SetExtrudeCharacterMode(sal_True); pObj->SetMergedItem(Svx3DCharacterModeItem(true)); } else if(nType == SdrObjKind::E3D_Lathe) { auto pLatheObj = static_cast(pNewObj.get()); basegfx::B2DPolygon aNewPolygon; aNewPolygon.append(basegfx::B2DPoint(0.0, 0.0)); aNewPolygon.append(basegfx::B2DPoint(0.0, 1.0)); aNewPolygon.append(basegfx::B2DPoint(1.0, 0.0)); aNewPolygon.setClosed(true); pLatheObj->SetPolyPoly2D(basegfx::B2DPolyPolygon(aNewPolygon)); // #107245# pObj->SetLatheCharacterMode(sal_True); pLatheObj->SetMergedItem(Svx3DCharacterModeItem(true)); } return pNewObj; } void SvxDrawPage::GetTypeAndInventor( SdrObjKind& rType, SdrInventor& rInventor, const OUString& aName ) noexcept { std::optional nTempType = UHashMap::getId( aName ); if( !nTempType ) { if( aName == "com.sun.star.drawing.TableShape" || aName == "com.sun.star.presentation.TableShape" ) { rInventor = SdrInventor::Default; rType = SdrObjKind::Table; } #if HAVE_FEATURE_AVMEDIA else if ( aName == "com.sun.star.presentation.MediaShape" ) { rInventor = SdrInventor::Default; rType = SdrObjKind::Media; } #endif } else if( IsInventorE3D(*nTempType) ) { rInventor = SdrInventor::E3d; rType = *nTempType; } else { rInventor = SdrInventor::Default; rType = *nTempType; switch( rType ) { case SdrObjKind::OLEPluginFrame: case SdrObjKind::OLE2Plugin: case SdrObjKind::OLE2Applet: rType = SdrObjKind::OLE2; break; default: break; } } } rtl::Reference SvxDrawPage::CreateShapeByTypeAndInventor( SdrObjKind nType, SdrInventor nInventor, SdrObject *pObj, SvxDrawPage *mpPage, OUString const & referer ) { rtl::Reference pRet; switch( nInventor ) { case SdrInventor::FmForm: { return new SvxShapeControl( pObj ); } case SdrInventor::E3d: { switch( nType ) { case SdrObjKind::E3D_Scene : pRet = new Svx3DSceneObject( pObj, mpPage ); break; case SdrObjKind::E3D_Cube : pRet = new Svx3DCubeObject( pObj ); break; case SdrObjKind::E3D_Sphere : pRet = new Svx3DSphereObject( pObj ); break; case SdrObjKind::E3D_Lathe : pRet = new Svx3DLatheObject( pObj ); break; case SdrObjKind::E3D_Extrusion : pRet = new Svx3DExtrudeObject( pObj ); break; case SdrObjKind::E3D_Polygon : pRet = new Svx3DPolygonObject( pObj ); break; default: // unknown 3D-object on page assert(false && "the IsInventor3D function must be wrong"); pRet = new SvxShape( pObj ); break; } break; } case SdrInventor::Default: { switch( nType ) { case SdrObjKind::Group: pRet = new SvxShapeGroup( pObj, mpPage ); break; case SdrObjKind::Line: pRet = new SvxShapePolyPolygon( pObj ); break; case SdrObjKind::Rectangle: pRet = new SvxShapeRect( pObj ); break; case SdrObjKind::CircleOrEllipse: case SdrObjKind::CircleSection: case SdrObjKind::CircleArc: case SdrObjKind::CircleCut: pRet = new SvxShapeCircle( pObj ); break; case SdrObjKind::Polygon: pRet = new SvxShapePolyPolygon( pObj ); break; case SdrObjKind::PolyLine: pRet = new SvxShapePolyPolygon( pObj ); break; case SdrObjKind::PathLine: pRet = new SvxShapePolyPolygon( pObj ); break; case SdrObjKind::PathFill: pRet = new SvxShapePolyPolygon( pObj ); break; case SdrObjKind::FreehandLine: pRet = new SvxShapePolyPolygon( pObj ); break; case SdrObjKind::FreehandFill: pRet = new SvxShapePolyPolygon( pObj ); break; case SdrObjKind::Caption: pRet = new SvxShapeCaption( pObj ); break; case SdrObjKind::TitleText: case SdrObjKind::OutlineText: case SdrObjKind::Text: pRet = new SvxShapeText( pObj ); break; case SdrObjKind::Graphic: pRet = new SvxGraphicObject( pObj ); break; case SdrObjKind::OLEPluginFrame: pRet = new SvxFrameShape( pObj, referer ); break; case SdrObjKind::OLE2Applet: pRet = new SvxAppletShape( pObj, referer ); break; case SdrObjKind::OLE2Plugin: pRet = new SvxPluginShape( pObj, referer ); break; case SdrObjKind::OLE2: { if( pObj && !pObj->IsEmptyPresObj() && mpPage ) { SdrPage* pSdrPage = mpPage->GetSdrPage(); if( pSdrPage ) { SdrModel& rSdrModel(pSdrPage->getSdrModelFromSdrPage()); ::comphelper::IEmbeddedHelper *pPersist = rSdrModel.GetPersist(); if( pPersist ) { uno::Reference < embed::XEmbeddedObject > xObject = pPersist->getEmbeddedObjectContainer(). GetEmbeddedObject( static_cast< SdrOle2Obj* >( pObj )->GetPersistName() ); // TODO CL->KA: Why is this not working anymore? if( xObject.is() ) { SvGlobalName aClassId( xObject->getClassID() ); const SvGlobalName aAppletClassId( SO3_APPLET_CLASSID ); const SvGlobalName aPluginClassId( SO3_PLUGIN_CLASSID ); const SvGlobalName aIFrameClassId( SO3_IFRAME_CLASSID ); if( aPluginClassId == aClassId ) { pRet = new SvxPluginShape( pObj, referer ); nType = SdrObjKind::OLE2Plugin; } else if( aAppletClassId == aClassId ) { pRet = new SvxAppletShape( pObj, referer ); nType = SdrObjKind::OLE2Applet; } else if( aIFrameClassId == aClassId ) { pRet = new SvxFrameShape( pObj, referer ); nType = SdrObjKind::OLEPluginFrame; } } } } } if( pRet == nullptr ) { SvxUnoPropertyMapProvider& rSvxMapProvider = getSvxMapProvider(); pRet = new SvxOle2Shape( pObj, referer, rSvxMapProvider.GetMap(SVXMAP_OLE2), rSvxMapProvider.GetPropertySet(SVXMAP_OLE2, SdrObject::GetGlobalDrawObjectItemPool()) ); } } break; case SdrObjKind::Edge: pRet = new SvxShapeConnector( pObj ); break; case SdrObjKind::PathPoly: pRet = new SvxShapePolyPolygon( pObj ); break; case SdrObjKind::PathPolyLine: pRet = new SvxShapePolyPolygon( pObj ); break; case SdrObjKind::Page: { SvxUnoPropertyMapProvider& rSvxMapProvider = getSvxMapProvider(); pRet = new SvxShape( pObj, rSvxMapProvider.GetMap(SVXMAP_PAGE), rSvxMapProvider.GetPropertySet(SVXMAP_PAGE, SdrObject::GetGlobalDrawObjectItemPool()) ); } break; case SdrObjKind::Measure: pRet = new SvxShapeDimensioning( pObj ); break; case SdrObjKind::UNO: pRet = new SvxShapeControl( pObj ); break; case SdrObjKind::CustomShape: pRet = new SvxCustomShape( pObj ); break; case SdrObjKind::Media: pRet = new SvxMediaShape( pObj, referer ); break; case SdrObjKind::Table: pRet = new SvxTableShape( pObj ); break; case SdrObjKind::Annotation: pRet = new SvxShapeRect( pObj ); break; default: // unknown 2D-object on page assert(false && "Not implemented Starone-Shape created"); pRet = new SvxShapeText( pObj ); break; } break; } default: // unknown inventor { assert(false && "Unknown Inventor in SvxDrawPage::CreateShape()"); break; } } if(pRet) { SdrObjKind nObjId = nType; switch(nObjId) { case SdrObjKind::CircleCut: // segment of circle case SdrObjKind::CircleArc: // arc of circle case SdrObjKind::CircleSection: // sector nObjId = SdrObjKind::CircleOrEllipse; break; case SdrObjKind::TitleText: case SdrObjKind::OutlineText: nObjId = SdrObjKind::Text; break; default: ; } pRet->setShapeKind(nObjId); } return pRet; } Reference< drawing::XShape > SvxDrawPage::CreateShape( SdrObject *pObj ) const { Reference< drawing::XShape > xShape( CreateShapeByTypeAndInventor(pObj->GetObjIdentifier(), pObj->GetObjInventor(), pObj, const_cast(this))); return xShape; } rtl::Reference SvxDrawPage::CreateSdrObject( const Reference< drawing::XShape > & xShape, bool bBeginning ) noexcept { rtl::Reference pObj = CreateSdrObject_( xShape ); if( pObj) { if ( !pObj->IsInserted() && !pObj->IsDoNotInsertIntoPageAutomatically() ) { if(bBeginning) mpPage->InsertObject( pObj.get(), 0 ); else mpPage->InsertObject( pObj.get() ); } } return pObj; } // css::lang::XServiceInfo OUString SAL_CALL SvxDrawPage::getImplementationName() { return u"SvxDrawPage"_ustr; } sal_Bool SAL_CALL SvxDrawPage::supportsService( const OUString& ServiceName ) { return cppu::supportsService( this, ServiceName ); } uno::Sequence< OUString > SAL_CALL SvxDrawPage::getSupportedServiceNames() { uno::Sequence aSeq { u"com.sun.star.drawing.ShapeCollection"_ustr }; return aSeq; } rtl::Reference CreateSvxShapeByTypeAndInventor(SdrObjKind nType, SdrInventor nInventor, OUString const & referer) { return SvxDrawPage::CreateShapeByTypeAndInventor( nType, nInventor, nullptr, nullptr, referer ); } /** returns a StarOffice API wrapper for the given SdrPage */ uno::Reference< drawing::XDrawPage > GetXDrawPageForSdrPage( SdrPage* pPage ) noexcept { if(pPage) { uno::Reference< drawing::XDrawPage > xDrawPage( pPage->getUnoPage(), uno::UNO_QUERY ); return xDrawPage; } return uno::Reference< drawing::XDrawPage >(); } /** returns the SdrObject from the given StarOffice API wrapper */ SdrPage* GetSdrPageFromXDrawPage( const uno::Reference< drawing::XDrawPage >& xDrawPage ) noexcept { if(xDrawPage.is()) { SvxDrawPage* pDrawPage = comphelper::getFromUnoTunnel( xDrawPage ); if(pDrawPage) { return pDrawPage->GetSdrPage(); } assert(false && "non-SvxDrawPage?"); } return nullptr; } // helper that returns true if the given XShape is member of the given // XDrawPage or it's MasterPage (aka associated) bool IsXShapeAssociatedWithXDrawPage( const css::uno::Reference& rxShape, const css::uno::Reference< css::drawing::XDrawPage >& rxDrawPage) noexcept { if (!rxShape) return false; if (!rxDrawPage) return false; const SdrObject* pSdrObject(SdrObject::getSdrObjectFromXShape(rxShape)); if (nullptr == pSdrObject) return false; SdrPage* pSdrPage(GetSdrPageFromXDrawPage(rxDrawPage)); if (nullptr == pSdrPage) return false; const SdrPage* pPageFromObj(pSdrObject->getSdrPageFromSdrObject()); if (nullptr == pPageFromObj) return false; if (pSdrPage == pPageFromObj) // given XShape is member of given XDrawPage return true; if (pSdrPage->TRG_HasMasterPage()) if (&pSdrPage->TRG_GetMasterPage() == pPageFromObj) // given XShape is member of MasterPage of given XDrawPage return true; return false; } // XFormsSupplier css::uno::Reference< css::container::XNameContainer > SAL_CALL SvxDrawPage::getForms() { SolarMutexGuard g; css::uno::Reference< css::container::XNameContainer > xForms; FmFormPage *pFmPage = dynamic_cast( GetSdrPage() ); if( pFmPage ) xForms.set( pFmPage->GetForms(), css::uno::UNO_QUERY_THROW ); return xForms; } // XFormsSupplier2 sal_Bool SAL_CALL SvxDrawPage::hasForms() { SolarMutexGuard g; bool bHas = false; FmFormPage* pFormPage = dynamic_cast( GetSdrPage() ); if ( pFormPage ) bHas = pFormPage->GetForms( false ).is(); return bHas; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */