diff options
Diffstat (limited to 'extensions/source/propctrlr/propcontroller.cxx')
-rw-r--r-- | extensions/source/propctrlr/propcontroller.cxx | 1773 |
1 files changed, 0 insertions, 1773 deletions
diff --git a/extensions/source/propctrlr/propcontroller.cxx b/extensions/source/propctrlr/propcontroller.cxx deleted file mode 100644 index bdb662621..000000000 --- a/extensions/source/propctrlr/propcontroller.cxx +++ /dev/null @@ -1,1773 +0,0 @@ -/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ -/************************************************************************* - * - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * Copyright 2000, 2010 Oracle and/or its affiliates. - * - * OpenOffice.org - a multi-platform office productivity suite - * - * 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 - * <http://www.openoffice.org/license.html> - * for a copy of the LGPLv3 License. - * - ************************************************************************/ - -// MARKER(update_precomp.py): autogen include statement, do not remove -#include "precompiled_extensions.hxx" -#include "propcontroller.hxx" -#include "pcrstrings.hxx" -#include "standardcontrol.hxx" -#include "linedescriptor.hxx" -#include "propresid.hrc" -#include "formresid.hrc" -#include "propertyeditor.hxx" -#include "modulepcr.hxx" -#include "formstrings.hxx" -#include "formmetadata.hxx" -#include "formbrowsertools.hxx" -#include "propertycomposer.hxx" - -/** === begin UNO includes === **/ -#include <com/sun/star/awt/XWindow.hpp> -#include <com/sun/star/util/XCloseable.hpp> -#include <com/sun/star/inspection/PropertyControlType.hpp> -#include <com/sun/star/ucb/AlreadyInitializedException.hpp> -/** === end UNO includes === **/ -#include <tools/debug.hxx> -#include <tools/diagnose_ex.h> -#include <comphelper/types.hxx> -#include <comphelper/extract.hxx> -#include <toolkit/awt/vclxwindow.hxx> -#include <toolkit/unohlp.hxx> -#include <comphelper/property.hxx> -#include <vcl/msgbox.hxx> -#include <vcl/svapp.hxx> -#include <osl/mutex.hxx> -#include <cppuhelper/component_context.hxx> -#include <cppuhelper/exc_hlp.hxx> - -#include <algorithm> -#include <functional> -#include <sal/macros.h> - -//------------------------------------------------------------------------ -// !!! outside the namespace !!! -extern "C" void SAL_CALL createRegistryInfo_OPropertyBrowserController() -{ - ::pcr::OAutoRegistration< ::pcr::OPropertyBrowserController > aAutoRegistration; -} - -//............................................................................ -namespace pcr -{ -//............................................................................ - - using namespace ::com::sun::star; - using namespace ::com::sun::star::uno; - using namespace ::com::sun::star::awt; - using namespace ::com::sun::star::form; - using namespace ::com::sun::star::beans; - using namespace ::com::sun::star::script; - using namespace ::com::sun::star::lang; - using namespace ::com::sun::star::container; - using namespace ::com::sun::star::frame; - using namespace ::com::sun::star::util; - using namespace ::com::sun::star::inspection; - using namespace ::com::sun::star::ucb; - using namespace ::comphelper; - -#define THISREF() static_cast< XController* >(this) - - //======================================================================== - //= OPropertyBrowserController - //======================================================================== - DBG_NAME(OPropertyBrowserController) - //------------------------------------------------------------------------ - OPropertyBrowserController::OPropertyBrowserController( const Reference< XComponentContext >& _rxContext ) - :m_aContext(_rxContext) - ,m_aDisposeListeners( m_aMutex ) - ,m_aControlObservers( m_aMutex ) - ,m_pView(NULL) - ,m_bContainerFocusListening( false ) - ,m_bSuspendingPropertyHandlers( false ) - ,m_bConstructed( false ) - ,m_bBindingIntrospectee( false ) - { - DBG_CTOR(OPropertyBrowserController,NULL); - } - - //------------------------------------------------------------------------ - OPropertyBrowserController::~OPropertyBrowserController() - { - // stop listening for property changes - acquire(); - stopInspection( true ); - DBG_DTOR(OPropertyBrowserController,NULL); - } - - //------------------------------------------------------------------------ - IMPLEMENT_FORWARD_REFCOUNT( OPropertyBrowserController, OPropertyBrowserController_Base ) - - //------------------------------------------------------------------------ - Any SAL_CALL OPropertyBrowserController::queryInterface( const Type& _rType ) throw (RuntimeException) - { - Any aReturn = OPropertyBrowserController_Base::queryInterface( _rType ); - if ( !aReturn.hasValue() ) - aReturn = ::cppu::queryInterface( - _rType, - static_cast< XObjectInspectorUI* >( this ) - ); - return aReturn; - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::startContainerWindowListening() - { - if (m_bContainerFocusListening) - return; - - if (m_xFrame.is()) - { - Reference< XWindow > xContainerWindow = m_xFrame->getContainerWindow(); - if (xContainerWindow.is()) - { - xContainerWindow->addFocusListener(this); - m_bContainerFocusListening = sal_True; - } - } - - DBG_ASSERT(m_bContainerFocusListening, "OPropertyBrowserController::startContainerWindowListening: unable to start listening (inconsistence)!"); - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::stopContainerWindowListening() - { - if (!m_bContainerFocusListening) - return; - - if (m_xFrame.is()) - { - Reference< XWindow > xContainerWindow = m_xFrame->getContainerWindow(); - if (xContainerWindow.is()) - { - xContainerWindow->removeFocusListener(this); - m_bContainerFocusListening = sal_False; - } - } - - DBG_ASSERT(!m_bContainerFocusListening, "OPropertyBrowserController::stopContainerWindowListening: unable to stop listening (inconsistence)!"); - } - - //-------------------------------------------------------------------- - Reference< XObjectInspectorModel > SAL_CALL OPropertyBrowserController::getInspectorModel() throw (RuntimeException) - { - return m_xModel; - } - - //-------------------------------------------------------------------- - void OPropertyBrowserController::impl_initializeView_nothrow() - { - OSL_PRECOND( haveView(), "OPropertyBrowserController::impl_initializeView_nothrow: not to be called when we have no view!" ); - if ( !haveView() ) - return; - - if ( !m_xModel.is() ) - // allowed - return; - - try - { - getPropertyBox().EnableHelpSection( m_xModel->getHasHelpSection() ); - getPropertyBox().SetHelpLineLimites( m_xModel->getMinHelpTextLines(), m_xModel->getMaxHelpTextLines() ); - } - catch( const Exception& ) - { - DBG_UNHANDLED_EXCEPTION(); - } - } - - //-------------------------------------------------------------------- - void OPropertyBrowserController::impl_updateReadOnlyView_nothrow() - { - // this is a huge cudgel, admitted. - // The problem is that in case we were previously read-only, all our controls - // were created read-only, too. We cannot simply switch them to not-read-only. - // Even if they had an API for this, we do not know whether they were - // originally created read-only, or if they are read-only just because - // the model was. - impl_rebindToInspectee_nothrow( m_aInspectedObjects ); - } - - //-------------------------------------------------------------------- - bool OPropertyBrowserController::impl_isReadOnlyModel_throw() const - { - if ( !m_xModel.is() ) - return false; - - return m_xModel->getIsReadOnly(); - } - - //-------------------------------------------------------------------- - void OPropertyBrowserController::impl_startOrStopModelListening_nothrow( bool _bDoListen ) const - { - try - { - Reference< XPropertySet > xModelProperties( m_xModel, UNO_QUERY ); - if ( !xModelProperties.is() ) - // okay, so the model doesn't want to change its properties - // dynamically - fine with us - return; - - void (SAL_CALL XPropertySet::*pListenerOperation)( const ::rtl::OUString&, const Reference< XPropertyChangeListener >& ) - = _bDoListen ? &XPropertySet::addPropertyChangeListener : &XPropertySet::removePropertyChangeListener; - - (xModelProperties.get()->*pListenerOperation)( - ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "IsReadOnly" ) ), - const_cast< OPropertyBrowserController* >( this ) - ); - } - catch( const Exception& ) - { - DBG_UNHANDLED_EXCEPTION(); - } - } - - //-------------------------------------------------------------------- - void OPropertyBrowserController::impl_bindToNewModel_nothrow( const Reference< XObjectInspectorModel >& _rxInspectorModel ) - { - impl_startOrStopModelListening_nothrow( false ); - m_xModel = _rxInspectorModel; - impl_startOrStopModelListening_nothrow( true ); - - // initialize the view, if we already have one - if ( haveView() ) - impl_initializeView_nothrow(); - - // inspect again, if we already have inspectees - if ( !m_aInspectedObjects.empty() ) - impl_rebindToInspectee_nothrow( m_aInspectedObjects ); - } - - //-------------------------------------------------------------------- - void SAL_CALL OPropertyBrowserController::setInspectorModel( const Reference< XObjectInspectorModel >& _inspectorModel ) throw (RuntimeException) - { - ::osl::MutexGuard aGuard( m_aMutex ); - - if ( m_xModel == _inspectorModel ) - return; - - impl_bindToNewModel_nothrow( _inspectorModel ); - } - - //-------------------------------------------------------------------- - Reference< XObjectInspectorUI > SAL_CALL OPropertyBrowserController::getInspectorUI() throw (RuntimeException) - { - // we're derived from this interface, though we do not expose it in queryInterface and getTypes. - return this; - } - - //-------------------------------------------------------------------- - void SAL_CALL OPropertyBrowserController::inspect( const Sequence< Reference< XInterface > >& _rObjects ) throw (com::sun::star::util::VetoException, RuntimeException) - { - SolarMutexGuard aSolarGuard; - ::osl::MutexGuard aGuard( m_aMutex ); - - if ( m_bSuspendingPropertyHandlers || !suspendAll_nothrow() ) - { // we already are trying to suspend the component (this is somewhere up the stack) - // OR one of our property handlers raised a veto against closing. Well, we *need* to close - // it in order to inspect another object. - throw VetoException(); - } - if ( m_bBindingIntrospectee ) - throw VetoException(); - - m_bBindingIntrospectee = true; - impl_rebindToInspectee_nothrow( InterfaceArray( _rObjects.getConstArray(), _rObjects.getConstArray() + _rObjects.getLength() ) ); - m_bBindingIntrospectee = false; - - } - - //-------------------------------------------------------------------- - Reference< XDispatch > SAL_CALL OPropertyBrowserController::queryDispatch( const URL& /*URL*/, const ::rtl::OUString& /*TargetFrameName*/, ::sal_Int32 /*SearchFlags*/ ) throw (RuntimeException) - { - // we don't have any dispatches at all, right now - return Reference< XDispatch >(); - } - - //-------------------------------------------------------------------- - Sequence< Reference< XDispatch > > SAL_CALL OPropertyBrowserController::queryDispatches( const Sequence< DispatchDescriptor >& Requests ) throw (RuntimeException) - { - Sequence< Reference< XDispatch > > aReturn; - sal_Int32 nLen = Requests.getLength(); - aReturn.realloc( nLen ); - - Reference< XDispatch >* pReturn = aReturn.getArray(); - const Reference< XDispatch >* pReturnEnd = aReturn.getArray() + nLen; - const DispatchDescriptor* pDescripts = Requests.getConstArray(); - - for ( ; pReturn != pReturnEnd; ++ pReturn, ++pDescripts ) - *pReturn = queryDispatch( pDescripts->FeatureURL, pDescripts->FrameName, pDescripts->SearchFlags ); - - return aReturn; - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::initialize( const Sequence< Any >& _arguments ) throw (Exception, RuntimeException) - { - if ( m_bConstructed ) - throw AlreadyInitializedException(); - - StlSyntaxSequence< Any > arguments( _arguments ); - if ( arguments.empty() ) - { // constructor: "createDefault()" - createDefault(); - return; - } - - Reference< XObjectInspectorModel > xModel; - if ( arguments.size() == 1 ) - { // constructor: "createWithModel( XObjectInspectorModel )" - if ( !( arguments[0] >>= xModel ) ) - throw IllegalArgumentException( ::rtl::OUString(), *this, 0 ); - createWithModel( xModel ); - return; - } - - throw IllegalArgumentException( ::rtl::OUString(), *this, 0 ); - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::createDefault() - { - m_bConstructed = true; - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::createWithModel( const Reference< XObjectInspectorModel >& _rxModel ) - { - osl_incrementInterlockedCount( &m_refCount ); - { - setInspectorModel( _rxModel ); - } - osl_decrementInterlockedCount( &m_refCount ); - - m_bConstructed = true; - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::attachFrame( const Reference< XFrame >& _rxFrame ) throw(RuntimeException) - { - SolarMutexGuard aSolarGuard; - ::osl::MutexGuard aGuard( m_aMutex ); - - if (_rxFrame.is() && haveView()) - throw RuntimeException(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("Unable to attach to a second frame.")),*this); - - // revoke as focus listener from the old container window - stopContainerWindowListening(); - - m_xFrame = _rxFrame; - if (!m_xFrame.is()) - return; - - // TODO: this construction perhaps should be done outside. Don't know the exact meaning of attachFrame. - // Maybe it is intended to only announce the frame to the controller, and the instance doing this - // announcement is responsible for calling setComponent, too. - Reference< XWindow > xContainerWindow = m_xFrame->getContainerWindow(); - VCLXWindow* pContainerWindow = VCLXWindow::GetImplementation(xContainerWindow); - Window* pParentWin = pContainerWindow ? pContainerWindow->GetWindow() : NULL; - if (!pParentWin) - throw RuntimeException(::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("The frame is invalid. Unable to extract the container window.")),*this); - - if ( Construct( pParentWin ) ) - { - try - { - m_xFrame->setComponent( VCLUnoHelper::GetInterface( m_pView ), this ); - } - catch( const Exception& ) - { - OSL_FAIL( "OPropertyBrowserController::attachFrame: caught an exception!" ); - } - } - - startContainerWindowListening(); - - UpdateUI(); - } - - //------------------------------------------------------------------------ - sal_Bool SAL_CALL OPropertyBrowserController::attachModel( const Reference< XModel >& _rxModel ) throw(RuntimeException) - { - Reference< XObjectInspectorModel > xModel( _rxModel, UNO_QUERY ); - if ( !xModel.is() ) - return false; - - setInspectorModel( xModel ); - return getInspectorModel() == _rxModel; - } - - //------------------------------------------------------------------------ - sal_Bool OPropertyBrowserController::suspendAll_nothrow() - { - // if there is a handle inside its "onInteractivePropertySelection" method, - // then veto - // Normally, we could expect every handler to do this itself, but being - // realistic, it's safer to handle this here in general. - if ( m_xInteractiveHandler.is() ) - return sal_False; - - m_bSuspendingPropertyHandlers = true; - sal_Bool bHandlerVeto = !suspendPropertyHandlers_nothrow( sal_True ); - m_bSuspendingPropertyHandlers = false; - if ( bHandlerVeto ) - return sal_False; - - return sal_True; - } - - //------------------------------------------------------------------------ - sal_Bool OPropertyBrowserController::suspendPropertyHandlers_nothrow( sal_Bool _bSuspend ) - { - PropertyHandlerArray aAllHandlers; // will contain every handler exactly once - for ( PropertyHandlerRepository::const_iterator handler = m_aPropertyHandlers.begin(); - handler != m_aPropertyHandlers.end(); - ++handler - ) - { - if ( ::std::find( aAllHandlers.begin(), aAllHandlers.end(), handler->second ) != aAllHandlers.end() ) - // already visited this particular handler (m_aPropertyHandlers usually contains - // the same handler more than once) - continue; - aAllHandlers.push_back( handler->second ); - } - - for ( PropertyHandlerArray::iterator loop = aAllHandlers.begin(); - loop != aAllHandlers.end(); - ++loop - ) - { - try - { - if ( !(*loop)->suspend( _bSuspend ) ) - if ( _bSuspend ) - // if we're not suspending, but reactivating, ignore the error - return sal_False; - } - catch( const Exception& ) - { - OSL_FAIL( "OPropertyBrowserController::suspendPropertyHandlers_nothrow: caught an exception!" ); - } - } - return sal_True; - } - - //------------------------------------------------------------------------ - sal_Bool SAL_CALL OPropertyBrowserController::suspend( sal_Bool _bSuspend ) throw(RuntimeException) - { - ::osl::MutexGuard aGuard( m_aMutex ); - OSL_ENSURE( haveView(), "OPropertyBrowserController::suspend: don't have a view anymore!" ); - - if ( !_bSuspend ) - { // this means a "suspend" is to be "revoked" - suspendPropertyHandlers_nothrow( sal_False ); - // we ourself cannot revoke our suspend - return sal_False; - } - - if ( !suspendAll_nothrow() ) - return sal_False; - - // commit the editor's content - if ( haveView() ) - getPropertyBox().CommitModified(); - - // stop listening - stopContainerWindowListening(); - - // outtahere - return sal_True; - } - - //------------------------------------------------------------------------ - Any SAL_CALL OPropertyBrowserController::getViewData( ) throw(RuntimeException) - { - return makeAny( m_sPageSelection ); - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::restoreViewData( const Any& Data ) throw(RuntimeException) - { - ::rtl::OUString sPageSelection; - if ( ( Data >>= sPageSelection ) && sPageSelection.getLength() ) - { - m_sPageSelection = sPageSelection; - selectPageFromViewData(); - } - } - - //------------------------------------------------------------------------ - Reference< XModel > SAL_CALL OPropertyBrowserController::getModel( ) throw(RuntimeException) - { - // have no model - return Reference< XModel >(); - } - - //------------------------------------------------------------------------ - Reference< XFrame > SAL_CALL OPropertyBrowserController::getFrame( ) throw(RuntimeException) - { - return m_xFrame; - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::dispose( ) throw(RuntimeException) - { - SolarMutexGuard aSolarGuard; - - // stop inspecting the current object - stopInspection( false ); - - // say our dispose listeners goodbye - ::com::sun::star::lang::EventObject aEvt; - aEvt.Source = static_cast< ::cppu::OWeakObject* >(this); - m_aDisposeListeners.disposeAndClear(aEvt); - m_aControlObservers.disposeAndClear(aEvt); - - // don't delete explicitly (this is done by the frame we reside in) - m_pView = NULL; - - Reference< XComponent > xViewAsComp( m_xView, UNO_QUERY ); - if ( xViewAsComp.is() ) - xViewAsComp->removeEventListener( static_cast< XPropertyChangeListener* >( this ) ); - m_xView.clear( ); - - m_aInspectedObjects.clear(); - impl_bindToNewModel_nothrow( NULL ); - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::addEventListener( const Reference< XEventListener >& _rxListener ) throw(RuntimeException) - { - m_aDisposeListeners.addInterface(_rxListener); - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::removeEventListener( const Reference< XEventListener >& _rxListener ) throw(RuntimeException) - { - m_aDisposeListeners.removeInterface(_rxListener); - } - - //------------------------------------------------------------------------ - ::rtl::OUString SAL_CALL OPropertyBrowserController::getImplementationName( ) throw(RuntimeException) - { - return getImplementationName_static(); - } - - //------------------------------------------------------------------------ - sal_Bool SAL_CALL OPropertyBrowserController::supportsService( const ::rtl::OUString& ServiceName ) throw(RuntimeException) - { - Sequence< ::rtl::OUString > aSupported(getSupportedServiceNames()); - const ::rtl::OUString* pArray = aSupported.getConstArray(); - for (sal_Int32 i = 0; i < aSupported.getLength(); ++i, ++pArray) - if (pArray->equals(ServiceName)) - return sal_True; - return sal_False; - } - - //------------------------------------------------------------------------ - Sequence< ::rtl::OUString > SAL_CALL OPropertyBrowserController::getSupportedServiceNames( ) throw(RuntimeException) - { - return getSupportedServiceNames_static(); - } - - //------------------------------------------------------------------------ - ::rtl::OUString OPropertyBrowserController::getImplementationName_static( ) throw(RuntimeException) - { - return ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("org.openoffice.comp.extensions.ObjectInspector")); - } - - //------------------------------------------------------------------------ - Sequence< ::rtl::OUString > OPropertyBrowserController::getSupportedServiceNames_static( ) throw(RuntimeException) - { - Sequence< ::rtl::OUString > aSupported(1); - aSupported[0] = ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.inspection.ObjectInspector")); - return aSupported; - } - - //------------------------------------------------------------------------ - Reference< XInterface > SAL_CALL OPropertyBrowserController::Create(const Reference< XComponentContext >& _rxContext) - { - return *(new OPropertyBrowserController( _rxContext ) ); - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::focusGained( const FocusEvent& _rSource ) throw (RuntimeException) - { - Reference< XWindow > xSourceWindow(_rSource.Source, UNO_QUERY); - Reference< XWindow > xContainerWindow; - if (m_xFrame.is()) - xContainerWindow = m_xFrame->getContainerWindow(); - - if ( xContainerWindow.get() == xSourceWindow.get() ) - { // our container window got the focus - if ( haveView() ) - getPropertyBox().GrabFocus(); - } - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::focusLost( const FocusEvent& /*_rSource*/ ) throw (RuntimeException) - { - // not interested in - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::disposing( const EventObject& _rSource ) throw(RuntimeException) - { - if ( m_xView.is() && ( m_xView == _rSource.Source ) ) - { - m_xView = NULL; - m_pView = NULL; - } - - for ( InterfaceArray::iterator loop = m_aInspectedObjects.begin(); - loop != m_aInspectedObjects.end(); - ++loop - ) - { - if ( *loop == _rSource.Source ) - { - m_aInspectedObjects.erase( loop ); - break; - } - } - } - - //------------------------------------------------------------------------ - IMPL_LINK(OPropertyBrowserController, OnPageActivation, void*, EMPTYARG) - { - updateViewDataFromActivePage(); - return 0L; - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::updateViewDataFromActivePage() - { - if (!haveView()) - return; - - ::rtl::OUString sOldSelection = m_sPageSelection; - m_sPageSelection = ::rtl::OUString(); - - const sal_uInt16 nCurrentPage = m_pView->getActivaPage(); - if ( (sal_uInt16)-1 != nCurrentPage ) - { - for ( HashString2Int16::const_iterator pageId = m_aPageIds.begin(); - pageId != m_aPageIds.end(); - ++pageId - ) - { - if ( nCurrentPage == pageId->second ) - { - m_sPageSelection = pageId->first; - break; - } - } - } - - if ( m_sPageSelection.getLength() ) - m_sLastValidPageSelection = m_sPageSelection; - else if ( sOldSelection.getLength() ) - m_sLastValidPageSelection = sOldSelection; - } - - //------------------------------------------------------------------------ - sal_uInt16 OPropertyBrowserController::impl_getPageIdForCategory_nothrow( const ::rtl::OUString& _rCategoryName ) const - { - sal_uInt16 nPageId = (sal_uInt16)-1; - HashString2Int16::const_iterator pagePos = m_aPageIds.find( _rCategoryName ); - if ( pagePos != m_aPageIds.end() ) - nPageId = pagePos->second; - return nPageId; - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::selectPageFromViewData() - { - sal_uInt16 nNewPage = impl_getPageIdForCategory_nothrow( m_sPageSelection ); - - if ( haveView() && ( nNewPage != (sal_uInt16)-1 ) ) - m_pView->activatePage( nNewPage ); - - // just in case ... - updateViewDataFromActivePage(); - } - - //------------------------------------------------------------------------ - sal_Bool OPropertyBrowserController::Construct(Window* _pParentWin) - { - DBG_ASSERT(!haveView(), "OPropertyBrowserController::Construct: already have a view!"); - DBG_ASSERT(_pParentWin, "OPropertyBrowserController::Construct: invalid parent window!"); - - m_pView = new OPropertyBrowserView(m_aContext.getLegacyServiceFactory(), _pParentWin); - m_pView->setPageActivationHandler(LINK(this, OPropertyBrowserController, OnPageActivation)); - - // add as dispose listener for our view. The view is disposed by the frame we're plugged into, - // and this disposal _deletes_ the view, so it would be deadly if we use our m_pView member - // after that - m_xView = VCLUnoHelper::GetInterface(m_pView); - Reference< XComponent > xViewAsComp(m_xView, UNO_QUERY); - if (xViewAsComp.is()) - xViewAsComp->addEventListener( static_cast< XPropertyChangeListener* >( this ) ); - - getPropertyBox().SetLineListener(this); - getPropertyBox().SetControlObserver(this); - impl_initializeView_nothrow(); - - m_pView->Show(); - - return sal_True; - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::propertyChange( const PropertyChangeEvent& _rEvent ) throw (RuntimeException) - { - if ( _rEvent.Source == m_xModel ) - { - if ( _rEvent.PropertyName.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "IsReadOnly" ) ) ) - impl_updateReadOnlyView_nothrow(); - return; - } - - if ( m_sCommittingProperty == _rEvent.PropertyName ) - return; - - if ( !haveView() ) - return; - - Any aNewValue( _rEvent.NewValue ); - if ( impl_hasPropertyHandlerFor_nothrow( _rEvent.PropertyName ) ) - { - // forward the new value to the property box, to reflect the change in the UI - aNewValue = impl_getPropertyValue_throw( _rEvent.PropertyName ); - - // check whether the state is ambiguous. This is interesting in case we display the properties - // for multiple objects at once: In this case, we'll get a notification from one of the objects, - // but need to care for the "composed" value, which can be "ambiguous". - PropertyHandlerRef xHandler( impl_getHandlerForProperty_throw( _rEvent.PropertyName ), UNO_SET_THROW ); - PropertyState ePropertyState( xHandler->getPropertyState( _rEvent.PropertyName ) ); - bool bAmbiguousValue = ( PropertyState_AMBIGUOUS_VALUE == ePropertyState ); - - getPropertyBox().SetPropertyValue( _rEvent.PropertyName, aNewValue, bAmbiguousValue ); - } - - // if it's a actuating property, then update the UI for any dependent - // properties - if ( impl_isActuatingProperty_nothrow( _rEvent.PropertyName ) ) - impl_broadcastPropertyChange_nothrow( _rEvent.PropertyName, aNewValue, _rEvent.OldValue, false ); - } - - //------------------------------------------------------------------------ - Reference< XPropertyControl > SAL_CALL OPropertyBrowserController::createPropertyControl( ::sal_Int16 ControlType, ::sal_Bool _CreateReadOnly ) throw (IllegalArgumentException, RuntimeException) - { - ::osl::MutexGuard aGuard( m_aMutex ); - - Reference< XPropertyControl > xControl; - - // default winbits: a border only - WinBits nWinBits = WB_BORDER; - - // read-only-ness - _CreateReadOnly |= (sal_Bool)impl_isReadOnlyModel_throw(); - if ( _CreateReadOnly ) - nWinBits |= WB_READONLY; - - switch ( ControlType ) - { - case PropertyControlType::StringListField: - xControl = new OMultilineEditControl( &getPropertyBox(), eStringList, nWinBits | WB_DROPDOWN | WB_TABSTOP ); - break; - - case PropertyControlType::MultiLineTextField: - xControl = new OMultilineEditControl( &getPropertyBox(), eMultiLineText, nWinBits | WB_DROPDOWN | WB_TABSTOP ); - break; - - case PropertyControlType::ListBox: - xControl = new OListboxControl( &getPropertyBox(), nWinBits | WB_TABSTOP | WB_DROPDOWN); - break; - - case PropertyControlType::ComboBox: - xControl = new OComboboxControl( &getPropertyBox(), nWinBits | WB_TABSTOP | WB_DROPDOWN); - break; - - case PropertyControlType::TextField: - xControl = new OEditControl( &getPropertyBox(), sal_False, nWinBits | WB_TABSTOP ); - break; - - case PropertyControlType::CharacterField: - xControl = new OEditControl( &getPropertyBox(), sal_True, nWinBits | WB_TABSTOP ); - break; - - case PropertyControlType::NumericField: - xControl = new ONumericControl( &getPropertyBox(), nWinBits | WB_TABSTOP | WB_SPIN | WB_REPEAT ); - break; - - case PropertyControlType::DateTimeField: - xControl = new ODateTimeControl( &getPropertyBox(), nWinBits | WB_TABSTOP ); - break; - - case PropertyControlType::DateField: - xControl = new ODateControl( &getPropertyBox(), nWinBits | WB_TABSTOP | WB_SPIN | WB_REPEAT ); - break; - - case PropertyControlType::TimeField: - xControl = new OTimeControl( &getPropertyBox(), nWinBits | WB_TABSTOP | WB_SPIN | WB_REPEAT ); - break; - - case PropertyControlType::ColorListBox: - xControl = new OColorControl( &getPropertyBox(), nWinBits | WB_TABSTOP | WB_DROPDOWN ); - break; - - case PropertyControlType::HyperlinkField: - xControl = new OHyperlinkControl( &getPropertyBox(), nWinBits | WB_TABSTOP | WB_DROPDOWN ); - break; - - default: - throw IllegalArgumentException( ::rtl::OUString(), *this, 1 ); - } - - return xControl; - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::impl_toggleInspecteeListening_nothrow( bool _bOn ) - { - for ( InterfaceArray::const_iterator loop = m_aInspectedObjects.begin(); - loop != m_aInspectedObjects.end(); - ++loop - ) - { - try - { - Reference< XComponent > xComp( *loop, UNO_QUERY ); - if ( xComp.is() ) - { - if ( _bOn ) - xComp->addEventListener( static_cast< XPropertyChangeListener* >( this ) ); - else - xComp->removeEventListener( static_cast< XPropertyChangeListener* >( this ) ); - } - } - catch( const Exception& ) - { - DBG_UNHANDLED_EXCEPTION(); - } - } - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::stopInspection( bool _bCommitModified ) - { - if ( haveView() ) - { - if ( _bCommitModified ) - // commit the editor's content - getPropertyBox().CommitModified(); - - // hide the property box so that it does not flicker - getPropertyBox().Hide(); - - // clear the property box - getPropertyBox().ClearAll(); - } - - // destroy the view first - if ( haveView() ) - { - // remove the pages - for ( HashString2Int16::const_iterator erase = m_aPageIds.begin(); - erase != m_aPageIds.end(); - ++erase - ) - getPropertyBox().RemovePage( erase->second ); - clearContainer( m_aPageIds ); - } - - clearContainer( m_aProperties ); - - // de-register as dispose-listener from our inspected objects - impl_toggleInspecteeListening_nothrow( false ); - - // handlers are obsolete, so is our "composer" for their UI requests - if ( m_pUIRequestComposer.get() ) - m_pUIRequestComposer->dispose(); - m_pUIRequestComposer.reset( NULL ); - - // clean up the property handlers - PropertyHandlerArray aAllHandlers; // will contain every handler exactly once - for ( PropertyHandlerRepository::const_iterator aHandler = m_aPropertyHandlers.begin(); - aHandler != m_aPropertyHandlers.end(); - ++aHandler - ) - if ( ::std::find( aAllHandlers.begin(), aAllHandlers.end(), aHandler->second ) == aAllHandlers.end() ) - aAllHandlers.push_back( aHandler->second ); - - for ( PropertyHandlerArray::iterator loop = aAllHandlers.begin(); - loop != aAllHandlers.end(); - ++loop - ) - { - try - { - (*loop)->removePropertyChangeListener( this ); - (*loop)->dispose(); - } - catch( const DisposedException& ) - { - } - catch( const Exception& ) - { - DBG_UNHANDLED_EXCEPTION(); - } - } - - clearContainer( m_aPropertyHandlers ); - clearContainer( m_aDependencyHandlers ); - } - - //------------------------------------------------------------------------ - bool OPropertyBrowserController::impl_hasPropertyHandlerFor_nothrow( const ::rtl::OUString& _rPropertyName ) const - { - PropertyHandlerRepository::const_iterator handlerPos = m_aPropertyHandlers.find( _rPropertyName ); - return ( handlerPos != m_aPropertyHandlers.end() ); - } - - //------------------------------------------------------------------------ - OPropertyBrowserController::PropertyHandlerRef OPropertyBrowserController::impl_getHandlerForProperty_throw( const ::rtl::OUString& _rPropertyName ) const - { - PropertyHandlerRepository::const_iterator handlerPos = m_aPropertyHandlers.find( _rPropertyName ); - if ( handlerPos == m_aPropertyHandlers.end() ) - throw RuntimeException(); - return handlerPos->second; - } - - //------------------------------------------------------------------------ - Any OPropertyBrowserController::impl_getPropertyValue_throw( const ::rtl::OUString& _rPropertyName ) - { - PropertyHandlerRef handler = impl_getHandlerForProperty_throw( _rPropertyName ); - return handler->getPropertyValue( _rPropertyName ); - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::impl_rebindToInspectee_nothrow( const InterfaceArray& _rObjects ) - { - try - { - // stop inspecting the old object(s) - stopInspection( true ); - - // inspect the new object(s) - m_aInspectedObjects = _rObjects; - doInspection(); - - // update the user interface - UpdateUI(); - } - - catch(Exception&) - { - OSL_FAIL("OPropertyBrowserController::impl_rebindToInspectee_nothrow: caught an exception !"); - } - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::doInspection() - { - try - { - ////////////////////////////////////////////////////////////////////// - // obtain the properties of the object - ::std::vector< Property > aProperties; - - PropertyHandlerArray aPropertyHandlers; - getPropertyHandlers( m_aInspectedObjects, aPropertyHandlers ); - - PropertyHandlerArray::iterator aHandler( aPropertyHandlers.begin() ); - while ( aHandler != aPropertyHandlers.end() ) - { - DBG_ASSERT( aHandler->get(), "OPropertyBrowserController::doInspection: invalid handler!" ); - - StlSyntaxSequence< Property > aThisHandlersProperties = (*aHandler)->getSupportedProperties(); - - if ( aThisHandlersProperties.empty() ) - { - // this handler doesn't know anything about the current inspectee -> ignore it - (*aHandler)->dispose(); - aHandler = aPropertyHandlers.erase( aHandler ); - continue; - } - - // append these properties to our "all properties" array - aProperties.reserve( aProperties.size() + aThisHandlersProperties.size() ); - for ( StlSyntaxSequence< Property >::const_iterator copyProperty = aThisHandlersProperties.begin(); - copyProperty != aThisHandlersProperties.end(); - ++copyProperty - ) - { - ::std::vector< Property >::const_iterator previous = ::std::find_if( - aProperties.begin(), - aProperties.end(), - FindPropertyByName( copyProperty->Name ) - ); - if ( previous == aProperties.end() ) - { - aProperties.push_back( *copyProperty ); - continue; - } - - // there already was another (previous) handler which supported this property. - // Don't add it to aProperties, again. - - // Also, ensure that handlers which previously expressed interest in *changes* - // of this property are not notified. - // This is 'cause we have a new handler which is responsible for this property, - // which means it can give it a completely different meaning than the previous - // handler for this property is prepared for. - ::std::pair< PropertyHandlerMultiRepository::iterator, PropertyHandlerMultiRepository::iterator > - aDepHandlers = m_aDependencyHandlers.equal_range( copyProperty->Name ); - m_aDependencyHandlers.erase( aDepHandlers.first, aDepHandlers.second ); - } - - // determine the superseded properties - StlSyntaxSequence< ::rtl::OUString > aSupersededByThisHandler = (*aHandler)->getSupersededProperties(); - for ( StlSyntaxSequence< ::rtl::OUString >::const_iterator superseded = aSupersededByThisHandler.begin(); - superseded != aSupersededByThisHandler.end(); - ++superseded - ) - { - ::std::vector< Property >::iterator existent = ::std::find_if( - aProperties.begin(), - aProperties.end(), - FindPropertyByName( *superseded ) - ); - if ( existent != aProperties.end() ) - // one of the properties superseded by this handler was supported by a previous - // one -> erase - aProperties.erase( existent ); - } - - // be notified of changes which this handler is responsible for - (*aHandler)->addPropertyChangeListener( this ); - - // remember this handler for every of the properties which it is responsible - // for - for ( StlSyntaxSequence< Property >::const_iterator remember = aThisHandlersProperties.begin(); - remember != aThisHandlersProperties.end(); - ++remember - ) - { - m_aPropertyHandlers[ remember->Name ] = *aHandler; - // note that this implies that if two handlers support the same property, - // the latter wins - } - - // see if the handler expresses interest in any actuating properties - StlSyntaxSequence< ::rtl::OUString > aInterestingActuations = (*aHandler)->getActuatingProperties(); - for ( StlSyntaxSequence< ::rtl::OUString >::const_iterator aLoop = aInterestingActuations.begin(); - aLoop != aInterestingActuations.end(); - ++aLoop - ) - { - m_aDependencyHandlers.insert( PropertyHandlerMultiRepository::value_type( - *aLoop, *aHandler ) ); - } - - ++aHandler; - } - - // create a new composer for UI requests coming from the handlers - m_pUIRequestComposer.reset( new ComposedPropertyUIUpdate( getInspectorUI(), this ) ); - - // sort the properties by relative position, as indicated by the model - for ( ::std::vector< Property >::const_iterator sourceProps = aProperties.begin(); - sourceProps != aProperties.end(); - ++sourceProps - ) - { - sal_Int32 nRelativePropertyOrder = sourceProps - aProperties.begin(); - if ( m_xModel.is() ) - nRelativePropertyOrder = m_xModel->getPropertyOrderIndex( sourceProps->Name ); - while ( m_aProperties.find( nRelativePropertyOrder ) != m_aProperties.end() ) - ++nRelativePropertyOrder; - m_aProperties[ nRelativePropertyOrder ] = *sourceProps; - } - - // be notified when one of our inspectees dies - impl_toggleInspecteeListening_nothrow( true ); - } - catch(Exception&) - { - OSL_FAIL("OPropertyBrowserController::doInspection : caught an exception !"); - } - } - - //------------------------------------------------------------------------ - ::com::sun::star::awt::Size SAL_CALL OPropertyBrowserController::getMinimumSize() throw (::com::sun::star::uno::RuntimeException) - { - ::com::sun::star::awt::Size aSize; - if( m_pView ) - return m_pView->getMinimumSize(); - else - return aSize; - } - - //------------------------------------------------------------------------ - ::com::sun::star::awt::Size SAL_CALL OPropertyBrowserController::getPreferredSize() throw (::com::sun::star::uno::RuntimeException) - { - return getMinimumSize(); - } - - //------------------------------------------------------------------------ - ::com::sun::star::awt::Size SAL_CALL OPropertyBrowserController::calcAdjustedSize( const ::com::sun::star::awt::Size& _rNewSize ) throw (::com::sun::star::uno::RuntimeException) - { - awt::Size aMinSize = getMinimumSize( ); - awt::Size aAdjustedSize( _rNewSize ); - if ( aAdjustedSize.Width < aMinSize.Width ) - aAdjustedSize.Width = aMinSize.Width; - if ( aAdjustedSize.Height < aMinSize.Height ) - aAdjustedSize.Height = aMinSize.Height; - return aAdjustedSize; - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::describePropertyLine( const Property& _rProperty, OLineDescriptor& _rDescriptor ) SAL_THROW((Exception)) - { - try - { - PropertyHandlerRepository::const_iterator handler = m_aPropertyHandlers.find( _rProperty.Name ); - if ( handler == m_aPropertyHandlers.end() ) - throw RuntimeException(); // caught below - - _rDescriptor.assignFrom( handler->second->describePropertyLine( _rProperty.Name, this ) ); - - ////////////////////////////////////////////////////////////////////// - - _rDescriptor.xPropertyHandler = handler->second; - _rDescriptor.sName = _rProperty.Name; - _rDescriptor.aValue = _rDescriptor.xPropertyHandler->getPropertyValue( _rProperty.Name ); - - if ( !_rDescriptor.DisplayName.getLength() ) - { - #ifdef DBG_UTIL - ::rtl::OString sMessage( "OPropertyBrowserController::describePropertyLine: handler did not provide a display name for '" ); - sMessage += ::rtl::OString( _rProperty.Name.getStr(), _rProperty.Name.getLength(), RTL_TEXTENCODING_ASCII_US ); - sMessage += ::rtl::OString( "'!" ); - DBG_ASSERT( _rDescriptor.DisplayName.getLength(), sMessage ); - #endif - _rDescriptor.DisplayName = _rProperty.Name; - } - - PropertyState ePropertyState( _rDescriptor.xPropertyHandler->getPropertyState( _rProperty.Name ) ); - if ( PropertyState_AMBIGUOUS_VALUE == ePropertyState ) - { - _rDescriptor.bUnknownValue = true; - _rDescriptor.aValue.clear(); - } - - _rDescriptor.bReadOnly = impl_isReadOnlyModel_throw(); - } - catch( const Exception& ) - { - OSL_FAIL( "OPropertyBrowserController::describePropertyLine: caught an exception!" ); - } - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::impl_buildCategories_throw() - { - OSL_PRECOND( m_aPageIds.empty(), "OPropertyBrowserController::impl_buildCategories_throw: duplicate call!" ); - - StlSyntaxSequence< PropertyCategoryDescriptor > aCategories; - if ( m_xModel.is() ) - aCategories = m_xModel->describeCategories(); - - for ( StlSyntaxSequence< PropertyCategoryDescriptor >::const_iterator category = aCategories.begin(); - category != aCategories.end(); - ++category - ) - { - OSL_ENSURE( m_aPageIds.find( category->ProgrammaticName ) == m_aPageIds.end(), - "OPropertyBrowserController::impl_buildCategories_throw: duplicate programmatic name!" ); - - m_aPageIds[ category->ProgrammaticName ] = - getPropertyBox().AppendPage( category->UIName, HelpIdUrl::getHelpId( category->HelpURL ) ); - } - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::UpdateUI() - { - try - { - if ( !haveView() ) - // too early, will return later - return; - - getPropertyBox().DisableUpdate(); - - sal_Bool bHaveFocus = getPropertyBox().HasChildPathFocus(); - - // create our tab pages - impl_buildCategories_throw(); - // (and allow for pages to be actually unused) - ::std::set< sal_uInt16 > aUsedPages; - - // when building the UI below, remember which properties are actuating, - // to allow for a initial actuatinPropertyChanged call - ::std::vector< ::rtl::OUString > aActuatingProperties; - ::std::vector< Any > aActuatingPropertyValues; - - // ask the handlers to describe the property UI, and insert the resulting - // entries into our list boxes - OrderedPropertyMap::const_iterator property( m_aProperties.begin() ); - for ( ; property != m_aProperties.end(); ++property ) - { - OLineDescriptor aDescriptor; - describePropertyLine( property->second, aDescriptor ); - - bool bIsActuatingProperty = impl_isActuatingProperty_nothrow( property->second.Name ); - - #if OSL_DEBUG_LEVEL > 0 - if ( !aDescriptor.Category.getLength() ) - { - ::rtl::OString sMessage( "OPropertyBrowserController::UpdateUI: empty category provided for property '" ); - sMessage += ::rtl::OString( property->second.Name.getStr(), property->second.Name.getLength(), osl_getThreadTextEncoding() ); - sMessage += "'!"; - OSL_FAIL( sMessage ); - } - #endif - // finally insert this property control - sal_uInt16 nTargetPageId = impl_getPageIdForCategory_nothrow( aDescriptor.Category ); - if ( nTargetPageId == (sal_uInt16)-1 ) - { - // this category does not yet exist. This is allowed, as an inspector model might be lazy, and not provide - // any category information of its own. In this case, we have a fallback ... - m_aPageIds[ aDescriptor.Category ] = - getPropertyBox().AppendPage( aDescriptor.Category, rtl::OString() ); - nTargetPageId = impl_getPageIdForCategory_nothrow( aDescriptor.Category ); - } - - getPropertyBox().InsertEntry( aDescriptor, nTargetPageId ); - aUsedPages.insert( nTargetPageId ); - - // if it's an actuating property, remember it - if ( bIsActuatingProperty ) - { - aActuatingProperties.push_back( property->second.Name ); - aActuatingPropertyValues.push_back( impl_getPropertyValue_throw( property->second.Name ) ); - } - } - - // update any dependencies for the actuating properties which we encountered - { - ::std::vector< ::rtl::OUString >::const_iterator aProperty = aActuatingProperties.begin(); - ::std::vector< Any >::const_iterator aPropertyValue = aActuatingPropertyValues.begin(); - for ( ; aProperty != aActuatingProperties.end(); ++aProperty, ++aPropertyValue ) - impl_broadcastPropertyChange_nothrow( *aProperty, *aPropertyValue, *aPropertyValue, true ); - } - - // remove any unused pages (which we did not encounter properties for) - HashString2Int16 aSurvivingPageIds; - for ( HashString2Int16::iterator pageId = m_aPageIds.begin(); - pageId != m_aPageIds.end(); - ++pageId - ) - { - if ( aUsedPages.find( pageId->second ) == aUsedPages.end() ) - getPropertyBox().RemovePage( pageId->second ); - else - aSurvivingPageIds.insert( *pageId ); - } - m_aPageIds.swap( aSurvivingPageIds ); - - - getPropertyBox().Show(); - getPropertyBox().EnableUpdate(); - if ( bHaveFocus ) - getPropertyBox().GrabFocus(); - - // activate the first page - if ( !m_aPageIds.empty() ) - { - Sequence< PropertyCategoryDescriptor > aCategories( m_xModel->describeCategories() ); - if ( aCategories.getLength() ) - m_pView->activatePage( m_aPageIds[ aCategories[0].ProgrammaticName ] ); - else - // allowed: if we default-created the pages ... - m_pView->activatePage( m_aPageIds.begin()->second ); - } - - // activate the previously active page (if possible) - if ( m_sLastValidPageSelection.getLength() ) - m_sPageSelection = m_sLastValidPageSelection; - selectPageFromViewData(); - } - catch( const Exception& ) - { - DBG_UNHANDLED_EXCEPTION(); - } - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::Clicked( const ::rtl::OUString& _rName, sal_Bool _bPrimary ) - { - try - { - // since the browse buttons do not get the focus when clicked with the mouse, - // we need to commit the changes in the current property field - getPropertyBox().CommitModified(); - - PropertyHandlerRepository::const_iterator handler = m_aPropertyHandlers.find( _rName ); - DBG_ASSERT( handler != m_aPropertyHandlers.end(), "OPropertyBrowserController::Clicked: a property without handler? This will crash!" ); - - ComposedUIAutoFireGuard aAutoFireGuard( *m_pUIRequestComposer ); - - Any aData; - m_xInteractiveHandler = handler->second; - InteractiveSelectionResult eResult = - handler->second->onInteractivePropertySelection( _rName, _bPrimary, aData, - m_pUIRequestComposer->getUIForPropertyHandler( handler->second ) ); - - switch ( eResult ) - { - case InteractiveSelectionResult_Cancelled: - case InteractiveSelectionResult_Success: - // okay, nothing to do - break; - case InteractiveSelectionResult_ObtainedValue: - handler->second->setPropertyValue( _rName, aData ); - break; - case InteractiveSelectionResult_Pending: - // also okay, we expect that the handler has disabled the UI as necessary - break; - default: - OSL_FAIL( "OPropertyBrowserController::Clicked: unknown result value!" ); - break; - } - } - catch (Exception&) - { - DBG_UNHANDLED_EXCEPTION(); - } - m_xInteractiveHandler = NULL; - } - - //------------------------------------------------------------------------ - sal_Bool SAL_CALL OPropertyBrowserController::hasPropertyByName( const ::rtl::OUString& _rName ) throw (RuntimeException) - { - for ( OrderedPropertyMap::const_iterator search = m_aProperties.begin(); - search != m_aProperties.end(); - ++search - ) - if ( search->second.Name == _rName ) - return true; - return false; - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::Commit( const ::rtl::OUString& rName, const Any& _rValue ) - { - try - { - rtl::OUString sPlcHolder = String( PcrRes( RID_EMBED_IMAGE_PLACEHOLDER ) ); - bool bIsPlaceHolderValue = false; - - if ( rName.equals( PROPERTY_IMAGE_URL ) ) - { - // if the prop value is the PlaceHolder - // can ignore it - rtl::OUString sVal; - _rValue >>= sVal; - if ( sVal.equals( sPlcHolder ) ) - bIsPlaceHolderValue = true; - } - m_sCommittingProperty = rName; - - bool bIsActuatingProperty = impl_isActuatingProperty_nothrow( rName ); - - Any aOldValue; - if ( bIsActuatingProperty ) - aOldValue = impl_getPropertyValue_throw( rName ); - - // do we have a dedicated handler for this property, which we can delegate some tasks to? - PropertyHandlerRef handler = impl_getHandlerForProperty_throw( rName ); - - ////////////////////////////////////////////////////////////////////// - // set the value ( only if it's not a placeholder ) - if ( !bIsPlaceHolderValue ) - handler->setPropertyValue( rName, _rValue ); - - ////////////////////////////////////////////////////////////////////// - // re-retrieve the value - Any aNormalizedValue = handler->getPropertyValue( rName ); - - // care for any inter-property dependencies - if ( bIsActuatingProperty ) - impl_broadcastPropertyChange_nothrow( rName, aNormalizedValue, aOldValue, false ); - - // and display it again. This ensures proper formatting - getPropertyBox().SetPropertyValue( rName, aNormalizedValue, false ); - } - catch(PropertyVetoException& eVetoException) - { - InfoBox(m_pView, eVetoException.Message).Execute(); - PropertyHandlerRef handler = impl_getHandlerForProperty_throw( rName ); - Any aNormalizedValue = handler->getPropertyValue( rName ); - getPropertyBox().SetPropertyValue( rName, aNormalizedValue, false ); - } - catch(Exception&) - { - OSL_FAIL("OPropertyBrowserController::Commit : caught an exception !"); - } - - m_sCommittingProperty = ::rtl::OUString(); - } - - //-------------------------------------------------------------------- - namespace - { - } - - //-------------------------------------------------------------------- - void OPropertyBrowserController::focusGained( const Reference< XPropertyControl >& _Control ) - { - m_aControlObservers.notifyEach( &XPropertyControlObserver::focusGained, _Control ); - } - - //-------------------------------------------------------------------- - void OPropertyBrowserController::valueChanged( const Reference< XPropertyControl >& _Control ) - { - m_aControlObservers.notifyEach( &XPropertyControlObserver::valueChanged, _Control ); - } - - //------------------------------------------------------------------------ - namespace - { - Reference< XPropertyHandler > lcl_createHandler( const ComponentContext& _rContext, const Any& _rFactoryDescriptor ) - { - Reference< XPropertyHandler > xHandler; - - ::rtl::OUString sServiceName; - Reference< XSingleServiceFactory > xServiceFac; - Reference< XSingleComponentFactory > xComponentFac; - - if ( _rFactoryDescriptor >>= sServiceName ) - _rContext.createComponent( sServiceName, xHandler ); - else if ( _rFactoryDescriptor >>= xServiceFac ) - xHandler = xHandler.query( xServiceFac->createInstance() ); - else if ( _rFactoryDescriptor >>= xComponentFac ) - xHandler = xHandler.query( xComponentFac->createInstanceWithContext( _rContext.getUNOContext() ) ); - OSL_ENSURE(xHandler.is(),"lcl_createHandler: Can not create handler"); - return xHandler; - } - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::getPropertyHandlers( const InterfaceArray& _rObjects, PropertyHandlerArray& _rHandlers ) - { - _rHandlers.resize( 0 ); - if ( _rObjects.empty() ) - return; - - // create a component context for the handlers, containing some information about where - // they live - Reference< XComponentContext > xHandlerContext( m_aContext.getUNOContext() ); - - // if our own creator did not pass a dialog parent window, use our own view for this - Reference< XWindow > xParentWindow( m_aContext.getContextValueByAsciiName( "DialogParentWindow" ), UNO_QUERY ); - if ( !xParentWindow.is() ) - { - ::cppu::ContextEntry_Init aHandlerContextInfo[] = - { - ::cppu::ContextEntry_Init( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "DialogParentWindow" ) ), makeAny( VCLUnoHelper::GetInterface( m_pView ) ) ) - }; - xHandlerContext = ::cppu::createComponentContext( - aHandlerContextInfo, SAL_N_ELEMENTS( aHandlerContextInfo ), - m_aContext.getUNOContext() ); - } - - Sequence< Any > aHandlerFactories; - if ( m_xModel.is() ) - aHandlerFactories = m_xModel->getHandlerFactories(); - - const Any* pHandlerFactory = aHandlerFactories.getConstArray(); - const Any* pHandlerFactoryEnd = aHandlerFactories.getConstArray() + aHandlerFactories.getLength(); - - while ( pHandlerFactory != pHandlerFactoryEnd ) - { - if ( _rObjects.size() == 1 ) - { // we're inspecting only one object -> one handler - Reference< XPropertyHandler > xHandler( lcl_createHandler( m_aContext, *pHandlerFactory ) ); - if ( xHandler.is() ) - { - xHandler->inspect( _rObjects[0] ); - _rHandlers.push_back( xHandler ); - } - } - else - { - // create a single handler for every single object - ::std::vector< Reference< XPropertyHandler > > aSingleHandlers( _rObjects.size() ); - ::std::vector< Reference< XPropertyHandler > >::iterator pHandler = aSingleHandlers.begin(); - - InterfaceArray::const_iterator pObject = _rObjects.begin(); - InterfaceArray::const_iterator pObjectEnd = _rObjects.end(); - - for ( ; pObject != pObjectEnd; ++pObject ) - { - *pHandler = lcl_createHandler( m_aContext, *pHandlerFactory ); - if ( pHandler->is() ) - { - (*pHandler)->inspect( *pObject ); - ++pHandler; - } - } - aSingleHandlers.resize( pHandler - aSingleHandlers.begin() ); - - // then create a handler which composes information out of those single handlers - if ( !aSingleHandlers.empty() ) - _rHandlers.push_back( new PropertyComposer( aSingleHandlers ) ); - } - - ++pHandlerFactory; - } - - // note that the handlers will not be used by our caller, if they indicate that there are no - // properties they feel responsible for - } - - //------------------------------------------------------------------------ - bool OPropertyBrowserController::impl_findObjectProperty_nothrow( const ::rtl::OUString& _rName, OrderedPropertyMap::const_iterator* _pProperty ) - { - OrderedPropertyMap::const_iterator search = m_aProperties.begin(); - for ( ; search != m_aProperties.end(); ++search ) - if ( search->second.Name == _rName ) - break; - if ( _pProperty ) - *_pProperty = search; - return ( search != m_aProperties.end() ); - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::rebuildPropertyUI( const ::rtl::OUString& _rPropertyName ) throw (RuntimeException) - { - ::osl::MutexGuard aGuard( m_aMutex ); - if ( !haveView() ) - throw RuntimeException(); - - OrderedPropertyMap::const_iterator propertyPos; - if ( !impl_findObjectProperty_nothrow( _rPropertyName, &propertyPos ) ) - return; - - OLineDescriptor aDescriptor; - try - { - describePropertyLine( propertyPos->second, aDescriptor ); - } - catch( const Exception& ) - { - OSL_FAIL( "OPropertyBrowserController::rebuildPropertyUI: caught an exception!" ); - } - - getPropertyBox().ChangeEntry( aDescriptor ); - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::enablePropertyUI( const ::rtl::OUString& _rPropertyName, sal_Bool _bEnable ) throw (RuntimeException) - { - ::osl::MutexGuard aGuard( m_aMutex ); - if ( !haveView() ) - throw RuntimeException(); - - if ( !impl_findObjectProperty_nothrow( _rPropertyName ) ) - return; - - getPropertyBox().EnablePropertyLine( _rPropertyName, _bEnable ); - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::enablePropertyUIElements( const ::rtl::OUString& _rPropertyName, sal_Int16 _nElements, sal_Bool _bEnable ) throw (RuntimeException) - { - ::osl::MutexGuard aGuard( m_aMutex ); - if ( !haveView() ) - throw RuntimeException(); - - if ( !impl_findObjectProperty_nothrow( _rPropertyName ) ) - return; - - getPropertyBox().EnablePropertyControls( _rPropertyName, _nElements, _bEnable ); - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::showPropertyUI( const ::rtl::OUString& _rPropertyName ) throw (RuntimeException) - { - ::osl::MutexGuard aGuard( m_aMutex ); - if ( !haveView() ) - throw RuntimeException(); - - // look up the property in our object properties - OrderedPropertyMap::const_iterator propertyPos; - if ( !impl_findObjectProperty_nothrow( _rPropertyName, &propertyPos ) ) - return; - - if ( getPropertyBox().GetPropertyPos( _rPropertyName ) != LISTBOX_ENTRY_NOTFOUND ) - { - rebuildPropertyUI( _rPropertyName ); - return; - } - - OLineDescriptor aDescriptor; - describePropertyLine( propertyPos->second, aDescriptor ); - - // look for the position to insert the property - - // side note: The methods GetPropertyPos and InsertEntry of the OPropertyEditor work - // only on the current page. This implies that it's impossible to use this method here - // to show property lines which are *not* on the current page. - // This is sufficient for now, but should be changed in the future. - - // by definition, the properties in m_aProperties are in the order in which they appear in the UI - // So all we need is a predecessor of pProperty in m_aProperties - sal_uInt16 nUIPos = LISTBOX_ENTRY_NOTFOUND; - do - { - if ( propertyPos != m_aProperties.begin() ) - --propertyPos; - nUIPos = getPropertyBox().GetPropertyPos( propertyPos->second.Name ); - } - while ( ( nUIPos == LISTBOX_ENTRY_NOTFOUND ) && ( propertyPos != m_aProperties.begin() ) ); - - if ( nUIPos == LISTBOX_ENTRY_NOTFOUND ) - // insert at the very top - nUIPos = 0; - else - // insert right after the predecessor we found - ++nUIPos; - - getPropertyBox().InsertEntry( - aDescriptor, impl_getPageIdForCategory_nothrow( aDescriptor.Category ), nUIPos ); - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::hidePropertyUI( const ::rtl::OUString& _rPropertyName ) throw (RuntimeException) - { - ::osl::MutexGuard aGuard( m_aMutex ); - if ( !haveView() ) - throw RuntimeException(); - - if ( !impl_findObjectProperty_nothrow( _rPropertyName ) ) - return; - - getPropertyBox().RemoveEntry( _rPropertyName ); - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::showCategory( const ::rtl::OUString& _rCategory, sal_Bool _bShow ) throw (RuntimeException) - { - ::osl::MutexGuard aGuard( m_aMutex ); - if ( !haveView() ) - throw RuntimeException(); - - sal_uInt16 nPageId = impl_getPageIdForCategory_nothrow( _rCategory ); - OSL_ENSURE( nPageId != (sal_uInt16)-1, "OPropertyBrowserController::showCategory: invalid category!" ); - - getPropertyBox().ShowPropertyPage( nPageId, _bShow ); - } - - //------------------------------------------------------------------------ - Reference< XPropertyControl > SAL_CALL OPropertyBrowserController::getPropertyControl( const ::rtl::OUString& _rPropertyName ) throw (RuntimeException) - { - ::osl::MutexGuard aGuard( m_aMutex ); - if ( !haveView() ) - throw RuntimeException(); - - Reference< XPropertyControl > xControl( getPropertyBox().GetPropertyControl( _rPropertyName ) ); - return xControl; - } - - //-------------------------------------------------------------------- - void SAL_CALL OPropertyBrowserController::registerControlObserver( const Reference< XPropertyControlObserver >& _Observer ) throw (RuntimeException) - { - m_aControlObservers.addInterface( _Observer ); - } - - //-------------------------------------------------------------------- - void SAL_CALL OPropertyBrowserController::revokeControlObserver( const Reference< XPropertyControlObserver >& _Observer ) throw (RuntimeException) - { - m_aControlObservers.removeInterface( _Observer ); - } - - //------------------------------------------------------------------------ - void SAL_CALL OPropertyBrowserController::setHelpSectionText( const ::rtl::OUString& _rHelpText ) throw (NoSupportException, RuntimeException) - { - SolarMutexGuard aSolarGuard; - ::osl::MutexGuard aGuard( m_aMutex ); - - if ( !haveView() ) - throw DisposedException(); - - if ( !getPropertyBox().HasHelpSection() ) - throw NoSupportException(); - - getPropertyBox().SetHelpText( _rHelpText ); - } - - //------------------------------------------------------------------------ - void OPropertyBrowserController::impl_broadcastPropertyChange_nothrow( const ::rtl::OUString& _rPropertyName, const Any& _rNewValue, const Any& _rOldValue, bool _bFirstTimeInit ) const - { - // are there one or more handlers which are interested in the actuation? - ::std::pair< PropertyHandlerMultiRepository::const_iterator, PropertyHandlerMultiRepository::const_iterator > aInterestedHandlers = - m_aDependencyHandlers.equal_range( _rPropertyName ); - if ( aInterestedHandlers.first == aInterestedHandlers.second ) - // none of our handlers is interested in this - return; - - ComposedUIAutoFireGuard aAutoFireGuard( *m_pUIRequestComposer ); - try - { - // collect the responses from all interested handlers - PropertyHandlerMultiRepository::const_iterator handler = aInterestedHandlers.first; - while ( handler != aInterestedHandlers.second ) - { - handler->second->actuatingPropertyChanged( _rPropertyName, _rNewValue, _rOldValue, - m_pUIRequestComposer->getUIForPropertyHandler( handler->second ), - _bFirstTimeInit ); - ++handler; - } - } - catch( const Exception& ) - { - DBG_UNHANDLED_EXCEPTION(); - } - } - -//............................................................................ -} // namespace pcr -//............................................................................ - - -/* vim:set shiftwidth=4 softtabstop=4 expandtab: */ |