/* -*- 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 "framecontrol.hxx" #include #include #include #include #include #include #include #include #include #include #include // namespaces using namespace ::osl; using namespace ::cppu; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::awt; using namespace ::com::sun::star::frame; using namespace ::com::sun::star::util; namespace unocontrols{ // construct/destruct FrameControl::FrameControl( const Reference< XComponentContext >& rxContext) : BaseControl ( rxContext ) , OBroadcastHelper ( m_aMutex ) , OPropertySetHelper ( *(static_cast< OBroadcastHelper * >(this)) ) , m_aConnectionPointContainer ( new OConnectionPointContainerHelper(m_aMutex) ) { } FrameControl::~FrameControl() { } // XInterface Any SAL_CALL FrameControl::queryInterface( const Type& rType ) throw( RuntimeException, std::exception ) { // Attention: // Don't use mutex or guard in this method!!! Is a method of XInterface. Any aReturn; Reference< XInterface > xDel = BaseControl::impl_getDelegator(); if ( xDel.is() ) { // If an delegator exist, forward question to his queryInterface. // Delegator will ask his own queryAggregation! aReturn = xDel->queryInterface( rType ); } else { // If an delegator unknown, forward question to own queryAggregation. aReturn = queryAggregation( rType ); } return aReturn; } // XInterface void SAL_CALL FrameControl::acquire() throw() { // Attention: // Don't use mutex or guard in this method!!! Is a method of XInterface. // Forward to baseclass BaseControl::acquire(); } // XInterface void SAL_CALL FrameControl::release() throw() { // Attention: // Don't use mutex or guard in this method!!! Is a method of XInterface. // Forward to baseclass BaseControl::release(); } // XTypeProvider Sequence< Type > SAL_CALL FrameControl::getTypes() throw( RuntimeException, std::exception ) { // Optimize this method ! // We initialize a static variable only one time. And we don't must use a mutex at every call! // For the first call; pTypeCollection is NULL - for the second call pTypeCollection is different from NULL! static OTypeCollection* pTypeCollection = nullptr; if ( pTypeCollection == nullptr ) { // Ready for multithreading; get global mutex for first call of this method only! see before MutexGuard aGuard( Mutex::getGlobalMutex() ); // Control these pointer again ... it can be, that another instance will be faster then these! if ( pTypeCollection == nullptr ) { // Create a static typecollection ... static OTypeCollection aTypeCollection ( cppu::UnoType::get(), cppu::UnoType::get(), cppu::UnoType::get(), BaseControl::getTypes() ); // ... and set his address to static pointer! pTypeCollection = &aTypeCollection; } } return pTypeCollection->getTypes(); } // XAggregation Any SAL_CALL FrameControl::queryAggregation( const Type& aType ) throw( RuntimeException, std::exception ) { // Ask for my own supported interfaces ... // Attention: XTypeProvider and XInterface are supported by OComponentHelper! Any aReturn ( ::cppu::queryInterface( aType , static_cast< XControlModel* > ( this ) , static_cast< XConnectionPointContainer* > ( this ) ) ); // If searched interface not supported by this class ... if ( !aReturn.hasValue() ) { // ... ask baseclasses. aReturn = OPropertySetHelper::queryInterface( aType ); if ( !aReturn.hasValue() ) { aReturn = BaseControl::queryAggregation( aType ); } } return aReturn; } OUString FrameControl::getImplementationName() throw (css::uno::RuntimeException, std::exception) { return impl_getStaticImplementationName(); } css::uno::Sequence FrameControl::getSupportedServiceNames() throw (css::uno::RuntimeException, std::exception) { return impl_getStaticSupportedServiceNames(); } // XControl void SAL_CALL FrameControl::createPeer( const Reference< XToolkit >& xToolkit , const Reference< XWindowPeer >& xParentPeer ) throw( RuntimeException, std::exception ) { BaseControl::createPeer( xToolkit, xParentPeer ); if ( impl_getPeerWindow().is() ) { if( !m_sComponentURL.isEmpty() ) { impl_createFrame( getPeer(), m_sComponentURL, m_seqLoaderArguments ); } } } // XControl sal_Bool SAL_CALL FrameControl::setModel( const Reference< XControlModel >& /*xModel*/ ) throw( RuntimeException, std::exception ) { // We have no model. return false; } // XControl Reference< XControlModel > SAL_CALL FrameControl::getModel() throw( RuntimeException, std::exception ) { // We have no model. return Reference< XControlModel >(); } // XControl void SAL_CALL FrameControl::dispose() throw( RuntimeException, std::exception ) { impl_deleteFrame(); BaseControl::dispose(); } // XView sal_Bool SAL_CALL FrameControl::setGraphics( const Reference< XGraphics >& /*xDevice*/ ) throw( RuntimeException, std::exception ) { // it is not possible to print this control return false; } // XView Reference< XGraphics > SAL_CALL FrameControl::getGraphics() throw( RuntimeException, std::exception ) { // when it's not possible to set graphics ! then it's possible to return null return Reference< XGraphics >(); } // XConnectionPointContainer Sequence< Type > SAL_CALL FrameControl::getConnectionPointTypes() throw( RuntimeException, std::exception ) { // Forwarded to helper class return m_aConnectionPointContainer->getConnectionPointTypes(); } // XConnectionPointContainer Reference< XConnectionPoint > SAL_CALL FrameControl::queryConnectionPoint( const Type& aType ) throw( RuntimeException, std::exception ) { // Forwarded to helper class return m_aConnectionPointContainer->queryConnectionPoint( aType ); } // XConnectionPointContainer void SAL_CALL FrameControl::advise( const Type& aType , const Reference< XInterface >& xListener ) throw( RuntimeException, std::exception ) { // Forwarded to helper class m_aConnectionPointContainer->advise( aType, xListener ); } // XConnectionPointContainer void SAL_CALL FrameControl::unadvise( const Type& aType , const Reference< XInterface >& xListener ) throw( RuntimeException, std::exception ) { // Forwarded to helper class m_aConnectionPointContainer->unadvise( aType, xListener ); } // impl but public method to register service const Sequence< OUString > FrameControl::impl_getStaticSupportedServiceNames() { Sequence seqServiceNames { SERVICENAME_FRAMECONTROL }; return seqServiceNames; } // impl but public method to register service const OUString FrameControl::impl_getStaticImplementationName() { return OUString(IMPLEMENTATIONNAME_FRAMECONTROL); } // OPropertySetHelper sal_Bool FrameControl::convertFastPropertyValue( Any& rConvertedValue , Any& rOldValue , sal_Int32 nHandle , const Any& rValue ) throw( IllegalArgumentException ) { bool bReturn = false; switch (nHandle) { case PROPERTYHANDLE_COMPONENTURL : rConvertedValue = rValue; rOldValue <<= m_sComponentURL; bReturn = true; break; case PROPERTYHANDLE_LOADERARGUMENTS : rConvertedValue = rValue; rOldValue <<= m_seqLoaderArguments; bReturn = true; break; } if ( !bReturn ) { throw IllegalArgumentException(); } return bReturn; } // OPropertySetHelper void FrameControl::setFastPropertyValue_NoBroadcast( sal_Int32 nHandle , const Any& rValue ) throw ( css::uno::Exception, std::exception ) { // this method only set the value MutexGuard aGuard (m_aMutex); switch (nHandle) { case PROPERTYHANDLE_COMPONENTURL : rValue >>= m_sComponentURL; if (getPeer().is()) { impl_createFrame ( getPeer(), m_sComponentURL, m_seqLoaderArguments ); } break; case PROPERTYHANDLE_LOADERARGUMENTS : rValue >>= m_seqLoaderArguments; break; default : OSL_ENSURE ( nHandle == -1, "This is an invalid property handle." ); } } // OPropertySetHelper void FrameControl::getFastPropertyValue( Any& rRet , sal_Int32 nHandle ) const { MutexGuard aGuard ( Mutex::getGlobalMutex() ); switch (nHandle) { case PROPERTYHANDLE_COMPONENTURL : rRet <<= m_sComponentURL; break; case PROPERTYHANDLE_LOADERARGUMENTS : rRet <<= m_seqLoaderArguments; break; case PROPERTYHANDLE_FRAME : rRet <<= m_xFrame; break; default : OSL_ENSURE ( nHandle == -1, "This is an invalid property handle." ); } } // OPropertySetHelper IPropertyArrayHelper& FrameControl::getInfoHelper() { // Create a table that map names to index values. static OPropertyArrayHelper* pInfo; if (!pInfo) { // global method must be guarded MutexGuard aGuard ( Mutex::getGlobalMutex() ); if (!pInfo) { pInfo = new OPropertyArrayHelper( impl_getStaticPropertyDescriptor(), true ); } } return *pInfo; } // OPropertySetHelper Reference< XPropertySetInfo > SAL_CALL FrameControl::getPropertySetInfo() throw( RuntimeException, std::exception ) { // Optimize this method ! // We initialize a static variable only one time. And we don't must use a mutex at every call! // For the first call; pInfo is NULL - for the second call pInfo is different from NULL! static Reference< XPropertySetInfo >* pInfo = nullptr; if ( pInfo == nullptr ) { // Ready for multithreading MutexGuard aGuard ( Mutex::getGlobalMutex () ); // Control this pointer again, another instance can be faster then these! if ( pInfo == nullptr ) { // Create structure of propertysetinfo for baseclass "OPropertySetHelper". // (Use method "getInfoHelper()".) static Reference< XPropertySetInfo > xInfo ( createPropertySetInfo ( getInfoHelper () ) ); pInfo = &xInfo; } } return ( *pInfo ); } // BaseControl WindowDescriptor* FrameControl::impl_getWindowDescriptor( const Reference< XWindowPeer >& xParentPeer ) { WindowDescriptor* pDescriptor = new WindowDescriptor; pDescriptor->Type = WindowClass_CONTAINER; pDescriptor->ParentIndex = -1; pDescriptor->Parent = xParentPeer; pDescriptor->Bounds = getPosSize (); pDescriptor->WindowAttributes = 0; return pDescriptor; } // private method void FrameControl::impl_createFrame( const Reference< XWindowPeer >& xPeer , const OUString& rURL , const Sequence< PropertyValue >& rArguments ) { Reference< XFrame2 > xOldFrame; Reference< XFrame2 > xNewFrame; { MutexGuard aGuard ( m_aMutex ); xOldFrame = m_xFrame; } xNewFrame = Frame::create( impl_getComponentContext() ); Reference< XWindow > xWP ( xPeer, UNO_QUERY ); xNewFrame->initialize ( xWP ); // option //xFrame->setName( "WhatYouWant" ); Reference< XURLTransformer > xTrans = URLTransformer::create( impl_getComponentContext() ); // load file URL aURL; aURL.Complete = rURL; xTrans->parseStrict( aURL ); Reference< XDispatch > xDisp = xNewFrame->queryDispatch ( aURL, OUString (), FrameSearchFlag::SELF ); if (xDisp.is()) { xDisp->dispatch ( aURL, rArguments ); } // set the frame { MutexGuard aGuard ( m_aMutex ); m_xFrame = xNewFrame; } // notify the listeners sal_Int32 nFrameId = PROPERTYHANDLE_FRAME; Any aNewFrame ( &xNewFrame, cppu::UnoType::get()); Any aOldFrame ( &xOldFrame, cppu::UnoType::get()); fire ( &nFrameId, &aNewFrame, &aOldFrame, 1, false ); if (xOldFrame.is()) { xOldFrame->dispose (); } } // private method void FrameControl::impl_deleteFrame() { Reference< XFrame2 > xOldFrame; Reference< XFrame2 > xNullFrame; { // do not dispose the frame in this guarded section (deadlock?) MutexGuard aGuard( m_aMutex ); xOldFrame = m_xFrame; m_xFrame.clear(); } // notify the listeners sal_Int32 nFrameId = PROPERTYHANDLE_FRAME; Any aNewFrame( &xNullFrame, cppu::UnoType::get()); Any aOldFrame( &xOldFrame, cppu::UnoType::get()); fire( &nFrameId, &aNewFrame, &aOldFrame, 1, false ); // dispose the frame if( xOldFrame.is() ) xOldFrame->dispose(); } // private method const Sequence< Property >& FrameControl::impl_getStaticPropertyDescriptor() { // All Properties of this implementation. The array must be sorted! static const Property pPropertys[PROPERTY_COUNT] = { Property( PROPERTYNAME_COMPONENTURL, PROPERTYHANDLE_COMPONENTURL, cppu::UnoType::get(), PropertyAttribute::BOUND | PropertyAttribute::CONSTRAINED ), Property( PROPERTYNAME_FRAME, PROPERTYHANDLE_FRAME, cppu::UnoType::get(), PropertyAttribute::BOUND | PropertyAttribute::TRANSIENT ), Property( PROPERTYNAME_LOADERARGUMENTS, PROPERTYHANDLE_LOADERARGUMENTS, cppu::UnoType>::get(), PropertyAttribute::BOUND | PropertyAttribute::CONSTRAINED ) }; static const Sequence< Property > seqPropertys( pPropertys, PROPERTY_COUNT ); return seqPropertys; } } // namespace unocontrols /* vim:set shiftwidth=4 softtabstop=4 expandtab: */