Files
libreoffice/toolkit/source/controls/unocontrol.cxx

1055 lines
34 KiB
C++
Raw Normal View History

2000-09-18 16:07:07 +00:00
/*************************************************************************
*
* $RCSfile: unocontrol.cxx,v $
*
* $Revision: 1.14 $
2000-09-18 16:07:07 +00:00
*
* last change: $Author: fs $ $Date: 2001-08-28 15:23:00 $
2000-09-18 16:07:07 +00:00
*
* The Contents of this file are made available subject to the terms of
* either of the following licenses
*
* - GNU Lesser General Public License Version 2.1
* - Sun Industry Standards Source License Version 1.1
*
* Sun Microsystems Inc., October, 2000
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2000 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1, as published by the Free Software Foundation.
*
* This library 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 for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
*
*
* Sun Industry Standards Source License Version 1.1
* =================================================
* The contents of this file are subject to the Sun Industry Standards
* Source License Version 1.1 (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.openoffice.org/license.html.
*
* Software provided under this License is provided on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
* WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
* MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
* See the License for the specific provisions governing your rights and
* obligations concerning the Software.
*
* The Initial Developer of the Original Code is: Sun Microsystems, Inc.
*
* Copyright: 2000 by Sun Microsystems, Inc.
*
* All Rights Reserved.
*
* Contributor(s): _______________________________________
*
*
************************************************************************/
#ifndef _COM_SUN_STAR_AWT_XVCLWINDOWPEER_HPP_
#include <com/sun/star/awt/XVclWindowPeer.hpp>
#endif
#ifndef _COM_SUN_STAR_AWT_XCONTROLCONTAINER_HPP_
#include <com/sun/star/awt/XControlContainer.hpp>
#endif
#ifndef _COM_SUN_STAR_AWT_WINDOWATTRIBUTE_HPP_
#include <com/sun/star/awt/WindowAttribute.hpp>
#endif
#ifndef _COM_SUN_STAR_AWT_VCLWINDOWPEERATTRIBUTE_HPP_
#include <com/sun/star/awt/VclWindowPeerAttribute.hpp>
#endif
#ifndef _COM_SUN_STAR_LAN_XMULTISERVICEFACTORY_HPP_
#include <com/sun/star/lang/XMultiServiceFactory.hpp>
#endif
#ifndef _COM_SUN_STAR_BEANS_PROPERTYVALUE_HPP_
#include <com/sun/star/beans/PropertyValue.hpp>
#endif
2000-09-18 16:07:07 +00:00
#ifndef _TOOLKIT_CONTROLS_UNOCONTROL_HXX_
2000-09-18 16:07:07 +00:00
#include <toolkit/controls/unocontrol.hxx>
#endif
#ifndef _TOOLKIT_HELPER_VCLUNOHELPER_HXX_
2000-09-18 16:07:07 +00:00
#include <toolkit/helper/vclunohelper.hxx>
#endif
#ifndef _CPPUHELPER_TYPEPROVIDER_HXX_
2000-09-18 16:07:07 +00:00
#include <cppuhelper/typeprovider.hxx>
#endif
#ifndef _RTL_MEMORY_H_
2000-09-18 16:07:07 +00:00
#include <rtl/memory.h>
#endif
#ifndef _RTL_UUID_H_
2000-09-18 16:07:07 +00:00
#include <rtl/uuid.h>
#endif
2000-09-18 16:07:07 +00:00
#ifndef _VOS_MUTEX_HXX_
2001-02-22 17:22:48 +00:00
#include <vos/mutex.hxx>
#endif
#ifndef _STRING_HXX
2000-09-18 16:07:07 +00:00
#include <tools/string.hxx>
#endif
#ifndef _TOOLS_TABLE_HXX
2000-09-18 16:07:07 +00:00
#include <tools/table.hxx>
#endif
#ifndef _DATE_HXX
2000-09-18 16:07:07 +00:00
#include <tools/date.hxx>
#endif
#ifndef _TOOLS_TIME_HXX
2000-09-18 16:07:07 +00:00
#include <tools/time.hxx>
#endif
#ifndef _URLOBJ_HXX
2000-09-18 16:07:07 +00:00
#include <tools/urlobj.hxx>
#endif
#ifndef _TOOLS_DEBUG_HXX
2000-09-18 16:07:07 +00:00
#include <tools/debug.hxx>
#endif
2000-09-18 16:07:07 +00:00
#ifndef _SV_SVAPP_HXX
2000-09-18 16:07:07 +00:00
#include <vcl/svapp.hxx>
#endif
#ifndef _SV_WRKWIN_HXX
2000-09-18 16:07:07 +00:00
#include <vcl/wrkwin.hxx>
#endif
#ifndef _COMPHELPER_STLTYPES_HXX_
#include <comphelper/stl_types.hxx>
#endif
2000-09-18 16:07:07 +00:00
#ifndef _TOOLKIT_HELPER_PROPERTY_HXX_
2000-09-18 16:07:07 +00:00
#include <toolkit/helper/property.hxx>
#endif
#ifndef _TOOLKIT_HELPER_SERVICENAMES_HXX_
2000-09-18 16:07:07 +00:00
#include <toolkit/helper/servicenames.hxx>
#endif
2000-09-18 16:07:07 +00:00
#ifndef _SV_SVAPP_HXX
#include <vcl/svapp.hxx>
#endif
#ifndef _VOS_MUTEX_HXX_
#include <vos/mutex.hxx>
#endif
using namespace ::com::sun::star;
using namespace ::com::sun::star::uno;
using namespace ::com::sun::star::awt;
using namespace ::com::sun::star::beans;
using namespace ::com::sun::star::lang;
2000-09-18 16:07:07 +00:00
WorkWindow* lcl_GetDefaultWindow()
{
static WorkWindow* pW = NULL;
if ( !pW )
pW = new WorkWindow( NULL, 0 );
return pW;
}
static Sequence< ::rtl::OUString> lcl_ImplGetPropertyNames( const Reference< XMultiPropertySet > & rxModel )
2000-09-18 16:07:07 +00:00
{
Sequence< ::rtl::OUString> aNames;
Reference< XPropertySetInfo > xPSInf = rxModel->getPropertySetInfo();
2000-09-18 16:07:07 +00:00
DBG_ASSERT( xPSInf.is(), "UpdateFromModel: No PropertySetInfo!" );
if ( xPSInf.is() )
{
Sequence< Property> aProps = xPSInf->getProperties();
2000-09-18 16:07:07 +00:00
sal_Int32 nLen = aProps.getLength();
aNames = Sequence< ::rtl::OUString>( nLen );
2000-09-18 16:07:07 +00:00
::rtl::OUString* pNames = aNames.getArray() + nLen - 1;
const Property* pProps = aProps.getConstArray() + nLen - 1;
2000-09-18 16:07:07 +00:00
for ( sal_uInt32 n = nLen; n; --n, --pProps, --pNames)
*pNames = pProps->Name;
}
return aNames;
}
// ----------------------------------------------------
// class UnoControl
// ----------------------------------------------------
UnoControl::UnoControl()
: maDisposeListeners( *this )
, maWindowListeners( *this )
, maFocusListeners( *this )
, maKeyListeners( *this )
, maMouseListeners( *this )
, maMouseMotionListeners( *this )
, maPaintListeners( *this )
{
mbUpdatingModel = sal_False;
mbDisposePeer = sal_True;
mbRefeshingPeer = sal_False;
2001-04-11 14:09:19 +00:00
mbCreatingPeer = sal_False;
2001-02-05 14:25:14 +00:00
mbCreatingCompatiblePeer = sal_False;
2000-09-18 16:07:07 +00:00
mbDesignMode = sal_False;
}
UnoControl::~UnoControl()
{
}
::rtl::OUString UnoControl::GetComponentServiceName()
{
return ::rtl::OUString();
}
Reference< XWindowPeer > UnoControl::ImplGetCompatiblePeer( sal_Bool bAcceptExistingPeer )
2000-09-18 16:07:07 +00:00
{
2001-02-05 14:25:14 +00:00
DBG_ASSERT( !mbCreatingCompatiblePeer, "ImplGetCompatiblePeer - rekursive?" );
mbCreatingCompatiblePeer = sal_True;
Reference< XWindowPeer > xP;
2000-09-18 16:07:07 +00:00
if ( bAcceptExistingPeer )
xP = mxPeer;
if ( !xP.is() )
{
// Peer unsichtbar erzeugen...
sal_Bool bVis = maComponentInfos.bVisible;
if( bVis )
maComponentInfos.bVisible = sal_False;
Reference< XWindowPeer > xCurrentPeer = mxPeer;
2000-09-18 16:07:07 +00:00
mxPeer = NULL;
// Ueber queryInterface, wegen Aggregation...
Any aAny = OWeakAggObject::queryInterface( ::getCppuType((const Reference< XControl>*)0) );
Reference< XControl > xMe;
2000-09-18 16:07:07 +00:00
aAny >>= xMe;
2001-02-22 17:22:48 +00:00
WorkWindow* pWW;
{
osl::Guard< vos::IMutex > aGuard( Application::GetSolarMutex() );
pWW = lcl_GetDefaultWindow();
}
xMe->createPeer( Reference< XToolkit >(), pWW->GetComponentInterface( sal_True ) );
2000-09-18 16:07:07 +00:00
xP = mxPeer;
mxPeer = xCurrentPeer;
if( bVis )
maComponentInfos.bVisible = sal_True;
}
2001-02-05 14:25:14 +00:00
mbCreatingCompatiblePeer = sal_False;
2000-09-18 16:07:07 +00:00
return xP;
}
void UnoControl::ImplSetPeerProperty( const ::rtl::OUString& rPropName, const Any& rVal )
2000-09-18 16:07:07 +00:00
{
Reference< XVclWindowPeer > xW( mxPeer, UNO_QUERY );
if (xW.is())
// since a change made in propertiesChange, we can't be sure that this is called with an valid mxPeer,
// this assumption may be false in some (seldom) multi-threading scenarios (cause propertiesChange
// releases our mutex before calling here in)
// That's why this additional check
xW->setProperty( rPropName, rVal );
2000-09-18 16:07:07 +00:00
}
void UnoControl::PrepareWindowDescriptor( WindowDescriptor& rDesc )
2000-09-18 16:07:07 +00:00
{
}
Reference< XWindow > UnoControl::getParentPeer() const
2000-09-18 16:07:07 +00:00
{
Reference< XWindow > xPeer;
2000-09-18 16:07:07 +00:00
if( mxContext.is() )
{
Reference< XControl > xContComp( mxContext, UNO_QUERY );
2001-05-18 13:05:16 +00:00
if ( xContComp.is() )
2001-05-28 06:41:22 +00:00
{
Reference< XWindowPeer > xP = xContComp->getPeer();
2001-05-28 06:41:22 +00:00
if ( xP.is() )
xP->queryInterface( ::getCppuType((const Reference< XWindow >*)0) ) >>= xPeer;
2001-05-28 06:41:22 +00:00
}
2000-09-18 16:07:07 +00:00
}
return xPeer;
}
void UnoControl::updateFromModel()
{
// Alle standard Properties werden ausgelesen und in das Peer uebertragen
if( mxPeer.is() )
{
Reference< XMultiPropertySet > xPropSet( mxModel, UNO_QUERY );
Sequence< ::rtl::OUString> aNames = lcl_ImplGetPropertyNames( xPropSet );
2000-09-18 16:07:07 +00:00
xPropSet->firePropertiesChangeEvent( aNames, this );
}
}
// XInterface
Any UnoControl::queryAggregation( const Type & rType ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
Any aRet = ::cppu::queryInterface( rType,
SAL_STATIC_CAST( XControl*, this ),
SAL_STATIC_CAST( XWindow*, this ),
SAL_STATIC_CAST( XComponent*, SAL_STATIC_CAST( XControl*, this ) ),
SAL_STATIC_CAST( XView*, this ),
SAL_STATIC_CAST( XPropertiesChangeListener*, this ),
SAL_STATIC_CAST( XEventListener*, this ),
SAL_STATIC_CAST( XServiceInfo*, this ),
SAL_STATIC_CAST( XTypeProvider*, this ) );
2000-09-18 16:07:07 +00:00
return (aRet.hasValue() ? aRet : OWeakAggObject::queryAggregation( rType ));
}
// XTypeProvider
2000-09-18 16:07:07 +00:00
IMPL_XTYPEPROVIDER_START( UnoControl )
getCppuType( ( Reference< XControl>* ) NULL ),
getCppuType( ( Reference< XWindow>* ) NULL ),
getCppuType( ( Reference< XView>* ) NULL ),
getCppuType( ( Reference< XPropertiesChangeListener>* ) NULL ),
getCppuType( ( Reference< XServiceInfo>* ) NULL )
2000-09-18 16:07:07 +00:00
IMPL_XTYPEPROVIDER_END
void UnoControl::dispose( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
if( mxPeer.is() && mbDisposePeer )
{
mxPeer->dispose();
mxPeer = NULL;
}
EventObject aEvt;
aEvt.Source = (XAggregation*)(::cppu::OWeakAggObject*)this;
2000-09-18 16:07:07 +00:00
maDisposeListeners.disposeAndClear(aEvt);
maWindowListeners.disposeAndClear(aEvt);
maFocusListeners.disposeAndClear(aEvt);
maKeyListeners.disposeAndClear(aEvt);
maMouseListeners.disposeAndClear(aEvt);
maMouseMotionListeners.disposeAndClear(aEvt);
maPaintListeners.disposeAndClear(aEvt);
// Model wieder freigeben
setModel( Reference< XControlModel > () );
setContext( Reference< XInterface > () );
2000-09-18 16:07:07 +00:00
}
void UnoControl::addEventListener( const Reference< XEventListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
maDisposeListeners.addInterface( rxListener );
}
void UnoControl::removeEventListener( const Reference< XEventListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
maDisposeListeners.removeInterface( rxListener );
}
// XPropertiesChangeListener
void UnoControl::propertiesChange( const Sequence< PropertyChangeEvent >& rEvents ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::ClearableGuard< ::osl::Mutex > aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
// kommt von xModel
if( !IsUpdatingModel() && mxPeer.is() )
{
DECLARE_STL_VECTOR( PropertyValue, PropertyValueVector);
PropertyValueVector aPeerPropertiesToSet;
sal_Int32 nIndependentPos = 0;
// position where to insert the independent properties, dependent ones are inserted at the end of the vector
sal_Bool bNeedNewPeer = sal_False;
// some properties require a re-creation of the peer, 'cause they can't be changed on the fly
2000-09-18 16:07:07 +00:00
sal_Int32 nLen = rEvents.getLength();
for( sal_Int32 i = 0; i < nLen; i++ )
{
const PropertyChangeEvent& rEvt = rEvents.getConstArray()[i];
Reference< XControlModel > xModel( rEvt.Source, UNO_QUERY );
sal_Bool bOwnModel = (XControlModel*)xModel.get() == (XControlModel*)getModel().get();
if ( bOwnModel )
2000-09-18 16:07:07 +00:00
{
sal_uInt16 nPType = GetPropertyId( rEvt.PropertyName );
2001-04-11 14:09:19 +00:00
if ( nPType && mbDesignMode && mbDisposePeer && !mbRefeshingPeer && !mbCreatingPeer )
2000-09-18 16:07:07 +00:00
{
// Im Design-Mode koennen sich Props aendern, die eine
// Neuerzeugung der Peer erfordern...
if ( ( nPType == BASEPROPERTY_BORDER ) ||
( nPType == BASEPROPERTY_MULTILINE ) ||
( nPType == BASEPROPERTY_DROPDOWN ) ||
( nPType == BASEPROPERTY_HSCROLL ) ||
( nPType == BASEPROPERTY_VSCROLL ) ||
( nPType == BASEPROPERTY_SPIN ) ||
( nPType == BASEPROPERTY_ALIGN ) )
{
bNeedNewPeer = sal_True;
break;
}
}
if ( nPType && ( nLen > 1 ) && DoesDependOnOthers( nPType ) )
{
// Properties die von anderen abhaengen erst hinterher einstellen,
// weil sie von anderen Properties abhaengig sind, die aber erst spaeter
// eingestellt werden, z.B. VALUE nach VALUEMIN/MAX.
aPeerPropertiesToSet.push_back(PropertyValue(rEvt.PropertyName, 0, rEvt.NewValue, PropertyState_DIRECT_VALUE));
}
else
{
aPeerPropertiesToSet.insert(aPeerPropertiesToSet.begin() + nIndependentPos,
PropertyValue(rEvt.PropertyName, 0, rEvt.NewValue, PropertyState_DIRECT_VALUE));
++nIndependentPos;
2000-09-18 16:07:07 +00:00
}
}
}
Reference< XWindow > xParent = getParentPeer();
Reference< XControl > xThis( (XAggregation*)(::cppu::OWeakAggObject*)this, UNO_QUERY );
2001-04-11 14:09:19 +00:00
// call createPeer via a interface got from queryInterface, so the aggregating class can intercept it
DBG_ASSERT( !bNeedNewPeer || xParent.is(), "Need new peer, but don't have a parent!" );
aGuard.clear();
2001-04-11 14:09:19 +00:00
// clear the guard before creating a new peer - as usual, our peer implementations use the SolarMutex
// 82300 - 12/21/00 - FS
if (bNeedNewPeer && xParent.is())
{
NAMESPACE_VOS(OGuard) aVclGuard( Application::GetSolarMutex() );
// and now this is the final withdrawal:
// With 83561, I have no other idea than locking the SolarMutex here ....
// I really hate the fact that VCL is not theadsafe ....
// 01.03.2001 - FS
// Funktioniert beim Container nicht!
mxPeer->dispose();
mxPeer.clear();
mbRefeshingPeer = sal_True;
Reference< XWindowPeer > xP( xParent, UNO_QUERY );
xThis->createPeer( Reference< XToolkit > (), xP );
mbRefeshingPeer = sal_False;
2001-04-11 14:09:19 +00:00
aPeerPropertiesToSet.clear();
}
// setting peer properties may result in an attemp to acquire the solar mutex, 'cause the peers
// usually don't have an own mutex but use the SolarMutex instead.
// To prevent deadlocks resulting from this, we do this without our own mutex locked
// FS - 11/03/2000
for ( PropertyValueVectorIterator aLoop = aPeerPropertiesToSet.begin();
aLoop != aPeerPropertiesToSet.end();
++aLoop
)
2000-09-18 16:07:07 +00:00
{
ImplSetPeerProperty( aLoop->Name, aLoop->Value );
2000-09-18 16:07:07 +00:00
}
}
}
void UnoControl::disposing( const EventObject& rEvt ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
// kommt von xModel, bei "Multible Inheritance" nicht unterschiedliche
// Typen vergleichen.
if( mxModel == rEvt.Source )
{
mxModel = NULL;
// #62337# Ohne Model wollen wir nicht weiterleben
Reference< XControl > xThis = this;
2000-09-18 16:07:07 +00:00
xThis->dispose();
}
}
// XWindow
void UnoControl::setPosSize( sal_Int32 X, sal_Int32 Y, sal_Int32 Width, sal_Int32 Height, sal_Int16 Flags ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
Reference< XWindow > xWindow;
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
maComponentInfos.nX = X;
maComponentInfos.nY = Y;
maComponentInfos.nWidth = Width;
maComponentInfos.nHeight = Height;
maComponentInfos.nFlags = Flags;
2000-09-18 16:07:07 +00:00
xWindow = xWindow.query( mxPeer );
2000-09-18 16:07:07 +00:00
}
if( xWindow.is() )
xWindow->setPosSize( X, Y, Width, Height, Flags );
2000-09-18 16:07:07 +00:00
}
awt::Rectangle UnoControl::getPosSize( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
return awt::Rectangle( maComponentInfos.nX, maComponentInfos.nY, maComponentInfos.nWidth, maComponentInfos.nHeight);
2000-09-18 16:07:07 +00:00
}
void UnoControl::setVisible( sal_Bool bVisible ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
Reference< XWindow > xWindow;
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
// Visible status ist Sache der View
maComponentInfos.bVisible = bVisible;
xWindow = xWindow.query( mxPeer );
2000-09-18 16:07:07 +00:00
}
if ( xWindow.is() )
xWindow->setVisible( bVisible );
2000-09-18 16:07:07 +00:00
}
void UnoControl::setEnable( sal_Bool bEnable ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
Reference< XWindow > xWindow;
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
// Enable status ist Sache der View
maComponentInfos.bEnable = bEnable;
xWindow = xWindow.query( mxPeer );
2000-09-18 16:07:07 +00:00
}
if ( xWindow.is() )
xWindow->setEnable( bEnable );
2000-09-18 16:07:07 +00:00
}
void UnoControl::setFocus( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
Reference< XWindow > xWindow;
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
xWindow = xWindow.query( mxPeer );
2000-09-18 16:07:07 +00:00
}
if ( xWindow.is() )
xWindow->setFocus();
2000-09-18 16:07:07 +00:00
}
void UnoControl::addWindowListener( const Reference< XWindowListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
maWindowListeners.addInterface( rxListener );
if( mxPeer.is() && maWindowListeners.getLength() == 1 )
{
// erster Focus Listener, also am Peer anmelden
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->addWindowListener( &maWindowListeners );
}
}
void UnoControl::removeWindowListener( const Reference< XWindowListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
if( mxPeer.is() && maWindowListeners.getLength() == 1 )
{
// letzter Focus Listener, also am Peer abmelden
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->removeWindowListener( &maWindowListeners );
}
maWindowListeners.removeInterface( rxListener );
}
void UnoControl::addFocusListener( const Reference< XFocusListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
maFocusListeners.addInterface( rxListener );
if( mxPeer.is() && maFocusListeners.getLength() == 1 )
{
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->addFocusListener( &maFocusListeners );
}
}
void UnoControl::removeFocusListener( const Reference< XFocusListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
if( mxPeer.is() && maFocusListeners.getLength() == 1 )
{
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->removeFocusListener( &maFocusListeners );
}
maFocusListeners.removeInterface( rxListener );
}
void UnoControl::addKeyListener( const Reference< XKeyListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
maKeyListeners.addInterface( rxListener );
if( mxPeer.is() && maKeyListeners.getLength() == 1 )
{
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->addKeyListener( &maKeyListeners);
}
}
void UnoControl::removeKeyListener( const Reference< XKeyListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
if( mxPeer.is() && maKeyListeners.getLength() == 1 )
{
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->removeKeyListener( &maKeyListeners);
}
maKeyListeners.removeInterface( rxListener );
}
void UnoControl::addMouseListener( const Reference< XMouseListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
maMouseListeners.addInterface( rxListener );
if( mxPeer.is() && maMouseListeners.getLength() == 1 )
{
Reference< XWindow > xW( mxPeer, UNO_QUERY );;
2000-09-18 16:07:07 +00:00
xW->addMouseListener( &maMouseListeners);
}
}
void UnoControl::removeMouseListener( const Reference< XMouseListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
if( mxPeer.is() && maMouseListeners.getLength() == 1 )
{
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->removeMouseListener( &maMouseListeners);
}
maMouseListeners.removeInterface( rxListener );
}
void UnoControl::addMouseMotionListener( const Reference< XMouseMotionListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
maMouseMotionListeners.addInterface( rxListener );
if( mxPeer.is() && maMouseMotionListeners.getLength() == 1 )
{
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->addMouseMotionListener( &maMouseMotionListeners);
}
}
void UnoControl::removeMouseMotionListener( const Reference< XMouseMotionListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
if( mxPeer.is() && maMouseMotionListeners.getLength() == 1 )
{
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->removeMouseMotionListener( &maMouseMotionListeners);
}
maMouseMotionListeners.removeInterface( rxListener );
}
void UnoControl::addPaintListener( const Reference< XPaintListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
maPaintListeners.addInterface( rxListener );
if( mxPeer.is() && maPaintListeners.getLength() == 1 )
{
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->addPaintListener( &maPaintListeners );
}
}
void UnoControl::removePaintListener( const Reference< XPaintListener >& rxListener ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
if( mxPeer.is() && maPaintListeners.getLength() == 1 )
{
Reference< XWindow > xW( mxPeer, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xW->removePaintListener( &maPaintListeners );
}
maPaintListeners.removeInterface( rxListener );
}
// XView
sal_Bool UnoControl::setGraphics( const Reference< XGraphics >& rDevice ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
Reference< XView > xView;
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
mxGraphics = rDevice;
xView = xView.query( mxPeer );
2000-09-18 16:07:07 +00:00
}
return xView.is() ? xView->setGraphics( rDevice ) : sal_True;
2000-09-18 16:07:07 +00:00
}
Reference< XGraphics > UnoControl::getGraphics( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
return mxGraphics;
}
awt::Size UnoControl::getSize( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
return awt::Size( maComponentInfos.nWidth, maComponentInfos.nHeight );
2000-09-18 16:07:07 +00:00
}
void UnoControl::draw( sal_Int32 x, sal_Int32 y ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
Reference< XWindowPeer > xP = ImplGetCompatiblePeer( sal_True );
2000-09-18 16:07:07 +00:00
DBG_ASSERT( xP.is(), "Layout: No Peer!" );
if ( xP.is() )
{
Reference< XView > xV( xP, UNO_QUERY );
2000-09-18 16:07:07 +00:00
xV->draw( x, y );
if ( !mxPeer.is() || ( mxPeer != xP ) )
xP->dispose();
}
}
void UnoControl::setZoom( float fZoomX, float fZoomY ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
Reference< XView > xView;
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
maComponentInfos.nZoomX = fZoomX;
maComponentInfos.nZoomY = fZoomY;
xView = xView.query( mxPeer );
2000-09-18 16:07:07 +00:00
}
if ( xView.is() )
xView->setZoom( fZoomX, fZoomY );
2000-09-18 16:07:07 +00:00
}
// XControl
void UnoControl::setContext( const Reference< XInterface >& rxContext ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
mxContext = rxContext;
}
Reference< XInterface > UnoControl::getContext( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
return mxContext;
}
void UnoControl::createPeer( const Reference< XToolkit >& rxToolkit, const Reference< XWindowPeer >& rParentPeer ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::ClearableMutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
2001-05-15 11:49:57 +00:00
if ( !mxModel.is() )
{
RuntimeException aException;
2001-05-15 11:49:57 +00:00
aException.Message = ::rtl::OUString::createFromAscii( "createPeer: no model!" );
aException.Context = (XAggregation*)(::cppu::OWeakAggObject*)this;
2001-05-15 11:49:57 +00:00
throw( aException );
}
2000-09-18 16:07:07 +00:00
if( !mxPeer.is() )
{
2001-04-11 14:09:19 +00:00
mbCreatingPeer = sal_True;
WindowClass eType;
Reference< XToolkit > xToolkit = rxToolkit;
2000-09-18 16:07:07 +00:00
if( rParentPeer.is() && mxContext.is() )
{
// kein TopWindow
if ( !xToolkit.is() )
xToolkit = rParentPeer->getToolkit();
Any aAny = OWeakAggObject::queryInterface( ::getCppuType((const Reference< XControlContainer>*)0) );
Reference< XControlContainer > xC;
2000-09-18 16:07:07 +00:00
aAny >>= xC;
if( xC.is() )
// Es ist ein Container
eType = WindowClass_CONTAINER;
2000-09-18 16:07:07 +00:00
else
eType = WindowClass_SIMPLE;
2000-09-18 16:07:07 +00:00
}
else
{ // Nur richtig, wenn es sich um ein Top Window handelt
if( rParentPeer.is() )
{
if ( !xToolkit.is() )
xToolkit = rParentPeer->getToolkit();
eType = WindowClass_CONTAINER;
2000-09-18 16:07:07 +00:00
}
else
{
if ( !xToolkit.is() )
xToolkit = VCLUnoHelper::CreateToolkit();
eType = WindowClass_TOP;
2000-09-18 16:07:07 +00:00
}
}
WindowDescriptor aDescr;
2000-09-18 16:07:07 +00:00
aDescr.Type = eType;
aDescr.WindowServiceName = GetComponentServiceName();
aDescr.Parent = rParentPeer;
aDescr.Bounds = getPosSize();
aDescr.WindowAttributes = 0;
// Border
Reference< XPropertySet > xPSet( mxModel, UNO_QUERY );
Reference< XPropertySetInfo > xInfo = xPSet->getPropertySetInfo();
2000-09-18 16:07:07 +00:00
Any aVal;
2000-09-18 16:07:07 +00:00
::rtl::OUString aPropName = GetPropertyName( BASEPROPERTY_BORDER );
if ( xInfo->hasPropertyByName( aPropName ) )
{
aVal = xPSet->getPropertyValue( aPropName );
sal_Int16 n;
if ( aVal >>= n )
{
if ( n )
aDescr.WindowAttributes |= WindowAttribute::BORDER;
2000-09-18 16:07:07 +00:00
else
aDescr.WindowAttributes |= VclWindowPeerAttribute::NOBORDER;
2000-09-18 16:07:07 +00:00
}
}
2001-01-24 14:00:58 +00:00
// Moveable
aPropName = GetPropertyName( BASEPROPERTY_MOVEABLE );
if ( xInfo->hasPropertyByName( aPropName ) )
{
aVal = xPSet->getPropertyValue( aPropName );
sal_Bool b;
if ( ( aVal >>= b ) && b)
aDescr.WindowAttributes |= WindowAttribute::MOVEABLE;
2001-01-24 14:00:58 +00:00
}
// Closeable
aPropName = GetPropertyName( BASEPROPERTY_CLOSEABLE );
if ( xInfo->hasPropertyByName( aPropName ) )
{
aVal = xPSet->getPropertyValue( aPropName );
sal_Bool b;
if ( ( aVal >>= b ) && b)
aDescr.WindowAttributes |= WindowAttribute::CLOSEABLE;
2001-01-24 14:00:58 +00:00
}
2000-09-18 16:07:07 +00:00
// Dropdown
aPropName = GetPropertyName( BASEPROPERTY_DROPDOWN );
if ( xInfo->hasPropertyByName( aPropName ) )
{
aVal = xPSet->getPropertyValue( aPropName );
sal_Bool b;
if ( ( aVal >>= b ) && b)
aDescr.WindowAttributes |= VclWindowPeerAttribute::DROPDOWN;
2000-09-18 16:07:07 +00:00
}
// Spin
aPropName = GetPropertyName( BASEPROPERTY_SPIN );
if ( xInfo->hasPropertyByName( aPropName ) )
{
aVal = xPSet->getPropertyValue( aPropName );
sal_Bool b;
if ( ( aVal >>= b ) && b)
aDescr.WindowAttributes |= VclWindowPeerAttribute::SPIN;
2000-09-18 16:07:07 +00:00
}
// HScroll
aPropName = GetPropertyName( BASEPROPERTY_HSCROLL );
if ( xInfo->hasPropertyByName( aPropName ) )
{
aVal = xPSet->getPropertyValue( aPropName );
sal_Bool b;
if ( ( aVal >>= b ) && b)
aDescr.WindowAttributes |= VclWindowPeerAttribute::HSCROLL;
2000-09-18 16:07:07 +00:00
}
// VScroll
aPropName = GetPropertyName( BASEPROPERTY_VSCROLL );
if ( xInfo->hasPropertyByName( aPropName ) )
{
aVal = xPSet->getPropertyValue( aPropName );
sal_Bool b;
if ( ( aVal >>= b ) && b)
aDescr.WindowAttributes |= VclWindowPeerAttribute::VSCROLL;
2000-09-18 16:07:07 +00:00
}
// Align
aPropName = GetPropertyName( BASEPROPERTY_ALIGN );
if ( xInfo->hasPropertyByName( aPropName ) )
{
aVal = xPSet->getPropertyValue( aPropName );
sal_Int16 n;
if ( aVal >>= n )
{
if ( n == PROPERTY_ALIGN_LEFT )
aDescr.WindowAttributes |= VclWindowPeerAttribute::LEFT;
2000-09-18 16:07:07 +00:00
else if ( n == PROPERTY_ALIGN_CENTER )
aDescr.WindowAttributes |= VclWindowPeerAttribute::CENTER;
2000-09-18 16:07:07 +00:00
else
aDescr.WindowAttributes |= VclWindowPeerAttribute::RIGHT;
2000-09-18 16:07:07 +00:00
}
}
// Ableitungen die Moeglichkeit geben die Attribute zu manipulieren
PrepareWindowDescriptor(aDescr);
// create the peer
mxPeer = xToolkit->createWindow( aDescr );
// release the mutex guard (and work with copies of our members)
// this is necessary as our peer may lock the SolarMutex (actually, all currently known peers do), so calling
// into the peer with our own mutex locked may cause deadlocks
// (We _really_ need peers which do not use the SolarMutex. It's really pissing me off that from time to
// time deadlocks pop up because the low-level components like our peers use a mutex which ususally
// is locked at the top of the stack (it protects the global message looping). This is always dangerous, and
// can not always be solved by tampering with other mutexes.
// Unfortunately, the VCL used in the peers is not threadsafe, and by definition needs a locked SolarMutex.)
// 82300 - 12/21/00 - FS
UnoControlComponentInfos aComponentInfos(maComponentInfos);
sal_Bool bDesignMode(mbDesignMode);
Reference< XGraphics > xGraphics( mxGraphics );
Reference< XView > xV(mxPeer, UNO_QUERY);
Reference< XWindow > xW(mxPeer, UNO_QUERY);
2000-09-18 16:07:07 +00:00
aGuard.clear();
2000-09-18 16:07:07 +00:00
// the updateFromModel is done without a locked mutex, too.
// The reason is that the only thing this method does is firing property changes, and this in general has
// to be done without locked mutexes (as every notification to external listeners).
// 82300 - 12/21/00 - FS
2000-09-18 16:07:07 +00:00
updateFromModel();
xV->setZoom( aComponentInfos.nZoomX, aComponentInfos.nZoomY );
2000-09-18 16:07:07 +00:00
if( aComponentInfos.bVisible && !bDesignMode )
2000-09-18 16:07:07 +00:00
// Erst nach dem setzen der Daten anzeigen
xW->setVisible( aComponentInfos.bVisible );
2000-09-18 16:07:07 +00:00
if( !aComponentInfos.bEnable )
xW->setEnable( aComponentInfos.bEnable );
2000-09-18 16:07:07 +00:00
if ( maWindowListeners.getLength() )
xW->addWindowListener( &maWindowListeners );
if ( maFocusListeners.getLength() )
xW->addFocusListener( &maFocusListeners );
if ( maKeyListeners.getLength() )
xW->addKeyListener( &maKeyListeners );
if ( maMouseListeners.getLength() )
xW->addMouseListener( &maMouseListeners );
if ( maMouseMotionListeners.getLength() )
xW->addMouseMotionListener( &maMouseMotionListeners );
if ( maPaintListeners.getLength() )
xW->addPaintListener( &maPaintListeners );
xV->setGraphics( xGraphics );
2001-04-11 14:09:19 +00:00
mbCreatingPeer = sal_False;
2000-09-18 16:07:07 +00:00
}
}
Reference< XWindowPeer > UnoControl::getPeer( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
return mxPeer;
}
sal_Bool UnoControl::setModel( const Reference< XControlModel >& rxModel ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
Reference< XMultiPropertySet > xPropSet( mxModel, UNO_QUERY );
2000-09-18 16:07:07 +00:00
if( xPropSet.is() )
xPropSet->removePropertiesChangeListener( this );
mxModel = rxModel;
if( mxModel.is() )
{
xPropSet = Reference< XMultiPropertySet > ( mxModel, UNO_QUERY );
2000-09-18 16:07:07 +00:00
if( xPropSet.is() )
{
Sequence< ::rtl::OUString> aNames = lcl_ImplGetPropertyNames( xPropSet );
2000-09-18 16:07:07 +00:00
xPropSet->addPropertiesChangeListener( aNames, this );
}
}
return mxModel.is();
}
Reference< XControlModel > UnoControl::getModel( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
return mxModel;
}
Reference< XView > UnoControl::getView( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
return static_cast< XView* >( this );
2000-09-18 16:07:07 +00:00
}
void UnoControl::setDesignMode( sal_Bool bOn ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
Reference< XWindow > xWindow;
{
::osl::MutexGuard aGuard( GetMutex() );
mbDesignMode = bOn;
xWindow = xWindow.query( mxPeer );
}
if ( xWindow.is() )
xWindow->setVisible( !bOn );
2000-09-18 16:07:07 +00:00
}
sal_Bool UnoControl::isDesignMode( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
return mbDesignMode;
}
sal_Bool UnoControl::isTransparent( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
return sal_False;
}
// XServiceInfo
::rtl::OUString UnoControl::getImplementationName( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
DBG_ERROR( "This method should be overloaded!" );
return ::rtl::OUString();
}
sal_Bool UnoControl::supportsService( const ::rtl::OUString& rServiceName ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
::osl::MutexGuard aGuard( GetMutex() );
2000-09-18 16:07:07 +00:00
Sequence< ::rtl::OUString > aSNL = getSupportedServiceNames();
const ::rtl::OUString* pArray = aSNL.getConstArray();
const ::rtl::OUString* pArrayEnd = aSNL.getConstArray();
for (; pArray != pArrayEnd; ++pArray )
if( *pArray == rServiceName )
break;
return pArray != pArrayEnd;
2000-09-18 16:07:07 +00:00
}
Sequence< ::rtl::OUString > UnoControl::getSupportedServiceNames( ) throw(RuntimeException)
2000-09-18 16:07:07 +00:00
{
DBG_ERROR( "This method should be overloaded!" );
return Sequence< ::rtl::OUString >();
2000-09-18 16:07:07 +00:00
}