1 /*************************************************************************
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5 * Copyright 2008 by Sun Microsystems, Inc.
7 * OpenOffice.org - a multi-platform office productivity suite
9 * $RCSfile: unocontrol.cxx,v $
10 * $Revision: 1.54.42.1 $
12 * This file is part of OpenOffice.org.
14 * OpenOffice.org is free software: you can redistribute it and/or modify
15 * it under the terms of the GNU Lesser General Public License version 3
16 * only, as published by the Free Software Foundation.
18 * OpenOffice.org is distributed in the hope that it will be useful,
19 * but WITHOUT ANY WARRANTY; without even the implied warranty of
20 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 * GNU Lesser General Public License version 3 for more details
22 * (a copy is included in the LICENSE file that accompanied this code).
24 * You should have received a copy of the GNU Lesser General Public License
25 * version 3 along with OpenOffice.org. If not, see
26 * <http://www.openoffice.org/license.html>
27 * for a copy of the LGPLv3 License.
29 ************************************************************************/
31 // MARKER(update_precomp.py): autogen include statement, do not remove
32 #include "precompiled_toolkit.hxx"
33 #include <com/sun/star/awt/XControlContainer.hpp>
34 #include <com/sun/star/awt/WindowAttribute.hpp>
35 #include <com/sun/star/awt/VclWindowPeerAttribute.hpp>
36 #include <com/sun/star/awt/PosSize.hpp>
37 #ifndef _COM_SUN_STAR_LAN_XMULTISERVICEFACTORY_HPP_
38 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
40 #include <com/sun/star/beans/PropertyValue.hpp>
41 #include <com/sun/star/resource/XStringResourceResolver.hpp>
42 #include <toolkit/controls/unocontrol.hxx>
43 #include <toolkit/helper/vclunohelper.hxx>
44 #include <cppuhelper/typeprovider.hxx>
45 #include <rtl/memory.h>
47 #include <vos/mutex.hxx>
48 #include <tools/string.hxx>
49 #include <tools/table.hxx>
50 #include <tools/date.hxx>
51 #include <tools/time.hxx>
52 #include <tools/urlobj.hxx>
53 #include <tools/debug.hxx>
54 #include <tools/diagnose_ex.h>
55 #include <vcl/svapp.hxx>
56 #include <vcl/wrkwin.hxx>
57 #include <comphelper/stl_types.hxx>
58 #include <toolkit/helper/property.hxx>
59 #include <toolkit/helper/servicenames.hxx>
60 #include <toolkit/helper/vclunohelper.hxx>
61 #include <toolkit/awt/vclxwindow.hxx>
62 #include <vcl/svapp.hxx>
63 #include <vos/mutex.hxx>
64 #include <toolkit/controls/accessiblecontrolcontext.hxx>
65 #include <comphelper/container.hxx>
70 using namespace ::com::sun::star
;
71 using namespace ::com::sun::star::uno
;
72 using namespace ::com::sun::star::awt
;
73 using namespace ::com::sun::star::beans
;
74 using namespace ::com::sun::star::lang
;
75 using namespace ::com::sun::star::util
;
77 using ::com::sun::star::accessibility::XAccessibleContext
;
78 using ::com::sun::star::accessibility::XAccessible
;
80 struct LanguageDependentProp
82 const char* pPropName
;
83 sal_Int32 nPropNameLength
;
86 static const LanguageDependentProp aLanguageDependentProp
[] =
92 { "CurrencySymbol", 14 },
93 { "StringItemList", 14 },
97 WorkWindow
* lcl_GetDefaultWindow()
99 static WorkWindow
* pW
= NULL
;
102 pW
= new WorkWindow( NULL
, 0 );
103 pW
->EnableChildTransparentMode();
108 static Sequence
< ::rtl::OUString
> lcl_ImplGetPropertyNames( const Reference
< XMultiPropertySet
> & rxModel
)
110 Sequence
< ::rtl::OUString
> aNames
;
111 Reference
< XPropertySetInfo
> xPSInf
= rxModel
->getPropertySetInfo();
112 DBG_ASSERT( xPSInf
.is(), "UpdateFromModel: No PropertySetInfo!" );
115 Sequence
< Property
> aProps
= xPSInf
->getProperties();
116 sal_Int32 nLen
= aProps
.getLength();
117 aNames
= Sequence
< ::rtl::OUString
>( nLen
);
118 ::rtl::OUString
* pNames
= aNames
.getArray();
119 const Property
* pProps
= aProps
.getConstArray();
120 for ( sal_Int32 n
= 0; n
< nLen
; ++n
, ++pProps
, ++pNames
)
121 *pNames
= pProps
->Name
;
126 // ====================================================
127 class VclListenerLock
130 VCLXWindow
* m_pLockWindow
;
133 inline VclListenerLock( VCLXWindow
* _pLockWindow
)
134 :m_pLockWindow( _pLockWindow
)
137 m_pLockWindow
->suspendVclEventListening( );
139 inline ~VclListenerLock( )
142 m_pLockWindow
->resumeVclEventListening( );
146 VclListenerLock(); // never implemented
147 VclListenerLock( const VclListenerLock
& ); // never implemented
148 VclListenerLock
& operator=( const VclListenerLock
& ); // never implemented
151 typedef ::std::map
< ::rtl::OUString
, sal_Int32
> MapString2Int
;
152 struct UnoControl_Data
154 MapString2Int aSuspendedPropertyNotifications
;
155 /// true if and only if our model has a property ResourceResolver
156 bool bLocalizationSupport
;
159 :aSuspendedPropertyNotifications()
160 ,bLocalizationSupport( false )
165 // ----------------------------------------------------
167 // ----------------------------------------------------
168 DBG_NAME( UnoControl
)
169 UnoControl::UnoControl()
170 : maDisposeListeners( *this )
171 , maWindowListeners( *this )
172 , maFocusListeners( *this )
173 , maKeyListeners( *this )
174 , maMouseListeners( *this )
175 , maMouseMotionListeners( *this )
176 , maPaintListeners( *this )
177 , maModeChangeListeners( GetMutex() )
178 , mpData( new UnoControl_Data
)
180 DBG_CTOR( UnoControl
, NULL
);
181 mbDisposePeer
= sal_True
;
182 mbRefeshingPeer
= sal_False
;
183 mbCreatingPeer
= sal_False
;
184 mbCreatingCompatiblePeer
= sal_False
;
185 mbDesignMode
= sal_False
;
188 UnoControl::~UnoControl()
191 DBG_DTOR( UnoControl
, NULL
);
194 ::rtl::OUString
UnoControl::GetComponentServiceName()
196 return ::rtl::OUString();
199 Reference
< XWindowPeer
> UnoControl::ImplGetCompatiblePeer( sal_Bool bAcceptExistingPeer
)
201 DBG_ASSERT( !mbCreatingCompatiblePeer
, "ImplGetCompatiblePeer - rekursive?" );
203 mbCreatingCompatiblePeer
= sal_True
;
205 Reference
< XWindowPeer
> xCompatiblePeer
;
207 if ( bAcceptExistingPeer
)
208 xCompatiblePeer
= getPeer();
210 if ( !xCompatiblePeer
.is() )
212 // Peer unsichtbar erzeugen...
213 sal_Bool bVis
= maComponentInfos
.bVisible
;
215 maComponentInfos
.bVisible
= sal_False
;
217 Reference
< XWindowPeer
> xCurrentPeer
= getPeer();
220 // queryInterface ourself, to allow aggregation
221 Reference
< XControl
> xMe
;
222 OWeakAggObject::queryInterface( ::getCppuType( &xMe
) ) >>= xMe
;
226 osl::Guard
< vos::IMutex
> aGuard( Application::GetSolarMutex() );
227 pWW
= lcl_GetDefaultWindow();
231 xMe
->createPeer( NULL
, pWW
->GetComponentInterface( sal_True
) );
233 catch( const Exception
& )
235 mbCreatingCompatiblePeer
= sal_False
;
238 xCompatiblePeer
= getPeer();
239 setPeer( xCurrentPeer
);
241 if ( xCompatiblePeer
.is() && mxGraphics
.is() )
243 Reference
< XView
> xPeerView( xCompatiblePeer
, UNO_QUERY
);
244 if ( xPeerView
.is() )
245 xPeerView
->setGraphics( mxGraphics
);
249 maComponentInfos
.bVisible
= sal_True
;
252 mbCreatingCompatiblePeer
= sal_False
;
254 return xCompatiblePeer
;
257 bool UnoControl::ImplCheckLocalize( ::rtl::OUString
& _rPossiblyLocalizable
)
259 if ( !mpData
->bLocalizationSupport
260 || ( _rPossiblyLocalizable
.getLength() == 0 )
261 || ( _rPossiblyLocalizable
[0] != '&' )
262 // TODO: make this reasonable. At the moment, everything which by accident starts with a & is considered
263 // localizable, which is probably wrong.
269 Reference
< XPropertySet
> xPropSet( mxModel
, UNO_QUERY_THROW
);
270 Reference
< resource::XStringResourceResolver
> xStringResourceResolver(
271 xPropSet
->getPropertyValue( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "ResourceResolver" ) ) ),
274 if ( xStringResourceResolver
.is() )
276 ::rtl::OUString
aLocalizationKey( _rPossiblyLocalizable
.copy( 1 ) );
277 _rPossiblyLocalizable
= xStringResourceResolver
->resolveString( aLocalizationKey
);
281 catch( const Exception
& )
283 DBG_UNHANDLED_EXCEPTION();
288 void UnoControl::ImplSetPeerProperty( const ::rtl::OUString
& rPropName
, const Any
& rVal
)
290 // since a change made in propertiesChange, we can't be sure that this is called with an valid getPeer(),
291 // this assumption may be false in some (seldom) multi-threading scenarios (cause propertiesChange
292 // releases our mutex before calling here in)
293 // That's why this additional check
295 if ( mxVclWindowPeer
.is() )
297 Any
aConvertedValue( rVal
);
299 if ( mpData
->bLocalizationSupport
)
301 // We now support a mapping for language dependent properties. This is the
302 // central method to implement it.
303 if (( rPropName
.equalsAsciiL( "Text", 4 )) ||
304 ( rPropName
.equalsAsciiL( "Label", 5 )) ||
305 ( rPropName
.equalsAsciiL( "Title", 5 )) ||
306 ( rPropName
.equalsAsciiL( "HelpText", 8 )) ||
307 ( rPropName
.equalsAsciiL( "CurrencySymbol", 14 )) ||
308 ( rPropName
.equalsAsciiL( "StringItemList", 14 )) )
310 ::rtl::OUString aValue
;
311 uno::Sequence
< rtl::OUString
> aSeqValue
;
312 if ( aConvertedValue
>>= aValue
)
314 if ( ImplCheckLocalize( aValue
) )
315 aConvertedValue
<<= aValue
;
317 else if ( aConvertedValue
>>= aSeqValue
)
319 for ( sal_Int32 i
= 0; i
< aSeqValue
.getLength(); i
++ )
320 ImplCheckLocalize( aSeqValue
[i
] );
321 aConvertedValue
<<= aSeqValue
;
326 mxVclWindowPeer
->setProperty( rPropName
, aConvertedValue
);
330 void UnoControl::PrepareWindowDescriptor( WindowDescriptor
& )
334 Reference
< XWindow
> UnoControl::getParentPeer() const
336 Reference
< XWindow
> xPeer
;
339 Reference
< XControl
> xContComp( mxContext
, UNO_QUERY
);
340 if ( xContComp
.is() )
342 Reference
< XWindowPeer
> xP
= xContComp
->getPeer();
344 xP
->queryInterface( ::getCppuType((const Reference
< XWindow
>*)0) ) >>= xPeer
;
350 void UnoControl::updateFromModel()
352 // Alle standard Properties werden ausgelesen und in das Peer uebertragen
355 Reference
< XMultiPropertySet
> xPropSet( mxModel
, UNO_QUERY
);
358 Sequence
< ::rtl::OUString
> aNames
= lcl_ImplGetPropertyNames( xPropSet
);
359 xPropSet
->firePropertiesChangeEvent( aNames
, this );
366 IMPL_IMPLEMENTATION_ID( UnoControl
)
368 void UnoControl::disposeAccessibleContext()
370 Reference
< XComponent
> xContextComp( maAccessibleContext
.get(), UNO_QUERY
);
371 if ( xContextComp
.is() )
373 maAccessibleContext
= NULL
;
376 xContextComp
->removeEventListener( this );
377 xContextComp
->dispose();
379 catch( const Exception
& )
381 DBG_ERROR( "UnoControl::disposeAccessibleContext: could not dispose my AccessibleContext!" );
386 void UnoControl::dispose( ) throw(RuntimeException
)
388 Reference
< XWindowPeer
> xPeer
;
390 ::osl::MutexGuard
aGuard( GetMutex() );
402 // dispose and release our AccessibleContext
403 disposeAccessibleContext();
405 EventObject aDisposeEvent
;
406 aDisposeEvent
.Source
= static_cast< XAggregation
* >( this );
408 maDisposeListeners
.disposeAndClear( aDisposeEvent
);
409 maWindowListeners
.disposeAndClear( aDisposeEvent
);
410 maFocusListeners
.disposeAndClear( aDisposeEvent
);
411 maKeyListeners
.disposeAndClear( aDisposeEvent
);
412 maMouseListeners
.disposeAndClear( aDisposeEvent
);
413 maMouseMotionListeners
.disposeAndClear( aDisposeEvent
);
414 maPaintListeners
.disposeAndClear( aDisposeEvent
);
415 maModeChangeListeners
.disposeAndClear( aDisposeEvent
);
417 // Model wieder freigeben
418 setModel( Reference
< XControlModel
> () );
419 setContext( Reference
< XInterface
> () );
422 void UnoControl::addEventListener( const Reference
< XEventListener
>& rxListener
) throw(RuntimeException
)
424 ::osl::MutexGuard
aGuard( GetMutex() );
426 maDisposeListeners
.addInterface( rxListener
);
429 void UnoControl::removeEventListener( const Reference
< XEventListener
>& rxListener
) throw(RuntimeException
)
431 ::osl::MutexGuard
aGuard( GetMutex() );
433 maDisposeListeners
.removeInterface( rxListener
);
436 sal_Bool
UnoControl::requiresNewPeer( const ::rtl::OUString
& /* _rPropertyName */ ) const
441 // XPropertiesChangeListener
442 void UnoControl::propertiesChange( const Sequence
< PropertyChangeEvent
>& rEvents
) throw(RuntimeException
)
444 Sequence
< PropertyChangeEvent
> aEvents( rEvents
);
446 ::osl::MutexGuard
aGuard( GetMutex() );
448 if ( !mpData
->aSuspendedPropertyNotifications
.empty() )
450 // strip the property which we are currently updating (somewhere up the stack)
451 PropertyChangeEvent
* pEvents
= aEvents
.getArray();
452 PropertyChangeEvent
* pEventsEnd
= pEvents
+ aEvents
.getLength();
453 for ( ; pEvents
< pEventsEnd
; )
454 if ( mpData
->aSuspendedPropertyNotifications
.find( pEvents
->PropertyName
) != mpData
->aSuspendedPropertyNotifications
.end() )
456 if ( pEvents
!= pEventsEnd
)
457 ::std::copy( pEvents
+ 1, pEventsEnd
, pEvents
);
462 aEvents
.realloc( pEventsEnd
- aEvents
.getConstArray() );
464 if ( !aEvents
.getLength() )
469 ImplModelPropertiesChanged( aEvents
);
472 void UnoControl::ImplLockPropertyChangeNotification( const ::rtl::OUString
& rPropertyName
, bool bLock
)
474 MapString2Int::iterator pos
= mpData
->aSuspendedPropertyNotifications
.find( rPropertyName
);
477 if ( pos
== mpData
->aSuspendedPropertyNotifications
.end() )
478 pos
= mpData
->aSuspendedPropertyNotifications
.insert( MapString2Int::value_type( rPropertyName
, 0 ) ).first
;
483 OSL_ENSURE( pos
!= mpData
->aSuspendedPropertyNotifications
.end(), "UnoControl::ImplLockPropertyChangeNotification: property not locked!" );
484 if ( pos
!= mpData
->aSuspendedPropertyNotifications
.end() )
486 OSL_ENSURE( pos
->second
> 0, "UnoControl::ImplLockPropertyChangeNotification: invalid suspension counter!" );
487 if ( 0 == --pos
->second
)
488 mpData
->aSuspendedPropertyNotifications
.erase( pos
);
493 void UnoControl::ImplLockPropertyChangeNotifications( const Sequence
< ::rtl::OUString
>& rPropertyNames
, bool bLock
)
495 for ( const ::rtl::OUString
* pPropertyName
= rPropertyNames
.getConstArray();
496 pPropertyName
!= rPropertyNames
.getConstArray() + rPropertyNames
.getLength();
499 ImplLockPropertyChangeNotification( *pPropertyName
, bLock
);
502 void UnoControl::ImplModelPropertiesChanged( const Sequence
< PropertyChangeEvent
>& rEvents
)
504 ::osl::ClearableGuard
< ::osl::Mutex
> aGuard( GetMutex() );
508 DECLARE_STL_VECTOR( PropertyValue
, PropertyValueVector
);
509 PropertyValueVector aPeerPropertiesToSet
;
510 sal_Int32 nIndependentPos
= 0;
511 bool bResourceResolverSet( false );
512 // position where to insert the independent properties into aPeerPropertiesToSet,
513 // dependent ones are inserted at the end of the vector
515 sal_Bool bNeedNewPeer
= sal_False
;
516 // some properties require a re-creation of the peer, 'cause they can't be changed on the fly
518 Reference
< XControlModel
> xOwnModel( getModel(), UNO_QUERY
);
519 // our own model for comparison
520 Reference
< XPropertySet
> xPS( xOwnModel
, UNO_QUERY
);
521 Reference
< XPropertySetInfo
> xPSI( xPS
->getPropertySetInfo(), UNO_QUERY
);
522 OSL_ENSURE( xPSI
.is(), "UnoControl::ImplModelPropertiesChanged: should have property set meta data!" );
524 const PropertyChangeEvent
* pEvents
= rEvents
.getConstArray();
526 sal_Int32 nLen
= rEvents
.getLength();
527 aPeerPropertiesToSet
.reserve(nLen
);
529 for( sal_Int32 i
= 0; i
< nLen
; ++i
, ++pEvents
)
531 Reference
< XControlModel
> xModel( pEvents
->Source
, UNO_QUERY
);
532 sal_Bool bOwnModel
= xModel
.get() == xOwnModel
.get();
536 // Detect changes on our resource resolver which invalidates
537 // automatically some language dependent properties.
538 if ( pEvents
->PropertyName
.equalsAsciiL( "ResourceResolver", 16 ))
540 Reference
< resource::XStringResourceResolver
> xStrResolver
;
541 if ( pEvents
->NewValue
>>= xStrResolver
)
542 bResourceResolverSet
= xStrResolver
.is();
545 sal_uInt16 nPType
= GetPropertyId( pEvents
->PropertyName
);
546 if ( mbDesignMode
&& mbDisposePeer
&& !mbRefeshingPeer
&& !mbCreatingPeer
)
548 // if we're in design mode, then some properties can change which
549 // require creating a *new* peer (since these properties cannot
550 // be switched at existing peers)
552 bNeedNewPeer
= ( nPType
== BASEPROPERTY_BORDER
)
553 || ( nPType
== BASEPROPERTY_MULTILINE
)
554 || ( nPType
== BASEPROPERTY_DROPDOWN
)
555 || ( nPType
== BASEPROPERTY_HSCROLL
)
556 || ( nPType
== BASEPROPERTY_VSCROLL
)
557 || ( nPType
== BASEPROPERTY_AUTOHSCROLL
)
558 || ( nPType
== BASEPROPERTY_AUTOVSCROLL
)
559 || ( nPType
== BASEPROPERTY_ORIENTATION
)
560 || ( nPType
== BASEPROPERTY_SPIN
)
561 || ( nPType
== BASEPROPERTY_ALIGN
)
562 || ( nPType
== BASEPROPERTY_PAINTTRANSPARENT
);
564 bNeedNewPeer
= requiresNewPeer( pEvents
->PropertyName
);
570 if ( nPType
&& ( nLen
> 1 ) && DoesDependOnOthers( nPType
) )
572 // Properties die von anderen abhaengen erst hinterher einstellen,
573 // weil sie von anderen Properties abhaengig sind, die aber erst spaeter
574 // eingestellt werden, z.B. VALUE nach VALUEMIN/MAX.
575 aPeerPropertiesToSet
.push_back(PropertyValue(pEvents
->PropertyName
, 0, pEvents
->NewValue
, PropertyState_DIRECT_VALUE
));
579 if ( bResourceResolverSet
)
581 // The resource resolver property change should be one of the first ones.
582 // All language dependent properties are dependent on this property.
583 // As BASEPROPERTY_NATIVE_WIDGET_LOOK is not dependent on resource
584 // resolver. We don't need to handle a special order for these two props.
585 aPeerPropertiesToSet
.insert(
586 aPeerPropertiesToSet
.begin(),
587 PropertyValue( pEvents
->PropertyName
, 0, pEvents
->NewValue
, PropertyState_DIRECT_VALUE
) );
590 else if ( nPType
== BASEPROPERTY_NATIVE_WIDGET_LOOK
)
592 // since *a lot* of other properties might be overruled by this one, we need
593 // a special handling:
594 // NativeWidgetLook needs to be set first: If it is set to ON, all other
595 // properties describing the look (e.g. BackgroundColor) are ignored, anyway.
596 // If it is switched OFF, then we need to do it first because else it will
597 // overrule other look-related properties, and re-initialize them from system
599 aPeerPropertiesToSet
.insert(
600 aPeerPropertiesToSet
.begin(),
601 PropertyValue( pEvents
->PropertyName
, 0, pEvents
->NewValue
, PropertyState_DIRECT_VALUE
) );
606 aPeerPropertiesToSet
.insert(aPeerPropertiesToSet
.begin() + nIndependentPos
,
607 PropertyValue(pEvents
->PropertyName
, 0, pEvents
->NewValue
, PropertyState_DIRECT_VALUE
));
613 Reference
< XWindow
> xParent
= getParentPeer();
614 Reference
< XControl
> xThis( (XAggregation
*)(::cppu::OWeakAggObject
*)this, UNO_QUERY
);
615 // call createPeer via a interface got from queryInterface, so the aggregating class can intercept it
617 DBG_ASSERT( !bNeedNewPeer
|| xParent
.is(), "Need new peer, but don't have a parent!" );
619 // Check if we have to update language dependent properties
620 if ( !bNeedNewPeer
&& bResourceResolverSet
)
622 // Add language dependent properties into the peer property set.
623 // Our resource resolver has been changed and we must be sure
624 // that language dependent props use the new resolver.
625 const LanguageDependentProp
* pLangDepProp
= aLanguageDependentProp
;
626 while ( pLangDepProp
->pPropName
!= 0 )
628 bool bMustBeInserted( true );
629 for ( sal_uInt32 i
= 0; i
< aPeerPropertiesToSet
.size(); i
++ )
631 if ( aPeerPropertiesToSet
[i
].Name
.equalsAsciiL(
632 pLangDepProp
->pPropName
, pLangDepProp
->nPropNameLength
))
634 bMustBeInserted
= false;
639 if ( bMustBeInserted
)
641 // Add language dependent props at the end
642 ::rtl::OUString
aPropName( ::rtl::OUString::createFromAscii( pLangDepProp
->pPropName
));
643 if ( xPSI
.is() && xPSI
->hasPropertyByName( aPropName
) )
645 aPeerPropertiesToSet
.push_back(
646 PropertyValue( aPropName
, 0, xPS
->getPropertyValue( aPropName
), PropertyState_DIRECT_VALUE
) );
655 // clear the guard before creating a new peer - as usual, our peer implementations use the SolarMutex
656 // #82300# - 2000-12-21 - fs@openoffice.org
657 if (bNeedNewPeer
&& xParent
.is())
659 NAMESPACE_VOS(OGuard
) aVclGuard( Application::GetSolarMutex() );
660 // and now this is the final withdrawal:
661 // With 83561, I have no other idea than locking the SolarMutex here ....
662 // I really hate the fact that VCL is not theadsafe ....
663 // #83561# - 2001-03-01 - fs@openoffice.org
665 // Funktioniert beim Container nicht!
666 getPeer()->dispose();
668 mxVclWindowPeer
= NULL
;
669 mbRefeshingPeer
= sal_True
;
670 Reference
< XWindowPeer
> xP( xParent
, UNO_QUERY
);
671 xThis
->createPeer( Reference
< XToolkit
> (), xP
);
672 mbRefeshingPeer
= sal_False
;
673 aPeerPropertiesToSet
.clear();
676 // lock the multiplexing of VCL events to our UNO listeners
677 // this is for compatibility reasons: in OOo 1.0.x, changes which were done at the
678 // model did not cause the listeners of the controls/peers to be called
679 // Since the implementations for the listeners changed a lot towards 1.1, this
680 // would not be the case anymore, if we would not do this listener-lock below
681 // #i14703# - 2003-05-23 - fs@openoffice.org
682 Window
* pVclPeer
= VCLUnoHelper::GetWindow( getPeer() );
683 VCLXWindow
* pPeer
= pVclPeer
? pVclPeer
->GetWindowPeer() : NULL
;
684 VclListenerLock
aNoVclEventMultiplexing( pPeer
);
686 // setting peer properties may result in an attemp to acquire the solar mutex, 'cause the peers
687 // usually don't have an own mutex but use the SolarMutex instead.
688 // To prevent deadlocks resulting from this, we do this without our own mutex locked
689 // 2000-11-03 - fs@openoffice.org
690 PropertyValueVectorIterator aEnd
= aPeerPropertiesToSet
.end();
691 for ( PropertyValueVectorIterator aLoop
= aPeerPropertiesToSet
.begin();
696 ImplSetPeerProperty( aLoop
->Name
, aLoop
->Value
);
701 void UnoControl::disposing( const EventObject
& rEvt
) throw(RuntimeException
)
703 ::osl::ClearableMutexGuard
aGuard( GetMutex() );
704 // bei "Multible Inheritance" nicht unterschiedliche Typen vergleichen.
706 if ( maAccessibleContext
.get() == rEvt
.Source
)
708 // just in case the context is disposed, but not released - ensure that we do not re-use it in the future
709 maAccessibleContext
= NULL
;
711 else if( mxModel
.get() == Reference
< XControlModel
>(rEvt
.Source
,UNO_QUERY
).get() )
713 // #62337# if the model dies, it does not make sense for us to live ...
714 Reference
< XControl
> xThis
= this;
719 DBG_ASSERT( !mxModel
.is(), "UnoControl::disposing: invalid dispose behaviour!" );
725 void SAL_CALL
UnoControl::setOutputSize( const awt::Size
& aSize
) throw (RuntimeException
)
727 Reference
< XWindow2
> xPeerWindow
;
729 ::osl::MutexGuard
aGuard( GetMutex() );
730 xPeerWindow
= xPeerWindow
.query( getPeer() );
733 if ( xPeerWindow
.is() )
734 xPeerWindow
->setOutputSize( aSize
);
739 template < typename RETVALTYPE
>
740 RETVALTYPE
lcl_askPeer( const uno::Reference
< awt::XWindowPeer
>& _rxPeer
, RETVALTYPE (SAL_CALL
XWindow2::*_pMethod
)(), RETVALTYPE _aDefault
)
742 RETVALTYPE
aReturn( _aDefault
);
744 Reference
< XWindow2
> xPeerWindow( _rxPeer
, UNO_QUERY
);
745 if ( xPeerWindow
.is() )
746 aReturn
= (xPeerWindow
.get()->*_pMethod
)();
752 awt::Size SAL_CALL
UnoControl::getOutputSize( ) throw (RuntimeException
)
754 return lcl_askPeer( getPeer(), &XWindow2::getOutputSize
, awt::Size() );
757 ::sal_Bool SAL_CALL
UnoControl::isVisible( ) throw (RuntimeException
)
759 return lcl_askPeer( getPeer(), &XWindow2::isVisible
, maComponentInfos
.bVisible
);
762 ::sal_Bool SAL_CALL
UnoControl::isActive( ) throw (RuntimeException
)
764 return lcl_askPeer( getPeer(), &XWindow2::isActive
, sal_False
);
767 ::sal_Bool SAL_CALL
UnoControl::isEnabled( ) throw (RuntimeException
)
769 return lcl_askPeer( getPeer(), &XWindow2::isEnabled
, maComponentInfos
.bEnable
);
772 ::sal_Bool SAL_CALL
UnoControl::hasFocus( ) throw (RuntimeException
)
774 return lcl_askPeer( getPeer(), &XWindow2::hasFocus
, sal_False
);
778 void UnoControl::setPosSize( sal_Int32 X
, sal_Int32 Y
, sal_Int32 Width
, sal_Int32 Height
, sal_Int16 Flags
) throw(RuntimeException
)
780 Reference
< XWindow
> xWindow
;
782 ::osl::MutexGuard
aGuard( GetMutex() );
784 if ( Flags
& awt::PosSize::X
)
785 maComponentInfos
.nX
= X
;
786 if ( Flags
& awt::PosSize::Y
)
787 maComponentInfos
.nY
= Y
;
788 if ( Flags
& awt::PosSize::WIDTH
)
789 maComponentInfos
.nWidth
= Width
;
790 if ( Flags
& awt::PosSize::HEIGHT
)
791 maComponentInfos
.nHeight
= Height
;
792 maComponentInfos
.nFlags
|= Flags
;
794 xWindow
= xWindow
.query( getPeer() );
798 xWindow
->setPosSize( X
, Y
, Width
, Height
, Flags
);
801 awt::Rectangle
UnoControl::getPosSize( ) throw(RuntimeException
)
803 awt::Rectangle
aRect( maComponentInfos
.nX
, maComponentInfos
.nY
, maComponentInfos
.nWidth
, maComponentInfos
.nHeight
);
804 Reference
< XWindow
> xWindow
;
807 ::osl::MutexGuard
aGuard( GetMutex() );
808 xWindow
= xWindow
.query( getPeer() );
812 aRect
= xWindow
->getPosSize();
816 void UnoControl::setVisible( sal_Bool bVisible
) throw(RuntimeException
)
818 Reference
< XWindow
> xWindow
;
820 ::osl::MutexGuard
aGuard( GetMutex() );
822 // Visible status ist Sache der View
823 maComponentInfos
.bVisible
= bVisible
;
824 xWindow
= xWindow
.query( getPeer() );
827 xWindow
->setVisible( bVisible
);
830 void UnoControl::setEnable( sal_Bool bEnable
) throw(RuntimeException
)
832 Reference
< XWindow
> xWindow
;
834 ::osl::MutexGuard
aGuard( GetMutex() );
836 // Enable status ist Sache der View
837 maComponentInfos
.bEnable
= bEnable
;
838 xWindow
= xWindow
.query( getPeer() );
841 xWindow
->setEnable( bEnable
);
844 void UnoControl::setFocus( ) throw(RuntimeException
)
846 Reference
< XWindow
> xWindow
;
848 ::osl::MutexGuard
aGuard( GetMutex() );
849 xWindow
= xWindow
.query( getPeer() );
855 void UnoControl::addWindowListener( const Reference
< XWindowListener
>& rxListener
) throw(RuntimeException
)
857 Reference
< XWindow
> xPeerWindow
;
859 ::osl::MutexGuard
aGuard( GetMutex() );
860 maWindowListeners
.addInterface( rxListener
);
861 if ( maWindowListeners
.getLength() == 1 )
862 xPeerWindow
= xPeerWindow
.query( getPeer() );
864 if ( xPeerWindow
.is() )
865 xPeerWindow
->addWindowListener( &maWindowListeners
);
868 void UnoControl::removeWindowListener( const Reference
< XWindowListener
>& rxListener
) throw(RuntimeException
)
870 Reference
< XWindow
> xPeerWindow
;
872 ::osl::MutexGuard
aGuard( GetMutex() );
873 if ( maWindowListeners
.getLength() == 1 )
874 xPeerWindow
= xPeerWindow
.query( getPeer() );
875 maWindowListeners
.removeInterface( rxListener
);
877 if ( xPeerWindow
.is() )
878 xPeerWindow
->removeWindowListener( &maWindowListeners
);
881 void UnoControl::addFocusListener( const Reference
< XFocusListener
>& rxListener
) throw(RuntimeException
)
883 Reference
< XWindow
> xPeerWindow
;
885 ::osl::MutexGuard
aGuard( GetMutex() );
886 maFocusListeners
.addInterface( rxListener
);
887 if ( maFocusListeners
.getLength() == 1 )
888 xPeerWindow
= xPeerWindow
.query( getPeer() );
890 if ( xPeerWindow
.is() )
891 xPeerWindow
->addFocusListener( &maFocusListeners
);
894 void UnoControl::removeFocusListener( const Reference
< XFocusListener
>& rxListener
) throw(RuntimeException
)
896 Reference
< XWindow
> xPeerWindow
;
898 ::osl::MutexGuard
aGuard( GetMutex() );
899 if ( maFocusListeners
.getLength() == 1 )
900 xPeerWindow
= xPeerWindow
.query( getPeer() );
901 maFocusListeners
.removeInterface( rxListener
);
903 if ( xPeerWindow
.is() )
904 xPeerWindow
->removeFocusListener( &maFocusListeners
);
907 void UnoControl::addKeyListener( const Reference
< XKeyListener
>& rxListener
) throw(RuntimeException
)
909 Reference
< XWindow
> xPeerWindow
;
911 ::osl::MutexGuard
aGuard( GetMutex() );
912 maKeyListeners
.addInterface( rxListener
);
913 if ( maKeyListeners
.getLength() == 1 )
914 xPeerWindow
= xPeerWindow
.query( getPeer() );
916 if ( xPeerWindow
.is() )
917 xPeerWindow
->addKeyListener( &maKeyListeners
);
920 void UnoControl::removeKeyListener( const Reference
< XKeyListener
>& rxListener
) throw(RuntimeException
)
922 Reference
< XWindow
> xPeerWindow
;
924 ::osl::MutexGuard
aGuard( GetMutex() );
925 if ( maKeyListeners
.getLength() == 1 )
926 xPeerWindow
= xPeerWindow
.query( getPeer() );
927 maKeyListeners
.removeInterface( rxListener
);
929 if ( xPeerWindow
.is() )
930 xPeerWindow
->removeKeyListener( &maKeyListeners
);
933 void UnoControl::addMouseListener( const Reference
< XMouseListener
>& rxListener
) throw(RuntimeException
)
935 Reference
< XWindow
> xPeerWindow
;
937 ::osl::MutexGuard
aGuard( GetMutex() );
938 maMouseListeners
.addInterface( rxListener
);
939 if ( maMouseListeners
.getLength() == 1 )
940 xPeerWindow
= xPeerWindow
.query( getPeer() );
942 if ( xPeerWindow
.is() )
943 xPeerWindow
->addMouseListener( &maMouseListeners
);
946 void UnoControl::removeMouseListener( const Reference
< XMouseListener
>& rxListener
) throw(RuntimeException
)
948 Reference
< XWindow
> xPeerWindow
;
950 ::osl::MutexGuard
aGuard( GetMutex() );
951 if ( maMouseListeners
.getLength() == 1 )
952 xPeerWindow
= xPeerWindow
.query( getPeer() );
953 maMouseListeners
.removeInterface( rxListener
);
955 if ( xPeerWindow
.is() )
956 xPeerWindow
->removeMouseListener( &maMouseListeners
);
959 void UnoControl::addMouseMotionListener( const Reference
< XMouseMotionListener
>& rxListener
) throw(RuntimeException
)
961 Reference
< XWindow
> xPeerWindow
;
963 ::osl::MutexGuard
aGuard( GetMutex() );
964 maMouseMotionListeners
.addInterface( rxListener
);
965 if ( maMouseMotionListeners
.getLength() == 1 )
966 xPeerWindow
= xPeerWindow
.query( getPeer() );
968 if ( xPeerWindow
.is() )
969 xPeerWindow
->addMouseMotionListener( &maMouseMotionListeners
);
972 void UnoControl::removeMouseMotionListener( const Reference
< XMouseMotionListener
>& rxListener
) throw(RuntimeException
)
974 Reference
< XWindow
> xPeerWindow
;
976 ::osl::MutexGuard
aGuard( GetMutex() );
977 if ( maMouseMotionListeners
.getLength() == 1 )
978 xPeerWindow
= xPeerWindow
.query( getPeer() );
979 maMouseMotionListeners
.removeInterface( rxListener
);
981 if ( xPeerWindow
.is() )
982 xPeerWindow
->removeMouseMotionListener( &maMouseMotionListeners
);
985 void UnoControl::addPaintListener( const Reference
< XPaintListener
>& rxListener
) throw(RuntimeException
)
987 Reference
< XWindow
> xPeerWindow
;
989 ::osl::MutexGuard
aGuard( GetMutex() );
990 maPaintListeners
.addInterface( rxListener
);
991 if ( maPaintListeners
.getLength() == 1 )
992 xPeerWindow
= xPeerWindow
.query( getPeer() );
994 if ( xPeerWindow
.is() )
995 xPeerWindow
->addPaintListener( &maPaintListeners
);
998 void UnoControl::removePaintListener( const Reference
< XPaintListener
>& rxListener
) throw(RuntimeException
)
1000 Reference
< XWindow
> xPeerWindow
;
1002 ::osl::MutexGuard
aGuard( GetMutex() );
1003 if ( maPaintListeners
.getLength() == 1 )
1004 xPeerWindow
= xPeerWindow
.query( getPeer() );
1005 maPaintListeners
.removeInterface( rxListener
);
1007 if ( xPeerWindow
.is() )
1008 xPeerWindow
->removePaintListener( &maPaintListeners
);
1012 sal_Bool
UnoControl::setGraphics( const Reference
< XGraphics
>& rDevice
) throw(RuntimeException
)
1014 Reference
< XView
> xView
;
1016 ::osl::MutexGuard
aGuard( GetMutex() );
1018 mxGraphics
= rDevice
;
1019 xView
= xView
.query( getPeer() );
1021 return xView
.is() ? xView
->setGraphics( rDevice
) : sal_True
;
1024 Reference
< XGraphics
> UnoControl::getGraphics( ) throw(RuntimeException
)
1029 awt::Size
UnoControl::getSize( ) throw(RuntimeException
)
1031 ::osl::MutexGuard
aGuard( GetMutex() );
1032 return awt::Size( maComponentInfos
.nWidth
, maComponentInfos
.nHeight
);
1035 void UnoControl::draw( sal_Int32 x
, sal_Int32 y
) throw(RuntimeException
)
1037 Reference
< XWindowPeer
> xDrawPeer
;
1038 Reference
< XView
> xDrawPeerView
;
1040 bool bDisposeDrawPeer( false );
1042 ::osl::MutexGuard
aGuard( GetMutex() );
1044 xDrawPeer
= ImplGetCompatiblePeer( sal_True
);
1045 bDisposeDrawPeer
= xDrawPeer
.is() && ( xDrawPeer
!= getPeer() );
1047 xDrawPeerView
.set( xDrawPeer
, UNO_QUERY
);
1048 DBG_ASSERT( xDrawPeerView
.is(), "UnoControl::draw: no peer!" );
1051 if ( xDrawPeerView
.is() )
1053 Reference
< XVclWindowPeer
> xWindowPeer
;
1054 xWindowPeer
.set( xDrawPeer
, UNO_QUERY
);
1055 if ( xWindowPeer
.is() )
1056 xWindowPeer
->setDesignMode( mbDesignMode
);
1057 xDrawPeerView
->draw( x
, y
);
1060 if ( bDisposeDrawPeer
)
1061 xDrawPeer
->dispose();
1064 void UnoControl::setZoom( float fZoomX
, float fZoomY
) throw(RuntimeException
)
1066 Reference
< XView
> xView
;
1068 ::osl::MutexGuard
aGuard( GetMutex() );
1070 maComponentInfos
.nZoomX
= fZoomX
;
1071 maComponentInfos
.nZoomY
= fZoomY
;
1073 xView
= xView
.query( getPeer() );
1076 xView
->setZoom( fZoomX
, fZoomY
);
1080 void UnoControl::setContext( const Reference
< XInterface
>& rxContext
) throw(RuntimeException
)
1082 ::osl::MutexGuard
aGuard( GetMutex() );
1084 mxContext
= rxContext
;
1087 Reference
< XInterface
> UnoControl::getContext( ) throw(RuntimeException
)
1089 ::osl::MutexGuard
aGuard( GetMutex() );
1094 void UnoControl::peerCreated()
1096 Reference
< XWindow
> xWindow( getPeer(), UNO_QUERY
);
1097 if ( !xWindow
.is() )
1100 if ( maWindowListeners
.getLength() )
1101 xWindow
->addWindowListener( &maWindowListeners
);
1103 if ( maFocusListeners
.getLength() )
1104 xWindow
->addFocusListener( &maFocusListeners
);
1106 if ( maKeyListeners
.getLength() )
1107 xWindow
->addKeyListener( &maKeyListeners
);
1109 if ( maMouseListeners
.getLength() )
1110 xWindow
->addMouseListener( &maMouseListeners
);
1112 if ( maMouseMotionListeners
.getLength() )
1113 xWindow
->addMouseMotionListener( &maMouseMotionListeners
);
1115 if ( maPaintListeners
.getLength() )
1116 xWindow
->addPaintListener( &maPaintListeners
);
1119 void UnoControl::createPeer( const Reference
< XToolkit
>& rxToolkit
, const Reference
< XWindowPeer
>& rParentPeer
) throw(RuntimeException
)
1121 ::osl::ClearableMutexGuard
aGuard( GetMutex() );
1123 if ( !mxModel
.is() )
1125 RuntimeException aException
;
1126 aException
.Message
= ::rtl::OUString::createFromAscii( "createPeer: no model!" );
1127 aException
.Context
= (XAggregation
*)(::cppu::OWeakAggObject
*)this;
1128 throw( aException
);
1131 if( !getPeer().is() )
1133 mbCreatingPeer
= sal_True
;
1136 Reference
< XToolkit
> xToolkit
= rxToolkit
;
1137 if( rParentPeer
.is() && mxContext
.is() )
1140 if ( !xToolkit
.is() )
1141 xToolkit
= rParentPeer
->getToolkit();
1142 Any aAny
= OWeakAggObject::queryInterface( ::getCppuType((const Reference
< XControlContainer
>*)0) );
1143 Reference
< XControlContainer
> xC
;
1146 // Es ist ein Container
1147 eType
= WindowClass_CONTAINER
;
1149 eType
= WindowClass_SIMPLE
;
1152 { // Nur richtig, wenn es sich um ein Top Window handelt
1153 if( rParentPeer
.is() )
1155 if ( !xToolkit
.is() )
1156 xToolkit
= rParentPeer
->getToolkit();
1157 eType
= WindowClass_CONTAINER
;
1161 if ( !xToolkit
.is() )
1162 xToolkit
= VCLUnoHelper::CreateToolkit();
1163 eType
= WindowClass_TOP
;
1166 WindowDescriptor aDescr
;
1167 aDescr
.Type
= eType
;
1168 aDescr
.WindowServiceName
= GetComponentServiceName();
1169 aDescr
.Parent
= rParentPeer
;
1170 aDescr
.Bounds
= getPosSize();
1171 aDescr
.WindowAttributes
= 0;
1174 Reference
< XPropertySet
> xPSet( mxModel
, UNO_QUERY
);
1175 Reference
< XPropertySetInfo
> xInfo
= xPSet
->getPropertySetInfo();
1178 ::rtl::OUString aPropName
= GetPropertyName( BASEPROPERTY_BORDER
);
1179 if ( xInfo
->hasPropertyByName( aPropName
) )
1181 aVal
= xPSet
->getPropertyValue( aPropName
);
1182 sal_Int16 n
= sal_Int16();
1186 aDescr
.WindowAttributes
|= WindowAttribute::BORDER
;
1188 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::NOBORDER
;
1192 // DESKTOP_AS_PARENT
1193 if ( aDescr
.Type
== WindowClass_TOP
)
1195 aPropName
= GetPropertyName( BASEPROPERTY_DESKTOP_AS_PARENT
);
1196 if ( xInfo
->hasPropertyByName( aPropName
) )
1198 aVal
= xPSet
->getPropertyValue( aPropName
);
1199 sal_Bool b
= sal_Bool();
1200 if ( ( aVal
>>= b
) && b
)
1201 aDescr
.ParentIndex
= -1;
1205 aPropName
= GetPropertyName( BASEPROPERTY_MOVEABLE
);
1206 if ( xInfo
->hasPropertyByName( aPropName
) )
1208 aVal
= xPSet
->getPropertyValue( aPropName
);
1209 sal_Bool b
= sal_Bool();
1210 if ( ( aVal
>>= b
) && b
)
1211 aDescr
.WindowAttributes
|= WindowAttribute::MOVEABLE
;
1215 aPropName
= GetPropertyName( BASEPROPERTY_CLOSEABLE
);
1216 if ( xInfo
->hasPropertyByName( aPropName
) )
1218 aVal
= xPSet
->getPropertyValue( aPropName
);
1219 sal_Bool b
= sal_Bool();
1220 if ( ( aVal
>>= b
) && b
)
1221 aDescr
.WindowAttributes
|= WindowAttribute::CLOSEABLE
;
1225 aPropName
= GetPropertyName( BASEPROPERTY_DROPDOWN
);
1226 if ( xInfo
->hasPropertyByName( aPropName
) )
1228 aVal
= xPSet
->getPropertyValue( aPropName
);
1229 sal_Bool b
= sal_Bool();
1230 if ( ( aVal
>>= b
) && b
)
1231 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::DROPDOWN
;
1235 aPropName
= GetPropertyName( BASEPROPERTY_SPIN
);
1236 if ( xInfo
->hasPropertyByName( aPropName
) )
1238 aVal
= xPSet
->getPropertyValue( aPropName
);
1239 sal_Bool b
= sal_Bool();
1240 if ( ( aVal
>>= b
) && b
)
1241 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::SPIN
;
1245 aPropName
= GetPropertyName( BASEPROPERTY_HSCROLL
);
1246 if ( xInfo
->hasPropertyByName( aPropName
) )
1248 aVal
= xPSet
->getPropertyValue( aPropName
);
1249 sal_Bool b
= sal_Bool();
1250 if ( ( aVal
>>= b
) && b
)
1251 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::HSCROLL
;
1255 aPropName
= GetPropertyName( BASEPROPERTY_VSCROLL
);
1256 if ( xInfo
->hasPropertyByName( aPropName
) )
1258 aVal
= xPSet
->getPropertyValue( aPropName
);
1259 sal_Bool b
= sal_Bool();
1260 if ( ( aVal
>>= b
) && b
)
1261 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::VSCROLL
;
1265 aPropName
= GetPropertyName( BASEPROPERTY_AUTOHSCROLL
);
1266 if ( xInfo
->hasPropertyByName( aPropName
) )
1268 aVal
= xPSet
->getPropertyValue( aPropName
);
1269 sal_Bool b
= sal_Bool();
1270 if ( ( aVal
>>= b
) && b
)
1271 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::AUTOHSCROLL
;
1275 aPropName
= GetPropertyName( BASEPROPERTY_AUTOVSCROLL
);
1276 if ( xInfo
->hasPropertyByName( aPropName
) )
1278 aVal
= xPSet
->getPropertyValue( aPropName
);
1279 sal_Bool b
= sal_Bool();
1280 if ( ( aVal
>>= b
) && b
)
1281 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::AUTOVSCROLL
;
1284 //added for issue79712
1286 aPropName
= GetPropertyName( BASEPROPERTY_NOLABEL
);
1287 if ( xInfo
->hasPropertyByName( aPropName
) )
1289 aVal
= xPSet
->getPropertyValue( aPropName
);
1290 sal_Bool b
= sal_Bool();
1291 if ( ( aVal
>>=b
) && b
)
1292 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::NOLABEL
;
1297 aPropName
= GetPropertyName( BASEPROPERTY_ALIGN
);
1298 if ( xInfo
->hasPropertyByName( aPropName
) )
1300 aVal
= xPSet
->getPropertyValue( aPropName
);
1301 sal_Int16 n
= sal_Int16();
1304 if ( n
== PROPERTY_ALIGN_LEFT
)
1305 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::LEFT
;
1306 else if ( n
== PROPERTY_ALIGN_CENTER
)
1307 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::CENTER
;
1309 aDescr
.WindowAttributes
|= VclWindowPeerAttribute::RIGHT
;
1313 // Ableitungen die Moeglichkeit geben die Attribute zu manipulieren
1314 PrepareWindowDescriptor(aDescr
);
1317 setPeer( xToolkit
->createWindow( aDescr
) );
1319 // release the mutex guard (and work with copies of our members)
1320 // this is necessary as our peer may lock the SolarMutex (actually, all currently known peers do), so calling
1321 // into the peer with our own mutex locked may cause deadlocks
1322 // (We _really_ need peers which do not use the SolarMutex. It's really pissing me off that from time to
1323 // time deadlocks pop up because the low-level components like our peers use a mutex which ususally
1324 // is locked at the top of the stack (it protects the global message looping). This is always dangerous, and
1325 // can not always be solved by tampering with other mutexes.
1326 // Unfortunately, the VCL used in the peers is not threadsafe, and by definition needs a locked SolarMutex.)
1327 // 82300 - 12/21/00 - FS
1328 UnoControlComponentInfos
aComponentInfos(maComponentInfos
);
1329 sal_Bool
bDesignMode(mbDesignMode
);
1331 Reference
< XGraphics
> xGraphics( mxGraphics
);
1332 Reference
< XView
> xView ( getPeer(), UNO_QUERY
);
1333 Reference
< XWindow
> xWindow ( getPeer(), UNO_QUERY
);
1337 // the updateFromModel is done without a locked mutex, too.
1338 // The reason is that the only thing this method does is firing property changes, and this in general has
1339 // to be done without locked mutexes (as every notification to external listeners).
1340 // 82300 - 12/21/00 - FS
1343 xView
->setZoom( aComponentInfos
.nZoomX
, aComponentInfos
.nZoomY
);
1345 setPosSize( aComponentInfos
.nX
, aComponentInfos
.nY
, aComponentInfos
.nWidth
, aComponentInfos
.nHeight
, aComponentInfos
.nFlags
);
1347 if( aComponentInfos
.bVisible
&& !bDesignMode
)
1348 // Erst nach dem setzen der Daten anzeigen
1349 xWindow
->setVisible( aComponentInfos
.bVisible
);
1351 if( !aComponentInfos
.bEnable
)
1352 xWindow
->setEnable( aComponentInfos
.bEnable
);
1354 xView
->setGraphics( xGraphics
);
1358 mbCreatingPeer
= sal_False
;
1362 Reference
< XWindowPeer
> UnoControl::getPeer() throw(RuntimeException
)
1364 ::osl::MutexGuard
aGuard( GetMutex() );
1368 sal_Bool
UnoControl::setModel( const Reference
< XControlModel
>& rxModel
) throw(RuntimeException
)
1370 ::osl::MutexGuard
aGuard( GetMutex() );
1372 Reference
< XMultiPropertySet
> xPropSet( mxModel
, UNO_QUERY
);
1374 // query for the XPropertiesChangeListener - our delegator is allowed to overwrite this interface
1375 Reference
< XPropertiesChangeListener
> xListener
;
1376 queryInterface( ::getCppuType( &xListener
) ) >>= xListener
;
1379 xPropSet
->removePropertiesChangeListener( xListener
);
1381 mpData
->bLocalizationSupport
= false;
1388 xPropSet
.set( mxModel
, UNO_QUERY_THROW
);
1389 Reference
< XPropertySetInfo
> xPSI( xPropSet
->getPropertySetInfo(), UNO_SET_THROW
);
1391 Sequence
< ::rtl::OUString
> aNames
= lcl_ImplGetPropertyNames( xPropSet
);
1392 xPropSet
->addPropertiesChangeListener( aNames
, xListener
);
1394 mpData
->bLocalizationSupport
= xPSI
->hasPropertyByName( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "ResourceResolver" ) ) );
1396 catch( const Exception
& )
1398 DBG_UNHANDLED_EXCEPTION();
1403 return mxModel
.is();
1406 Reference
< XControlModel
> UnoControl::getModel( ) throw(RuntimeException
)
1411 Reference
< XView
> UnoControl::getView( ) throw(RuntimeException
)
1413 return static_cast< XView
* >( this );
1416 void UnoControl::setDesignMode( sal_Bool bOn
) throw(RuntimeException
)
1418 ModeChangeEvent aModeChangeEvent
;
1420 Reference
< XWindow
> xWindow
;
1422 ::osl::MutexGuard
aGuard( GetMutex() );
1423 if ( bOn
== mbDesignMode
)
1428 xWindow
= xWindow
.query( getPeer() );
1429 // dispose our current AccessibleContext, if we have one
1430 // (changing the design mode implies having a new implementation for this context,
1431 // so the old one must be declared DEFUNC)
1432 disposeAccessibleContext();
1434 aModeChangeEvent
.Source
= *this;
1435 aModeChangeEvent
.NewMode
= ::rtl::OUString::createFromAscii( mbDesignMode
? "design" : "alive" );
1438 // ajust the visibility of our window
1440 xWindow
->setVisible( !bOn
);
1442 // and notify our mode listeners
1443 maModeChangeListeners
.notifyEach( &XModeChangeListener::modeChanged
, aModeChangeEvent
);
1446 sal_Bool
UnoControl::isDesignMode( ) throw(RuntimeException
)
1448 return mbDesignMode
;
1451 sal_Bool
UnoControl::isTransparent( ) throw(RuntimeException
)
1457 ::rtl::OUString
UnoControl::getImplementationName( ) throw(RuntimeException
)
1459 DBG_ERROR( "This method should be overloaded!" );
1460 return ::rtl::OUString();
1463 sal_Bool
UnoControl::supportsService( const ::rtl::OUString
& rServiceName
) throw(RuntimeException
)
1465 ::osl::MutexGuard
aGuard( GetMutex() );
1467 Sequence
< ::rtl::OUString
> aSNL
= getSupportedServiceNames();
1468 const ::rtl::OUString
* pArray
= aSNL
.getConstArray();
1469 const ::rtl::OUString
* pArrayEnd
= aSNL
.getConstArray();
1470 for (; pArray
!= pArrayEnd
; ++pArray
)
1471 if( *pArray
== rServiceName
)
1474 return pArray
!= pArrayEnd
;
1477 Sequence
< ::rtl::OUString
> UnoControl::getSupportedServiceNames( ) throw(RuntimeException
)
1479 ::rtl::OUString
sName( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.awt.UnoControl" ) );
1480 return Sequence
< ::rtl::OUString
>( &sName
, 1 );
1483 // ------------------------------------------------------------------------
1484 Reference
< XAccessibleContext
> SAL_CALL
UnoControl::getAccessibleContext( ) throw (RuntimeException
)
1486 // creation of the context will certainly require the SolarMutex ...
1487 ::vos::OGuard
aSolarGuard( Application::GetSolarMutex() );
1488 ::osl::MutexGuard
aGuard( GetMutex() );
1490 Reference
< XAccessibleContext
> xCurrentContext( maAccessibleContext
.get(), UNO_QUERY
);
1491 if ( !xCurrentContext
.is() )
1493 if ( !mbDesignMode
)
1494 { // in alive mode, use the AccessibleContext of the peer
1495 Reference
< XAccessible
> xPeerAcc( getPeer(), UNO_QUERY
);
1496 if ( xPeerAcc
.is() )
1497 xCurrentContext
= xPeerAcc
->getAccessibleContext( );
1500 // in design mode, use a fallback
1501 xCurrentContext
= ::toolkit::OAccessibleControlContext::create( this );
1503 DBG_ASSERT( xCurrentContext
.is(), "UnoControl::getAccessibleContext: invalid context (invalid peer?)!" );
1504 maAccessibleContext
= xCurrentContext
;
1506 // get notified when the context is disposed
1507 Reference
< XComponent
> xContextComp( xCurrentContext
, UNO_QUERY
);
1508 if ( xContextComp
.is() )
1509 xContextComp
->addEventListener( this );
1510 // In an ideal world, this is not necessary - there the object would be released as soon as it has been
1511 // disposed, and thus our weak reference would be empty, too.
1512 // But 'til this ideal world comes (means 'til we do never have any refcount/lifetime bugs anymore), we
1513 // need to listen for disposal and reset our weak reference then.
1516 return xCurrentContext
;
1519 void SAL_CALL
UnoControl::addModeChangeListener( const Reference
< XModeChangeListener
>& _rxListener
) throw (RuntimeException
)
1521 ::osl::MutexGuard
aGuard( GetMutex() );
1522 maModeChangeListeners
.addInterface( _rxListener
);
1525 void SAL_CALL
UnoControl::removeModeChangeListener( const Reference
< XModeChangeListener
>& _rxListener
) throw (RuntimeException
)
1527 ::osl::MutexGuard
aGuard( GetMutex() );
1528 maModeChangeListeners
.removeInterface( _rxListener
);
1531 void SAL_CALL
UnoControl::addModeChangeApproveListener( const Reference
< XModeChangeApproveListener
>& ) throw (NoSupportException
, RuntimeException
)
1533 throw NoSupportException( );
1536 void SAL_CALL
UnoControl::removeModeChangeApproveListener( const Reference
< XModeChangeApproveListener
>& ) throw (NoSupportException
, RuntimeException
)
1538 throw NoSupportException( );