/************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2008 by Sun Microsystems, Inc. * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: viewobjectcontactofunocontrol.cxx,v $ * $Revision: 1.18 $ * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_svx.hxx" #include #include #include #include #include /** === begin UNO includes === **/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include /** === end UNO includes === **/ #include #include #include #include #include "sdrpaintwindow.hxx" #include #include #include #include #include #include #include #include #include #include #include //........................................................................ namespace sdr { namespace contact { //........................................................................ /** === begin UNO using === **/ using namespace ::com::sun::star::awt::InvalidateStyle; using ::com::sun::star::uno::Reference; using ::com::sun::star::uno::XInterface; using ::com::sun::star::uno::UNO_QUERY; using ::com::sun::star::uno::UNO_QUERY_THROW; using ::com::sun::star::uno::Exception; using ::com::sun::star::uno::RuntimeException; using ::com::sun::star::awt::XControl; using ::com::sun::star::lang::XMultiServiceFactory; using ::com::sun::star::awt::XControlModel; using ::com::sun::star::awt::XControlContainer; using ::com::sun::star::awt::XWindow; using ::com::sun::star::awt::XWindow2; using ::com::sun::star::awt::XWindowListener; using ::com::sun::star::awt::PosSize::POSSIZE; using ::com::sun::star::awt::XView; using ::com::sun::star::awt::XGraphics; using ::com::sun::star::awt::WindowEvent; using ::com::sun::star::beans::XPropertySet; using ::com::sun::star::beans::XPropertySetInfo; using ::com::sun::star::lang::XComponent; using ::com::sun::star::awt::XWindowPeer; using ::com::sun::star::beans::XPropertyChangeListener; using ::com::sun::star::util::XModeChangeListener; using ::com::sun::star::util::XModeChangeBroadcaster; using ::com::sun::star::util::ModeChangeEvent; using ::com::sun::star::lang::EventObject; using ::com::sun::star::beans::PropertyChangeEvent; using ::com::sun::star::lang::XComponent; using ::com::sun::star::container::XContainerListener; using ::com::sun::star::container::XContainer; using ::com::sun::star::container::ContainerEvent; using ::com::sun::star::uno::Any; /** === end UNO using === **/ //==================================================================== //= ControlHolder //==================================================================== class ControlHolder { private: Reference< XControl > m_xControl; Reference< XWindow2 > m_xControlWindow; Reference< XView > m_xControlView; public: ControlHolder() :m_xControl() ,m_xControlWindow() ,m_xControlView() { } explicit ControlHolder( const Reference< XControl >& _rxControl ) :m_xControl() ,m_xControlWindow() ,m_xControlView() { *this = _rxControl; } ControlHolder& operator=( const Reference< XControl >& _rxControl ) { clear(); m_xControl = _rxControl; if ( m_xControl.is() ) { m_xControlWindow.set( m_xControl, UNO_QUERY ); m_xControlView.set( m_xControl, UNO_QUERY ); if ( !m_xControlWindow.is() || !m_xControlView.is() ) { OSL_ENSURE( false, "ControlHolder::operator=: invalid XControl, missing required interfaces!" ); clear(); } } return *this; } public: inline bool is() const { return m_xControl.is() && m_xControlWindow.is() && m_xControlView.is(); } inline void clear() { m_xControl.clear(); m_xControlWindow.clear(); m_xControlView.clear(); } // delegators for the methods of the UNO interfaces // Note all those will crash if called for a NULL object. inline bool isDesignMode() const { return m_xControl->isDesignMode(); } inline void setDesignMode( const bool _bDesign ) const { m_xControl->setDesignMode( _bDesign ); } inline bool isVisible() const { return m_xControlWindow->isVisible(); } inline void setVisible( const bool _bVisible ) const { m_xControlWindow->setVisible( _bVisible ); } inline Reference< XControlModel > getModel() const { return m_xControl->getModel(); } inline void setModel( const Reference< XControlModel >& _m ) const { m_xControl->setModel( _m ); } inline bool isTransparent() const { return m_xControl->isTransparent(); } inline Reference< XWindowPeer > getPeer() const { return m_xControl->getPeer(); } inline void addWindowListener( const Reference< XWindowListener >& _l ) const { m_xControlWindow->addWindowListener( _l ); } inline void removeWindowListener( const Reference< XWindowListener >& _l ) const { m_xControlWindow->removeWindowListener( _l ); } void setPosSize( const Rectangle& _rPosSize ) const; void setZoom( const MapMode& _rMapMode ) const; inline void setGraphics( const Reference< XGraphics >& _g ) const { m_xControlView->setGraphics( _g ); } inline Reference< XGraphics > getGraphics() const { return m_xControlView->getGraphics(); } inline void draw( const Point& _rTopLeft ) const { m_xControlView->draw( _rTopLeft.X(), _rTopLeft.Y() ); } public: inline const Reference< XControl >& getControl() const { return m_xControl; } }; //-------------------------------------------------------------------- bool operator==( const ControlHolder& _rControl, const Reference< XInterface >& _rxCompare ) { return _rControl.getControl() == _rxCompare; } //-------------------------------------------------------------------- bool operator==( const Reference< XInterface >& _rxCompare, const ControlHolder& _rControl ) { return _rxCompare == _rControl.getControl(); } //-------------------------------------------------------------------- bool operator==( const ControlHolder& _rControl, const Any& _rxCompare ) { return _rControl == Reference< XInterface >( _rxCompare, UNO_QUERY ); } //-------------------------------------------------------------------- bool operator==( const Any& _rxCompare, const ControlHolder& _rControl ) { return Reference< XInterface >( _rxCompare, UNO_QUERY ) == _rControl; } //-------------------------------------------------------------------- void ControlHolder::setPosSize( const Rectangle& _rPosSize ) const { // no check whether we're valid, this is the responsibility of the caller m_xControlWindow->setPosSize( _rPosSize.Left(), _rPosSize.Top(), _rPosSize.GetWidth(), _rPosSize.GetHeight(), POSSIZE ); } //-------------------------------------------------------------------- void ControlHolder::setZoom( const MapMode& _rMapMode ) const { // no check whether we're valid, this is the responsibility of the caller m_xControlView->setZoom( (float)double( _rMapMode.GetScaleX() ), (float)double( _rMapMode.GetScaleY() ) ); } //==================================================================== //= UnoControlContactHelper //==================================================================== class UnoControlContactHelper { public: /** positions a control relative to a device @precond _pDevice is not */ static void positionControl_throw( const ControlHolder& _rControl, const Rectangle& _rLogicBoundingRect, const OutputDevice* _pDevice ); /** sets the zoom at a UNO control, according to a Device's MapMode @precond _pDevice is not */ static void setControlZoom( const ControlHolder& _rControl, const OutputDevice* _pDevice ); /** draws a given control onto it's current XGraphics, at given coordinates Note that the control is not drawn onto the given device, instead you must use ->XView::setGraphics yourself, before calling this method. The given ->OutputDevice is only used to calculate pixel coordinates from logic coordinates @precond _pDevice is not */ static void drawControl( const ControlHolder& _rControl, const Point& _rLogicTopLeft, const OutputDevice* _pDevice ); /** disposes the given control */ static void disposeAndClearControl_nothrow( ControlHolder& _rControl ); private: UnoControlContactHelper(); // never implemented UnoControlContactHelper( const UnoControlContactHelper& ); // never implemented UnoControlContactHelper& operator=( const UnoControlContactHelper& ); // never implemented }; //-------------------------------------------------------------------- void UnoControlContactHelper::positionControl_throw( const ControlHolder& _rControl, const Rectangle& _rLogicBoundingRect, const OutputDevice* _pDevice ) { OSL_PRECOND( _pDevice, "UnoControlContactHelper::positionControl_throw: no device -> no survival!" ); if ( _rControl.is() ) { const Rectangle aPaintRectPixel( _pDevice->LogicToPixel( _rLogicBoundingRect.TopLeft() ), _pDevice->LogicToPixel( _rLogicBoundingRect.GetSize() ) ); _rControl.setPosSize( aPaintRectPixel ); } } //-------------------------------------------------------------------- void UnoControlContactHelper::setControlZoom( const ControlHolder& _rControl, const OutputDevice* _pDevice ) { OSL_PRECOND( _pDevice, "UnoControlContactHelper::setControlZoom: no device -> no survival!" ); OSL_PRECOND( _rControl.is(), "UnoControlContactHelper::setControlZoom: illegal control!" ); if ( _rControl.is() ) _rControl.setZoom( _pDevice->GetMapMode() ); } //-------------------------------------------------------------------- void UnoControlContactHelper::drawControl( const ControlHolder& _rControl, const Point& _rLogicTopLeft, const OutputDevice* _pDevice ) { OSL_PRECOND( _rControl.is(), "UnoControlContactHelper::drawControl: invalid control!" ); if ( !_rControl.is() ) return; try { _rControl.draw( _pDevice->LogicToPixel( _rLogicTopLeft ) ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } //-------------------------------------------------------------------- void UnoControlContactHelper::disposeAndClearControl_nothrow( ControlHolder& _rControl ) { try { Reference< XComponent > xControlComp( _rControl.getControl(), UNO_QUERY ); if ( xControlComp.is() ) xControlComp->dispose(); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } _rControl.clear(); } //==================================================================== //= IPageViewAccess //==================================================================== /** interface encapsulating access to an SdrPageView, stripped down to the methods we really need */ class IPageViewAccess { public: /** determines whether the view is currently in design mode */ virtual bool isDesignMode() const = 0; /** retrieves the control container for a given output device */ virtual Reference< XControlContainer > getControlContainer( const OutputDevice& _rDevice ) const = 0; /** determines whether a given layer is visible */ virtual bool isLayerVisible( SdrLayerID _nLayerID ) const = 0; }; //==================================================================== //= SdrPageViewAccess //==================================================================== /** is a ->IPageViewAccess implementation based on a real ->SdrPageView instance */ class SdrPageViewAccess : public IPageViewAccess { const SdrPageView& m_rPageView; public: SdrPageViewAccess( const SdrPageView& _rPageView ) : m_rPageView( _rPageView ) { } virtual bool isDesignMode() const; virtual Reference< XControlContainer > getControlContainer( const OutputDevice& _rDevice ) const; virtual bool isLayerVisible( SdrLayerID _nLayerID ) const; }; //-------------------------------------------------------------------- bool SdrPageViewAccess::isDesignMode() const { return m_rPageView.GetView().IsDesignMode(); } //-------------------------------------------------------------------- Reference< XControlContainer > SdrPageViewAccess::getControlContainer( const OutputDevice& _rDevice ) const { Reference< XControlContainer > xControlContainer = m_rPageView.GetControlContainer( _rDevice ); DBG_ASSERT( xControlContainer.is() || NULL == m_rPageView.FindPageWindow( ( const_cast< OutputDevice& >( _rDevice ) ) ), "SdrPageViewAccess::getControlContainer: the output device is known, but there is no control container for it?" ); return xControlContainer; } //-------------------------------------------------------------------- bool SdrPageViewAccess::isLayerVisible( SdrLayerID _nLayerID ) const { return m_rPageView.GetVisibleLayers().IsSet( _nLayerID ); } //==================================================================== //= DummyPageViewAccess //==================================================================== /** is a ->IPageViewAccess implementation which defaults all attribute access, and thus can be operated without an actual ->SdrPageView */ class DummyPageViewAccess : public IPageViewAccess { public: virtual bool isDesignMode() const; virtual Reference< XControlContainer > getControlContainer( const OutputDevice& _rDevice ) const; virtual bool isLayerVisible( SdrLayerID _nLayerID ) const; }; //-------------------------------------------------------------------- bool DummyPageViewAccess::isDesignMode() const { return true; } //-------------------------------------------------------------------- Reference< XControlContainer > DummyPageViewAccess::getControlContainer( const OutputDevice& /*_rDevice*/ ) const { return Reference< XControlContainer >(); } //-------------------------------------------------------------------- bool DummyPageViewAccess::isLayerVisible( SdrLayerID /*_nLayerID*/ ) const { return true; } //==================================================================== //= DummyPageViewAccess //==================================================================== /** is a ->IPageViewAccess implementation which can be used to create an invisble control for an arbitrary device */ class InvisibleControlViewAccess : public IPageViewAccess { private: Reference< XControlContainer >& m_rControlContainer; public: InvisibleControlViewAccess( Reference< XControlContainer >& _inout_ControlContainer ) :m_rControlContainer( _inout_ControlContainer ) { } virtual bool isDesignMode() const; virtual Reference< XControlContainer > getControlContainer( const OutputDevice& _rDevice ) const; virtual bool isLayerVisible( SdrLayerID _nLayerID ) const; }; //-------------------------------------------------------------------- bool InvisibleControlViewAccess::isDesignMode() const { return true; } //-------------------------------------------------------------------- Reference< XControlContainer > InvisibleControlViewAccess::getControlContainer( const OutputDevice& _rDevice ) const { if ( !m_rControlContainer.is() ) { const Window* pWindow = dynamic_cast< const Window* >( &_rDevice ); OSL_ENSURE( pWindow, "InvisibleControlViewAccess::getControlContainer: expected to be called for a window only!" ); if ( pWindow ) m_rControlContainer = VCLUnoHelper::CreateControlContainer( const_cast< Window* >( pWindow ) ); } return m_rControlContainer; } //-------------------------------------------------------------------- bool InvisibleControlViewAccess::isLayerVisible( SdrLayerID /*_nLayerID*/ ) const { return false; } //==================================================================== //= ViewObjectContactOfUnoControl_Impl //==================================================================== typedef ::cppu::WeakImplHelper4 < XWindowListener , XPropertyChangeListener , XContainerListener , XModeChangeListener > ViewObjectContactOfUnoControl_Impl_Base; class SVX_DLLPRIVATE ViewObjectContactOfUnoControl_Impl : public ViewObjectContactOfUnoControl_Impl_Base { private: /// the instance whose IMPL we are ViewObjectContactOfUnoControl* m_pAntiImpl; /** thread safety (not really. ATM only our X* implementations are guarded with this, but not the object as a whole.) */ mutable ::osl::Mutex m_aMutex; /// the control we're responsible for ControlHolder m_aControl; /// the ControlContainer where we inserted our control Reference< XContainer > m_xContainer; /// the output device for which the control was created OutputDevice const* m_pOutputDeviceForWindow; /// flag indicating whether the control is currently visible bool m_bControlIsVisible; /// are we currently listening at a design mode control? bool m_bIsDesignModeListening; enum ViewControlMode { eDesign, eAlive, eUnknown }; /// is the control currently in design mode? mutable ViewControlMode m_eControlDesignMode; public: ViewObjectContactOfUnoControl_Impl( ViewObjectContactOfUnoControl* _pAntiImpl ); /** disposes the instance, which is nonfunctional afterwards */ void dispose(); /** determines whether the instance is disposed */ bool isDisposed() const { return impl_isDisposed_nofail(); } /** determines whether the instance is alive */ bool isAlive() const { return !isDisposed(); } /** returns the SdrUnoObject associated with the ViewContact @precond We're not disposed. */ bool getUnoObject( SdrUnoObj*& _out_rpObject ) const; /** ensures that we have an XControl which can be painted onto the given display */ bool ensureControl( const DisplayInfo& _rDisplayInfo ); /** ensures that we have an ->XControl Must only be called if a control is needed when no DisplayInfo is present, yet. For creating a control, an ->OutputDevice is needed, and an ->SdrPageView. Both can only be obtained from a ->DisplayInfo struct, or alternatively a ->ObjectContactOfPageView. So, if our (anti-impl's) object contact is not a ->ObjectContactOfPageView, this method fill fail. Failure of this method will be reported via an assertion in a non-product version. */ bool ensureControl(); /** returns our XControl, if it already has been created If you want to ensure that the control exists before accessing it, use ->ensureControl */ inline const ControlHolder& getExistentControl() const { return m_aControl; } /** positions our XControl according to the geometry settings in the SdrUnoObj, and sets proper zoom settings according to our device @precond ->m_pOutputDeviceForWindow and ->m_aControl are not @tolerant If the preconditions are not met, nothing is done at all */ void positionAndZoomControl() const; /** positions the control for a paint onto a given device If we do not (yet) have a control, or the control does not belong to the device for which a paint is requested, no positioning happens. */ void positionControlForPaint( const DisplayInfo& _rDisplayInfo ) const; /** determines whether or not our control is printable Effectively, this method returns the value of the "Printable" property of the control's model. If we have no control, is returned. */ bool isPrintableControl() const; /** sets the design mode on the control, or at least remembers the flag for the time the control is created */ void setControlDesignMode( bool _bDesignMode ) const; /** determines whether our control is currently visible @nofail */ bool isControlVisible() const { return impl_isControlVisible_nofail(); } /** determines whether the instance belongs to a given OutputDevice @precond The instance knows the device it belongs to, or can determine it. If this is not the case, you will notice an assertion, and the method will return false. */ bool belongsToDevice( const OutputDevice* _pDevice ) const; /// creates an XControl for the given device and SdrUnoObj static bool createControlForDevice( IPageViewAccess& _rPageView, const OutputDevice& _rDevice, const SdrUnoObj& _rUnoObject, ControlHolder& _out_rControl ); struct GuardAccess { friend class VOCGuard; private: GuardAccess() { } }; ::osl::Mutex& getMutex( GuardAccess ) const { return m_aMutex; } protected: ~ViewObjectContactOfUnoControl_Impl(); // XEventListener virtual void SAL_CALL disposing( const EventObject& Source ) throw(RuntimeException); // XWindowListener virtual void SAL_CALL windowResized( const WindowEvent& e ) throw(RuntimeException); virtual void SAL_CALL windowMoved( const WindowEvent& e ) throw(RuntimeException); virtual void SAL_CALL windowShown( const EventObject& e ) throw(RuntimeException); virtual void SAL_CALL windowHidden( const EventObject& e ) throw(RuntimeException); // XPropertyChangeListener virtual void SAL_CALL propertyChange( const PropertyChangeEvent& evt ) throw(RuntimeException); // XModeChangeListener virtual void SAL_CALL modeChanged( const ModeChangeEvent& _rSource ) throw (RuntimeException); // XContainerListener virtual void SAL_CALL elementInserted( const ::com::sun::star::container::ContainerEvent& Event ) throw (::com::sun::star::uno::RuntimeException); virtual void SAL_CALL elementRemoved( const ::com::sun::star::container::ContainerEvent& Event ) throw (::com::sun::star::uno::RuntimeException); virtual void SAL_CALL elementReplaced( const ::com::sun::star::container::ContainerEvent& Event ) throw (::com::sun::star::uno::RuntimeException); private: /** retrieves the SdrPageView which our associated SdrPageViewWindow belongs to @param out_rpPageView a reference to a pointer holding, upon return, the desired SdrPageView @return if and only if a ->SdrPageView could be obtained @precond We really belong to an SdrPageViewWindow. Perhaps (I'm not sure ATM :) there are instance for which this might not be true, but those instances should never have a need to call this method. @precond We're not disposed. @postcond The method expects success, if it returns with , this will have been asserted. @nothrow */ bool impl_getPageView_nothrow( SdrPageView*& _out_rpPageView ); /** adjusts the control visibility so it respects its layer's visibility @param _bForce set to if you want to force a ->XWindow::setVisible call, no matter if the control visibility is already correct @precond ->m_aControl is not @precond We're not disposed. @precond We really belong to an SdrPageViewWindow. There are instance for which this might not be true, but those instances should never have a need to call this method. */ void impl_adjustControlVisibilityToLayerVisibility_throw( bool _bForce ); /** adjusts the control visibility so it respects its layer's visibility The control must never be visibile if it's in design mode. In alive mode, it must be visibility if and only it's on a visible layer. @param _rxControl the control whose visibility is to be adjusted @param _rPageView provides access to the attributes of the SdrPageView which the control finally belongs to @param _rUnoObject our SdrUnoObj @param _bIsCurrentlyVisible determines whether the control is currently visible. Note that this is only a shortcut for querying _rxControl for the XWindow2 interface, and calling isVisible at this interface. This shortcut has been chosen since the caller usually already has this information. If _bForce is , _bIsCurrentlyVisible is ignored. @param _bForce set to if you want to force a ->XWindow::setVisible call, no matter if the control visibility is already correct @precond We're not disposed. */ static void impl_adjustControlVisibilityToLayerVisibility_throw( const ControlHolder& _rxControl, const SdrUnoObj& _rUnoObject, IPageViewAccess& _rPageView, bool _bIsCurrentlyVisible, bool _bForce ); /** starts or stops listening at various aspects of our control @precond ->m_aControl is not */ void impl_switchControlListening_nothrow( bool _bStart ); /** starts or stops listening at our control container @precond ->m_xContainer is not */ void impl_switchContainerListening_nothrow( bool _bStart ); /** starts or stops listening at the control for design-mode relevant facets */ void impl_switchDesignModeListening_nothrow( bool _bStart ); /** starts or stops listening for all properties at our control @param _bStart determines whether to start or to stop listening @precond ->m_aControl is not */ void impl_switchPropertyListening_nothrow( bool _bStart ); /** disposes the instance @param _bAlsoDisposeControl determines whether the XControl should be disposed, too */ void impl_dispose_nothrow( bool _bAlsoDisposeControl ); /** determines whether the instance is disposed @nofail */ bool impl_isDisposed_nofail() const { return m_pAntiImpl == NULL; } /** determines whether our control is currently visible @nofail */ bool impl_isControlVisible_nofail() const { return m_bControlIsVisible; } /** determines whether we are currently a listener at the control for desgin-mode relevant facets @nofail */ bool impl_isDesignModeListening_nofail() const { return m_bIsDesignModeListening; } /** determines whether the control currently is in design mode @precond The design mode must already be known. It is known when we first had access to an SdrPageView (which carries this flag), or somebody explicitly set it from outside. */ inline bool impl_isControlDesignMode_nothrow() const { DBG_ASSERT( m_eControlDesignMode != eUnknown, "ViewObjectContactOfUnoControl_Impl::impl_isControlDesignMode_nothrow: mode is still unknown!" ); return m_eControlDesignMode == eDesign; } /** ensures that we have a control for the given PageView/OutputDevice */ bool impl_ensureControl_nothrow( IPageViewAccess& _rPageView, const OutputDevice& _rDevice ); /** retrieves the device which a PageView belongs to, starting from its ObjectContactOfPageView Since #i72752#, the PaintWindow (and thus the OutputDevice) associated with a PageView is not constant over its lifetime. Instead, during some paint operations, the PaintWindow/OutputDevice might be temporarily patched. This method cares for this, by retrieving the very original OutputDevice. */ const OutputDevice& imp_getPageViewDevice_nothrow( const ObjectContactOfPageView& _rObjectContact ) const; private: ViewObjectContactOfUnoControl_Impl(); // never implemented ViewObjectContactOfUnoControl_Impl( const ViewObjectContactOfUnoControl_Impl& ); // never implemented ViewObjectContactOfUnoControl_Impl& operator=( const ViewObjectContactOfUnoControl_Impl& ); // never implemented }; //==================================================================== //= VOCGuard //==================================================================== /** class for guarding a ViewObjectContactOfUnoControl_Impl method */ class VOCGuard { const ViewObjectContactOfUnoControl_Impl& m_rImpl; ::osl::MutexGuard m_aMutexGuard; public: VOCGuard( const ViewObjectContactOfUnoControl_Impl& _rImpl ) :m_rImpl( _rImpl ) ,m_aMutexGuard( _rImpl.getMutex( ViewObjectContactOfUnoControl_Impl::GuardAccess() ) ) { } }; //==================================================================== //= ViewObjectContactOfUnoControl_Impl //==================================================================== DBG_NAME( ViewObjectContactOfUnoControl_Impl ) //-------------------------------------------------------------------- ViewObjectContactOfUnoControl_Impl::ViewObjectContactOfUnoControl_Impl( ViewObjectContactOfUnoControl* _pAntiImpl ) :m_pAntiImpl( _pAntiImpl ) ,m_pOutputDeviceForWindow( NULL ) ,m_bControlIsVisible( false ) ,m_bIsDesignModeListening( false ) ,m_eControlDesignMode( eUnknown ) { DBG_CTOR( ViewObjectContactOfUnoControl_Impl, NULL ); DBG_ASSERT( m_pAntiImpl, "ViewObjectContactOfUnoControl_Impl::ViewObjectContactOfUnoControl_Impl: invalid AntiImpl!" ); } //-------------------------------------------------------------------- ViewObjectContactOfUnoControl_Impl::~ViewObjectContactOfUnoControl_Impl() { if ( !impl_isDisposed_nofail() ) { acquire(); dispose(); } DBG_DTOR( ViewObjectContactOfUnoControl_Impl, NULL ); } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl_Impl::impl_dispose_nothrow( bool _bAlsoDisposeControl ) { if ( impl_isDisposed_nofail() ) return; if ( m_aControl.is() ) impl_switchControlListening_nothrow( false ); if ( m_xContainer.is() ) impl_switchContainerListening_nothrow( false ); // dispose the control if ( _bAlsoDisposeControl ) UnoControlContactHelper::disposeAndClearControl_nothrow( m_aControl ); m_aControl.clear(); m_xContainer.clear(); m_pOutputDeviceForWindow = NULL; m_bControlIsVisible = false; m_pAntiImpl = NULL; } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl_Impl::dispose() { VOCGuard aGuard( *this ); impl_dispose_nothrow( true ); } //-------------------------------------------------------------------- bool ViewObjectContactOfUnoControl_Impl::getUnoObject( SdrUnoObj*& _out_rpObject ) const { OSL_PRECOND( !impl_isDisposed_nofail(), "ViewObjectContactOfUnoControl_Impl::getUnoObject: already disposed()" ); if ( impl_isDisposed_nofail() ) _out_rpObject = NULL; else { _out_rpObject = dynamic_cast< SdrUnoObj* >( m_pAntiImpl->GetViewContact().TryToGetSdrObject() ); DBG_ASSERT( _out_rpObject || !m_pAntiImpl->GetViewContact().TryToGetSdrObject(), "ViewObjectContactOfUnoControl_Impl::getUnoObject: invalid SdrObject!" ); } return ( _out_rpObject != NULL ); } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl_Impl::positionControlForPaint( const DisplayInfo& /* #i74769# _rDisplayInfo*/ ) const { if ( !m_aControl.is() ) return; positionAndZoomControl(); } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl_Impl::positionAndZoomControl() const { OSL_PRECOND( m_pOutputDeviceForWindow && m_aControl.is(), "ViewObjectContactOfUnoControl_Impl::positionAndZoomControl: no output device or no control!" ); if ( !m_pOutputDeviceForWindow || !m_aControl.is() ) return; try { SdrUnoObj* pUnoObject( NULL ); if ( getUnoObject( pUnoObject ) ) UnoControlContactHelper::positionControl_throw( m_aControl, pUnoObject->GetLogicRect(), m_pOutputDeviceForWindow ); UnoControlContactHelper::setControlZoom( m_aControl, m_pOutputDeviceForWindow ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } //-------------------------------------------------------------------- bool ViewObjectContactOfUnoControl_Impl::ensureControl( const DisplayInfo& /*_rDisplayInfo*/ ) { OSL_PRECOND( !impl_isDisposed_nofail(), "ViewObjectContactOfUnoControl_Impl::ensureControl: already disposed()" ); if ( impl_isDisposed_nofail() ) return false; const OutputDevice* pDeviceForControl( NULL ); // if we're working for a page view, use the respective OutputDevice at the proper // PaintWindow. The DisplayInfo might only contain a temporary (virtual) device, which // is dangerous to remember // 2006-10-24 / #i70604# / frank.schoenheit@sun.com ObjectContactOfPageView* pPageViewContact = dynamic_cast< ObjectContactOfPageView* >( &m_pAntiImpl->GetObjectContact() ); if ( pPageViewContact ) pDeviceForControl = &imp_getPageViewDevice_nothrow( *pPageViewContact ); if ( !pDeviceForControl && pPageViewContact) pDeviceForControl = pPageViewContact->TryToGetOutputDevice(); DBG_ASSERT( pDeviceForControl, "ViewObjectContactOfUnoControl_Impl::ensureControl: no output device!" ); if ( !pDeviceForControl ) return false; SdrPageView* pPageView = m_pAntiImpl->GetObjectContact().TryToGetSdrPageView(); ::std::auto_ptr< IPageViewAccess > pPVAccess; pPVAccess.reset( pPageView ? (IPageViewAccess*)new SdrPageViewAccess( *pPageView ) : (IPageViewAccess*)new DummyPageViewAccess() ); return impl_ensureControl_nothrow( *pPVAccess, *pDeviceForControl ); } //-------------------------------------------------------------------- bool ViewObjectContactOfUnoControl_Impl::ensureControl() { OSL_PRECOND( !impl_isDisposed_nofail(), "ViewObjectContactOfUnoControl_Impl::ensureControl: already disposed()" ); if ( impl_isDisposed_nofail() ) return false; ObjectContactOfPageView* pPageViewContact = dynamic_cast< ObjectContactOfPageView* >( &m_pAntiImpl->GetObjectContact() ); DBG_ASSERT( pPageViewContact, "ViewObjectContactOfUnoControl_Impl::ensureControl: cannot create a control if I don't have a PageView!" ); if ( !pPageViewContact ) return false; SdrPageViewAccess aPVAccess( pPageViewContact->GetPageWindow().GetPageView() ); return impl_ensureControl_nothrow( aPVAccess, imp_getPageViewDevice_nothrow( *pPageViewContact ) ); } //-------------------------------------------------------------------- const OutputDevice& ViewObjectContactOfUnoControl_Impl::imp_getPageViewDevice_nothrow( const ObjectContactOfPageView& _rObjectContact ) const { // if the PageWindow has a patched PaintWindow, use the original PaintWindow // this ensures that our control is _not_ re-created just because somebody // (temporarily) changed the window to paint onto. // #i72429# / 2007-02-20 / frank.schoenheit@sun.com const SdrPageWindow& rPageWindow( _rObjectContact.GetPageWindow() ); if ( rPageWindow.GetOriginalPaintWindow() ) return rPageWindow.GetOriginalPaintWindow()->GetOutputDevice(); return rPageWindow.GetPaintWindow().GetOutputDevice(); } //-------------------------------------------------------------------- bool ViewObjectContactOfUnoControl_Impl::impl_ensureControl_nothrow( IPageViewAccess& _rPageView, const OutputDevice& _rDevice ) { if ( m_aControl.is() ) { if ( m_pOutputDeviceForWindow == &_rDevice ) return true; // Somebody requested a control for a new device, which means either of // - our PageView's paint window changed since we were last here // - we don't belong to a page view, and are simply painted onto different devices // The first sounds strange (doens't it?), the second means we could perhaps // optimize this in the future - there is no need to re-create the control every time, // is it? // #i74523# / 2007-02-15 / frank.schoenheit@sun.com if ( m_xContainer.is() ) impl_switchContainerListening_nothrow( false ); impl_switchControlListening_nothrow( false ); UnoControlContactHelper::disposeAndClearControl_nothrow( m_aControl ); } SdrUnoObj* pUnoObject( NULL ); if ( !getUnoObject( pUnoObject ) ) return false; ControlHolder aControl; if ( !createControlForDevice( _rPageView, _rDevice, *pUnoObject, aControl ) ) return false; m_pOutputDeviceForWindow = &_rDevice; m_aControl = aControl; m_xContainer = m_xContainer.query( _rPageView.getControlContainer( _rDevice ) ); DBG_ASSERT( ( m_xContainer.is() // either have a XControlContainer || ( ( !_rPageView.getControlContainer( _rDevice ).is() ) // or don't have any container, && ( dynamic_cast< const Window* >( &_rDevice ) == NULL ) // which is allowed for non-Window instances only ) ), "ViewObjectContactOfUnoControl_Impl::impl_ensureControl_nothrow: no XContainer at the ControlContainer!" ); try { m_eControlDesignMode = m_aControl.isDesignMode() ? eDesign : eAlive; m_bControlIsVisible = m_aControl.isVisible(); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } // start listening at all aspects of the control which are interesting to us ... impl_switchControlListening_nothrow( true ); // start listening at the control container, in case somebody tampers with our control if ( m_xContainer.is() ) impl_switchContainerListening_nothrow( true ); return m_aControl.is(); } //-------------------------------------------------------------------- bool ViewObjectContactOfUnoControl_Impl::createControlForDevice( IPageViewAccess& _rPageView, const OutputDevice& _rDevice, const SdrUnoObj& _rUnoObject, ControlHolder& _out_rControl ) { _out_rControl.clear(); Reference< XControlModel > xControlModel( _rUnoObject.GetUnoControlModel() ); DBG_ASSERT( xControlModel.is(), "ViewObjectContactOfUnoControl_Impl::createControlForDevice: no control model at the SdrUnoObject!?" ); if ( !xControlModel.is() ) return false; bool bSuccess = false; try { do { const ::rtl::OUString sControlServiceName( _rUnoObject.GetUnoControlTypeName() ); Reference< XMultiServiceFactory > xFactory( ::comphelper::getProcessServiceFactory() ); if ( xFactory.is() ) { _out_rControl = Reference< XControl >( xFactory->createInstance( sControlServiceName ), UNO_QUERY ); } DBG_ASSERT( _out_rControl.is(), "ViewObjectContactOfUnoControl_Impl::createControlForDevice: no control could be created!" ); if ( !_out_rControl.is() ) break; // knit the model and the control _out_rControl.setModel( xControlModel ); UnoControlContactHelper::positionControl_throw( _out_rControl, _rUnoObject.GetLogicRect(), &_rDevice ); // proper zoom UnoControlContactHelper::setControlZoom( _out_rControl, &_rDevice ); // #107049# set design mode before peer is created, // this is also needed for accessibility _out_rControl.setDesignMode( _rPageView.isDesignMode() ); // adjust the initial visibility according to the visibility of the layer // 2003-06-03 - #110592# - fs@openoffice.org impl_adjustControlVisibilityToLayerVisibility_throw( _out_rControl, _rUnoObject, _rPageView, false, true ); // add the control to the respective control container // #108327# do this last Reference< XControlContainer > xControlContainer( _rPageView.getControlContainer( _rDevice ) ); if ( xControlContainer.is() ) xControlContainer->addControl( sControlServiceName, _out_rControl.getControl() ); bSuccess = true; } while ( false ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } if ( !bSuccess ) { // delete the control which might have been created already UnoControlContactHelper::disposeAndClearControl_nothrow( _out_rControl ); } return _out_rControl.is(); } //-------------------------------------------------------------------- bool ViewObjectContactOfUnoControl_Impl::impl_getPageView_nothrow( SdrPageView*& _out_rpPageView ) { OSL_PRECOND( !impl_isDisposed_nofail(), "ViewObjectContactOfUnoControl_Impl::impl_getPageView_nothrow: already disposed!" ); _out_rpPageView = NULL; if ( impl_isDisposed_nofail() ) return false; ObjectContactOfPageView* pPageViewContact = dynamic_cast< ObjectContactOfPageView* >( &m_pAntiImpl->GetObjectContact() ); if ( pPageViewContact ) _out_rpPageView = &pPageViewContact->GetPageWindow().GetPageView(); DBG_ASSERT( _out_rpPageView != NULL, "ViewObjectContactOfUnoControl_Impl::impl_getPageView_nothrow: this method is expected to always have success!" ); return ( _out_rpPageView != NULL ); } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl_Impl::impl_adjustControlVisibilityToLayerVisibility_throw( bool _bForce ) { OSL_PRECOND( m_aControl.is(), "ViewObjectContactOfUnoControl_Impl::impl_adjustControlVisibilityToLayerVisibility_throw: only valid if we have a control!" ); SdrPageView* pPageView( NULL ); if ( !impl_getPageView_nothrow( pPageView ) ) return; SdrUnoObj* pUnoObject( NULL ); if ( !getUnoObject( pUnoObject ) ) return; SdrPageViewAccess aPVAccess( *pPageView ); impl_adjustControlVisibilityToLayerVisibility_throw( m_aControl, *pUnoObject, aPVAccess, impl_isControlVisible_nofail(), _bForce ); } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl_Impl::impl_adjustControlVisibilityToLayerVisibility_throw( const ControlHolder& _rControl, const SdrUnoObj& _rUnoObject, IPageViewAccess& _rPageView, bool _bIsCurrentlyVisible, bool _bForce ) { // in design mode, there is no problem with the visibility: The XControl is hidden by // default, and the Drawing Layer will simply not call our paint routine, if we're in // a hidden layer. So, only alive mode matters. if ( !_rControl.isDesignMode() ) { // the layer of our object SdrLayerID nObjectLayer = _rUnoObject.GetLayer(); // is the layer we're residing in visible in this view? bool bIsObjectLayerVisible = _rPageView.isLayerVisible( nObjectLayer ); if ( _bForce || ( bIsObjectLayerVisible != _bIsCurrentlyVisible ) ) { _rControl.setVisible( bIsObjectLayerVisible ); } } } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl_Impl::impl_switchContainerListening_nothrow( bool _bStart ) { OSL_PRECOND( m_xContainer.is(), "ViewObjectContactOfUnoControl_Impl::impl_switchContainerListening_nothrow: no control container!" ); if ( !m_xContainer.is() ) return; try { if ( _bStart ) m_xContainer->addContainerListener( this ); else m_xContainer->removeContainerListener( this ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl_Impl::impl_switchControlListening_nothrow( bool _bStart ) { OSL_PRECOND( m_aControl.is(), "ViewObjectContactOfUnoControl_Impl::impl_switchControlListening_nothrow: invalid control!" ); if ( !m_aControl.is() ) return; try { // listen for visibility changes if ( _bStart ) m_aControl.addWindowListener( this ); else m_aControl.removeWindowListener( this ); // in design mode, listen for some more aspects impl_switchDesignModeListening_nothrow( impl_isControlDesignMode_nothrow() && _bStart ); // listen for design mode changes Reference< XModeChangeBroadcaster > xDesignModeChanges( m_aControl.getControl(), UNO_QUERY_THROW ); if ( _bStart ) xDesignModeChanges->addModeChangeListener( this ); else xDesignModeChanges->removeModeChangeListener( this ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl_Impl::impl_switchDesignModeListening_nothrow( bool _bStart ) { if ( impl_isDesignModeListening_nofail() != _bStart ) { m_bIsDesignModeListening = _bStart; impl_switchPropertyListening_nothrow( _bStart ); } } //------------------------------------------------------------------------------ void ViewObjectContactOfUnoControl_Impl::impl_switchPropertyListening_nothrow( bool _bStart ) { OSL_PRECOND( m_aControl.is(), "ViewObjectContactOfUnoControl_Impl::impl_switchPropertyListening_nothrow: no control!" ); if ( !m_aControl.is() ) return; try { Reference< XPropertySet > xModelProperties( m_aControl.getModel(), UNO_QUERY_THROW ); if ( _bStart ) xModelProperties->addPropertyChangeListener( ::rtl::OUString(), this ); else xModelProperties->removePropertyChangeListener( ::rtl::OUString(), this ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } //-------------------------------------------------------------------- bool ViewObjectContactOfUnoControl_Impl::isPrintableControl() const { if ( !m_aControl.is() ) return false; bool bIsPrintable = false; try { Reference< XPropertySet > xModelProperties( m_aControl.getModel(), UNO_QUERY ); Reference< XPropertySetInfo > xPropertyInfo( xModelProperties.is() ? xModelProperties->getPropertySetInfo() : Reference< XPropertySetInfo >() ); const ::rtl::OUString sPrintablePropertyName( RTL_CONSTASCII_USTRINGPARAM( "Printable" ) ); if ( xPropertyInfo.is() && xPropertyInfo->hasPropertyByName( sPrintablePropertyName ) ) OSL_VERIFY( xModelProperties->getPropertyValue( sPrintablePropertyName ) >>= bIsPrintable ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } return bIsPrintable; } //-------------------------------------------------------------------- void SAL_CALL ViewObjectContactOfUnoControl_Impl::disposing( const EventObject& Source ) throw(RuntimeException) { ::vos::OGuard aSolarGuard( Application::GetSolarMutex() ); // some code below - in particular our disposal - might trigger actions which require the // SolarMutex. In particular, in our disposal, we remove ourself as listener from the control, // which alone needs the SolarMutex. Of course this - a removeFooListener needed the SolarMutex - // is the real bug. Toolkit really is infested with solar mutex usage ... :( // #i82169# / 2007-11-14 / frank.schoenheit@sun.com VOCGuard aGuard( *this ); if ( !m_aControl.is() ) return; if ( ( m_aControl == Source.Source ) || ( m_aControl.getModel() == Source.Source ) ) { // the model or the control is dying ... hmm, not much sense in that we ourself continue // living impl_dispose_nothrow( false ); return; } DBG_ASSERT( Source.Source == m_xContainer, "ViewObjectContactOfUnoControl_Impl::disposing: Who's this?" ); } //-------------------------------------------------------------------- void SAL_CALL ViewObjectContactOfUnoControl_Impl::windowResized( const WindowEvent& /*e*/ ) throw(RuntimeException) { // not interested in } //-------------------------------------------------------------------- void SAL_CALL ViewObjectContactOfUnoControl_Impl::windowMoved( const WindowEvent& /*e*/ ) throw(RuntimeException) { // not interested in } //-------------------------------------------------------------------- void SAL_CALL ViewObjectContactOfUnoControl_Impl::windowShown( const EventObject& /*e*/ ) throw(RuntimeException) { VOCGuard aGuard( *this ); m_bControlIsVisible = true; } //-------------------------------------------------------------------- void SAL_CALL ViewObjectContactOfUnoControl_Impl::windowHidden( const EventObject& /*e*/ ) throw(RuntimeException) { VOCGuard aGuard( *this ); m_bControlIsVisible = false; } //-------------------------------------------------------------------- void SAL_CALL ViewObjectContactOfUnoControl_Impl::propertyChange( const PropertyChangeEvent& /*_rEvent*/ ) throw(RuntimeException) { ::vos::OGuard aSolarGuard( Application::GetSolarMutex() ); // (re)painting might require VCL operations, which need the SolarMutex OSL_PRECOND( !impl_isDisposed_nofail(), "ViewObjectContactOfUnoControl_Impl::propertyChange: already disposed()" ); if ( impl_isDisposed_nofail() ) return; VOCGuard aGuard( *this ); DBG_ASSERT( m_aControl.is(), "ViewObjectContactOfUnoControl_Impl::propertyChange: " ); if ( !m_aControl.is() ) return; // a generic property changed. If we're in design mode, we need to repaint the control if ( impl_isControlDesignMode_nothrow() ) { m_pAntiImpl->propertyChange(); } } //-------------------------------------------------------------------- void SAL_CALL ViewObjectContactOfUnoControl_Impl::modeChanged( const ModeChangeEvent& _rSource ) throw (RuntimeException) { VOCGuard aGuard( *this ); DBG_ASSERT( _rSource.NewMode.equalsAscii( "design" ) || _rSource.NewMode.equalsAscii( "alive" ), "ViewObjectContactOfUnoControl_Impl::modeChanged: unexpected mode!" ); m_eControlDesignMode = _rSource.NewMode.equalsAscii( "design" ) ? eDesign : eAlive; impl_switchDesignModeListening_nothrow( impl_isControlDesignMode_nothrow() ); try { // if the control is part of a invisible layer, we need to explicitly hide it in alive mode // 2003-06-03 - #110592# - fs@openoffice.org impl_adjustControlVisibilityToLayerVisibility_throw( false ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } //-------------------------------------------------------------------- void SAL_CALL ViewObjectContactOfUnoControl_Impl::elementInserted( const ContainerEvent& /*_Event*/ ) throw (RuntimeException) { // not interested in } //-------------------------------------------------------------------- void SAL_CALL ViewObjectContactOfUnoControl_Impl::elementRemoved( const ContainerEvent& Event ) throw (RuntimeException) { ::vos::OGuard aSolarGuard( Application::GetSolarMutex() ); // some code below - in particular our disposal - might trigger actions which require the // SolarMutex. In particular, in our disposal, we remove ourself as listener from the control, // which alone needs the SolarMutex. Of course this - a removeFooListener needed the SolarMutex - // is the real bug. Toolkit really is infested with solar mutex usage ... :( // #i82169# / 2007-11-14 / frank.schoenheit@sun.com VOCGuard aGuard( *this ); DBG_ASSERT( Event.Source == m_xContainer, "ViewObjectContactOfUnoControl_Impl::elementRemoved: where did this come from?" ); if ( m_aControl == Event.Element ) impl_dispose_nothrow( false ); } //-------------------------------------------------------------------- void SAL_CALL ViewObjectContactOfUnoControl_Impl::elementReplaced( const ContainerEvent& Event ) throw (RuntimeException) { VOCGuard aGuard( *this ); DBG_ASSERT( Event.Source == m_xContainer, "ViewObjectContactOfUnoControl_Impl::elementReplaced: where did this come from?" ); if ( m_aControl == Event.ReplacedElement ) { Reference< XControl > xNewControl( Event.Element, UNO_QUERY ); DBG_ASSERT( xNewControl.is(), "ViewObjectContactOfUnoControl_Impl::elementReplaced: invalid new control!" ); if ( !xNewControl.is() ) return; DBG_ASSERT( xNewControl->getModel() == m_aControl.getModel(), "ViewObjectContactOfUnoControl_Impl::elementReplaced: another model at the new control?" ); // another model should - in the drawing layer - also imply another SdrUnoObj, which // should also result in new ViewContact, and thus in new ViewObjectContacts impl_switchControlListening_nothrow( false ); m_aControl = xNewControl; positionAndZoomControl(); m_aControl.setDesignMode( impl_isControlDesignMode_nothrow() ); m_bControlIsVisible = m_aControl.isVisible(); impl_switchControlListening_nothrow( true ); } } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl_Impl::setControlDesignMode( bool _bDesignMode ) const { if ( ( m_eControlDesignMode != eUnknown ) && ( _bDesignMode == impl_isControlDesignMode_nothrow() ) ) // nothing to do return; m_eControlDesignMode = _bDesignMode ? eDesign : eAlive; if ( !m_aControl.is() ) // nothing to do, the setting will be respected as soon as the control // is created return; try { m_aControl.setDesignMode( _bDesignMode ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } //-------------------------------------------------------------------- bool ViewObjectContactOfUnoControl_Impl::belongsToDevice( const OutputDevice* _pDevice ) const { DBG_ASSERT( _pDevice, "ViewObjectContactOfUnoControl_Impl::belongsToDevice: invalid device!" ); OSL_PRECOND( !impl_isDisposed_nofail(), "ViewObjectContactOfUnoControl_Impl::belongsToDevice: already disposed!" ); if ( impl_isDisposed_nofail() ) return false; if ( m_pOutputDeviceForWindow ) { if ( _pDevice == m_pOutputDeviceForWindow ) return true; return false; } ObjectContactOfPageView* pPageViewContact = dynamic_cast< ObjectContactOfPageView* >( &m_pAntiImpl->GetObjectContact() ); if ( pPageViewContact ) return ( _pDevice == &imp_getPageViewDevice_nothrow( *pPageViewContact ) ); DBG_ERROR( "ViewObjectContactOfUnoControl_Impl::belongsToDevice: could not determine the device I belong to!" ); return false; } //==================================================================== //= ViewObjectContactOfUnoControl //==================================================================== DBG_NAME( ViewObjectContactOfUnoControl ) //-------------------------------------------------------------------- ViewObjectContactOfUnoControl::ViewObjectContactOfUnoControl( ObjectContact& _rObjectContact, ViewContactOfUnoControl& _rViewContact ) :ViewObjectContactOfSdrObj( _rObjectContact, _rViewContact ) ,m_pImpl( new ViewObjectContactOfUnoControl_Impl( this ) ) { DBG_CTOR( ViewObjectContactOfUnoControl, NULL ); } //-------------------------------------------------------------------- ViewObjectContactOfUnoControl::~ViewObjectContactOfUnoControl() { m_pImpl->dispose(); m_pImpl = NULL; DBG_DTOR( ViewObjectContactOfUnoControl, NULL ); } //-------------------------------------------------------------------- bool ViewObjectContactOfUnoControl::isControlVisible() const { VOCGuard aGuard( *m_pImpl ); const ControlHolder& rControl( m_pImpl->getExistentControl() ); return rControl.is() && rControl.isVisible(); } //-------------------------------------------------------------------- Reference< XControl > ViewObjectContactOfUnoControl::getControl() { VOCGuard aGuard( *m_pImpl ); m_pImpl->ensureControl(); return m_pImpl->getExistentControl().getControl(); } //-------------------------------------------------------------------- Reference< XControl > ViewObjectContactOfUnoControl::getTemporaryControlForWindow( const Window& _rWindow, Reference< XControlContainer >& _inout_ControlContainer, const SdrUnoObj& _rUnoObject ) { ControlHolder aControl; InvisibleControlViewAccess aSimulatePageView( _inout_ControlContainer ); OSL_VERIFY( ViewObjectContactOfUnoControl_Impl::createControlForDevice( aSimulatePageView, _rWindow, _rUnoObject, aControl ) ); return aControl.getControl(); } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl::positionControlForPaint( const DisplayInfo& _rDisplayInfo ) const { VOCGuard aGuard( *m_pImpl ); // ensure we have a control. If we don't, then the Drawing Layer might be tempted to // never draw the complete form layer. // #i75095# / 2007-03-05 / frank.schoenheit@sun.com m_pImpl->ensureControl( _rDisplayInfo ); // position the control m_pImpl->positionControlForPaint( _rDisplayInfo ); } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl::ensureControlVisibility( bool _bVisible ) const { VOCGuard aGuard( *m_pImpl ); try { const ControlHolder& rControl( m_pImpl->getExistentControl() ); if ( !rControl.is() ) return; // only need to care for alive mode if ( rControl.isDesignMode() ) return; // is the visibility correct? if ( m_pImpl->isControlVisible() == _bVisible ) return; // no -> adjust it rControl.setVisible( _bVisible ); DBG_ASSERT( m_pImpl->isControlVisible() == _bVisible, "ViewObjectContactOfUnoControl::ensureControlVisibility: this didn't work!" ); // now this would mean that either isControlVisible is not reliable, // or that showing/hiding the window did not work as intended. } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } //-------------------------------------------------------------------- void ViewObjectContactOfUnoControl::setControlDesignMode( bool _bDesignMode ) const { VOCGuard aGuard( *m_pImpl ); m_pImpl->setControlDesignMode( _bDesignMode ); if(!_bDesignMode) { // when live mode is switched on, a refresh is needed. The edit mode visualisation // needs to be repainted and the now used VCL-Window needs to be positioned and // sized. Both is done from the repant refresh. const_cast< ViewObjectContactOfUnoControl* >(this)->ActionChanged(); } } //-------------------------------------------------------------------- bool ViewObjectContactOfUnoControl::belongsToDevice( const OutputDevice* _pDevice ) const { VOCGuard aGuard( *m_pImpl ); return m_pImpl->belongsToDevice( _pDevice ); } //-------------------------------------------------------------------- drawinglayer::primitive2d::Primitive2DSequence ViewObjectContactOfUnoControl::createPrimitive2DSequence(const DisplayInfo& rDisplayInfo) const { // force control here to make it a VCL ChildWindow. Will be fetched // and used below by getExistentControl() m_pImpl->ensureControl(); m_pImpl->positionControlForPaint(rDisplayInfo); // get needed data const ViewContactOfUnoControl& rViewContactOfUnoControl(static_cast< const ViewContactOfUnoControl& >(GetViewContact())); Reference< XControlModel > xControlModel(rViewContactOfUnoControl.GetSdrUnoObj().GetUnoControlModel()); const ControlHolder& rControl(m_pImpl->getExistentControl()); // check if we already have a XControl. if(xControlModel.is() && rControl.is()) { // create a primitive and hand over the existing xControl. This will // allow the primitive to not need to create another one on demand. // Do use model data directly to create the correct geometry. Do NOT // use getBoundRect()/getSnapRect() here; tese will use the sequence of // primitives themselves in the long run. const Rectangle aUnoControlModelData(rViewContactOfUnoControl.GetSdrUnoObj().GetGeoRect()); const basegfx::B2DRange aRange(aUnoControlModelData.Left(), aUnoControlModelData.Top(), aUnoControlModelData.Right(), aUnoControlModelData.Bottom()); // create object transform basegfx::B2DHomMatrix aTransform; aTransform.set(0, 0, aRange.getWidth()); aTransform.set(1, 1, aRange.getHeight()); aTransform.set(0, 2, aRange.getMinX()); aTransform.set(1, 2, aRange.getMinY()); // create control primitive with existing XControl const drawinglayer::primitive2d::Primitive2DReference xRetval(new drawinglayer::primitive2d::ControlPrimitive2D( aTransform, xControlModel, rControl.getControl())); return drawinglayer::primitive2d::Primitive2DSequence(&xRetval, 1); } else { // use the default mechanism. This will create a ControlPrimitive2D without // handing over a XControl. If not even a XControlModel exists, it will // create the SdrObject fallback visualisation return rViewContactOfUnoControl.getViewIndependentPrimitive2DSequence(); } } void ViewObjectContactOfUnoControl::propertyChange() { // graphical invalidate at all views ActionChanged(); // #i93318# flush Primitive2DSequence to force recreation with updated XControlModel // since e.g. background color has changed and existing decompositions are evtl. no // longer valid. Unfortunately this is not detected from ControlPrimitive2D::operator== // since it only has a uno reference to the XControlModel flushPrimitive2DSequence(); } void ViewObjectContactOfUnoControl::ActionChanged() { // call parent ViewObjectContactOfSdrObj::ActionChanged(); const ControlHolder& rControl(m_pImpl->getExistentControl()); if(rControl.is() && !rControl.isDesignMode()) { // #i93180# if layer visibility has changed and control is in live mode, it is necessary // to correct visibility to make those control vanish on SdrObject LayerID changes const SdrPageView* pSdrPageView = GetObjectContact().TryToGetSdrPageView(); if(pSdrPageView) { const bool bIsLayerVisible(pSdrPageView->GetVisibleLayers().IsSet(getSdrObject().GetLayer())); if(rControl.isVisible() != bIsLayerVisible) { rControl.setVisible(bIsLayerVisible); } } } } //==================================================================== //= UnoControlDefaultContact //==================================================================== DBG_NAME( UnoControlDefaultContact ) //-------------------------------------------------------------------- UnoControlDefaultContact::UnoControlDefaultContact( ObjectContact& _rObjectContact, ViewContactOfUnoControl& _rViewContact ) :ViewObjectContactOfUnoControl( _rObjectContact, _rViewContact ) { DBG_CTOR( UnoControlDefaultContact, NULL ); } //-------------------------------------------------------------------- UnoControlDefaultContact::~UnoControlDefaultContact() { DBG_DTOR( UnoControlDefaultContact, NULL ); } //==================================================================== //= UnoControlWindowContact //==================================================================== DBG_NAME( UnoControlWindowContact ) //-------------------------------------------------------------------- UnoControlWindowContact::UnoControlWindowContact( ObjectContactOfPageView& _rObjectContact, ViewContactOfUnoControl& _rViewContact ) :ViewObjectContactOfUnoControl( _rObjectContact, _rViewContact ) { DBG_CTOR( UnoControlWindowContact, NULL ); } //-------------------------------------------------------------------- UnoControlWindowContact::~UnoControlWindowContact() { DBG_DTOR( UnoControlWindowContact, NULL ); } //==================================================================== //= UnoControlPrintOrPreviewContact //==================================================================== DBG_NAME( UnoControlPrintOrPreviewContact ) //-------------------------------------------------------------------- UnoControlPrintOrPreviewContact::UnoControlPrintOrPreviewContact( ObjectContactOfPageView& _rObjectContact, ViewContactOfUnoControl& _rViewContact ) :ViewObjectContactOfUnoControl( _rObjectContact, _rViewContact ) { DBG_CTOR( UnoControlPrintOrPreviewContact, NULL ); } //-------------------------------------------------------------------- UnoControlPrintOrPreviewContact::~UnoControlPrintOrPreviewContact() { DBG_DTOR( UnoControlPrintOrPreviewContact, NULL ); } //==================================================================== //= UnoControlPDFExportContact //==================================================================== DBG_NAME( UnoControlPDFExportContact ) //-------------------------------------------------------------------- UnoControlPDFExportContact::UnoControlPDFExportContact( ObjectContactOfPageView& _rObjectContact, ViewContactOfUnoControl& _rViewContact ) :ViewObjectContactOfUnoControl( _rObjectContact, _rViewContact ) { DBG_CTOR( UnoControlPDFExportContact, NULL ); } //-------------------------------------------------------------------- UnoControlPDFExportContact::~UnoControlPDFExportContact() { DBG_DTOR( UnoControlPDFExportContact, NULL ); } //........................................................................ } } // namespace sdr::contact //........................................................................