1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_comphelper.hxx"
26 #include "comphelper/accessiblewrapper.hxx"
27 #include <com/sun/star/reflection/XProxyFactory.hpp>
28 #include <com/sun/star/accessibility/AccessibleEventId.hpp>
29 #include <com/sun/star/accessibility/AccessibleStateType.hpp>
30 
31 #include <algorithm>
32 
33 using namespace	::comphelper;
34 using namespace	::com::sun::star::accessibility;
35 using namespace	::com::sun::star::uno;
36 using namespace	::com::sun::star::lang;
37 
38 //.............................................................................
39 namespace comphelper
40 {
41 //.............................................................................
42 
43 	//=========================================================================
44 	//= OWrappedAccessibleChildrenManager
45 	//=========================================================================
46 	//--------------------------------------------------------------------
47 	struct RemoveEventListener
48 			: public ::std::unary_function< AccessibleMap::value_type, void >
49 	{
50 	private:
51 		Reference< XEventListener >	m_xListener;
52 
53 	public:
54 		RemoveEventListener( const Reference< XEventListener >& _rxListener )
55 			:m_xListener( _rxListener  )
56 		{
57 		}
58 
59 		void operator()( const AccessibleMap::value_type& _rMapEntry ) const
60 		{
61 			Reference< XComponent > xComp( _rMapEntry.first, UNO_QUERY );
62 			if ( xComp.is() )
63 				xComp->removeEventListener( m_xListener );
64 		}
65 	};
66 
67 	//--------------------------------------------------------------------
68 	struct DisposeMappedChild
69 			: public ::std::unary_function< AccessibleMap::value_type, void >
70 	{
71 		void operator()( const AccessibleMap::value_type& _rMapEntry ) const
72 		{
73             Reference< XComponent > xContextComponent;
74             if ( _rMapEntry.second.is() )
75                 xContextComponent = xContextComponent.query( _rMapEntry.second->getAccessibleContext() );
76 			if ( xContextComponent.is() )
77 				xContextComponent->dispose();
78 		}
79 	};
80 
81 	//-------------------------------------------------------------------------
82 	OWrappedAccessibleChildrenManager::OWrappedAccessibleChildrenManager( const Reference< XMultiServiceFactory >& _rxORB )
83 		:m_xORB( _rxORB )
84 		,m_bTransientChildren( sal_True )
85 	{
86 	}
87 
88 	//-------------------------------------------------------------------------
89 	OWrappedAccessibleChildrenManager::~OWrappedAccessibleChildrenManager( )
90 	{
91 	}
92 
93 	//-------------------------------------------------------------------------
94 	void OWrappedAccessibleChildrenManager::setTransientChildren( sal_Bool _bSet )
95 	{
96 		m_bTransientChildren = _bSet;
97 	}
98 
99 	//-------------------------------------------------------------------------
100 	void OWrappedAccessibleChildrenManager::setOwningAccessible( const Reference< XAccessible >& _rxAcc )
101 	{
102 		OSL_ENSURE( !m_aOwningAccessible.get().is(), "OWrappedAccessibleChildrenManager::setOwningAccessible: to be called only once!" );
103 		m_aOwningAccessible = WeakReference< XAccessible >( _rxAcc );
104 	}
105 
106 	//-------------------------------------------------------------------------
107 	void OWrappedAccessibleChildrenManager::removeFromCache( const Reference< XAccessible >& _rxKey )
108 	{
109 		AccessibleMap::iterator aRemovedPos = m_aChildrenMap.find( _rxKey );
110 		if ( m_aChildrenMap.end() != aRemovedPos )
111 		{	// it was cached
112 			// remove ourself as event listener
113 			RemoveEventListener aOperator( this );
114 			aOperator( *aRemovedPos );
115 			// and remove the entry from the map
116 			m_aChildrenMap.erase( aRemovedPos );
117 		}
118 	}
119 
120 	//-------------------------------------------------------------------------
121 	void OWrappedAccessibleChildrenManager::invalidateAll( )
122 	{
123 		// remove as event listener from the map elements
124 		::std::for_each( m_aChildrenMap.begin(), m_aChildrenMap.end(), RemoveEventListener( this ) );
125 		// clear the map
126 		AccessibleMap aMap;
127 		m_aChildrenMap.swap( aMap );
128 	}
129 
130 	//-------------------------------------------------------------------------
131 	Reference< XAccessible > OWrappedAccessibleChildrenManager::getAccessibleWrapperFor(
132 		const Reference< XAccessible >& _rxKey,	sal_Bool _bCreate )
133 	{
134 		Reference< XAccessible > xValue;
135 
136 		if( !_rxKey.is() )
137 		{
138 			// fprintf( stderr, "It was this path that was crashing stuff\n" );
139 			return xValue;
140 		}
141 
142 		// do we have this child in the cahce?
143 		AccessibleMap::const_iterator aPos = m_aChildrenMap.find( _rxKey );
144 		if ( m_aChildrenMap.end() != aPos )
145 		{
146 			xValue = aPos->second;
147 		}
148 		else if ( _bCreate )
149 		{	// not found in the cache, and allowed to create
150 			// -> new wrapper
151 			xValue = new OAccessibleWrapper( m_xORB, _rxKey, (Reference< XAccessible >)m_aOwningAccessible );
152 
153 			// see if we do cache children
154 			if ( !m_bTransientChildren )
155 			{
156 				if (!m_aChildrenMap.insert(
157                         AccessibleMap::value_type( _rxKey, xValue ) ).second)
158                 {
159                     OSL_ENSURE(
160                         false,
161                         "OWrappedAccessibleChildrenManager::"
162                             "getAccessibleWrapperFor: element was already"
163                             " inserted!" );
164                 }
165 
166 				// listen for disposals of inner children - this may happen when the inner context
167 				// is the owner for the inner children (it will dispose these children, and of course
168 				// not our wrapper for these children)
169 				Reference< XComponent > xComp( _rxKey, UNO_QUERY );
170 				if ( xComp.is() )
171 					xComp->addEventListener( this );
172 			}
173 		}
174 
175 		return xValue;
176 	}
177 
178 	//-------------------------------------------------------------------------
179 	void OWrappedAccessibleChildrenManager::dispose()
180 	{
181 		// dispose our children
182 		::std::for_each( m_aChildrenMap.begin(), m_aChildrenMap.end(), RemoveEventListener( this ) );
183 		::std::for_each( m_aChildrenMap.begin(), m_aChildrenMap.end(), DisposeMappedChild( ) );
184 		// clear our children
185 		AccessibleMap aMap;
186 		m_aChildrenMap.swap( aMap );
187 	}
188 
189 	//--------------------------------------------------------------------
190 	void OWrappedAccessibleChildrenManager::implTranslateChildEventValue( const Any& _rInValue, Any& _rOutValue )
191 	{
192 		_rOutValue.clear();
193 		Reference< XAccessible > xChild;
194 		if ( _rInValue >>= xChild )
195 			_rOutValue <<= getAccessibleWrapperFor( xChild, sal_True );
196 	}
197 
198 	//-------------------------------------------------------------------------
199 	void OWrappedAccessibleChildrenManager::translateAccessibleEvent( const AccessibleEventObject& _rEvent, AccessibleEventObject& _rTranslatedEvent )
200 	{
201 		// just in case we can't translate some of the values:
202 		_rTranslatedEvent.NewValue = _rEvent.NewValue;
203 		_rTranslatedEvent.OldValue = _rEvent.OldValue;
204 
205 		switch ( _rEvent.EventId )
206 		{
207 			case AccessibleEventId::CHILD:
208 			case AccessibleEventId::ACTIVE_DESCENDANT_CHANGED:
209 			case AccessibleEventId::CONTROLLED_BY_RELATION_CHANGED:
210 			case AccessibleEventId::CONTROLLER_FOR_RELATION_CHANGED:
211 			case AccessibleEventId::LABEL_FOR_RELATION_CHANGED:
212 			case AccessibleEventId::LABELED_BY_RELATION_CHANGED:
213 			case AccessibleEventId::CONTENT_FLOWS_FROM_RELATION_CHANGED:
214 			case AccessibleEventId::CONTENT_FLOWS_TO_RELATION_CHANGED:
215 				// these are events where both the old and the new value contain child references
216 				implTranslateChildEventValue( _rEvent.OldValue, _rTranslatedEvent.OldValue );
217 				implTranslateChildEventValue( _rEvent.NewValue, _rTranslatedEvent.NewValue );
218 				break;
219 
220             case AccessibleEventId::NAME_CHANGED:
221             case AccessibleEventId::DESCRIPTION_CHANGED:
222             case AccessibleEventId::ACTION_CHANGED:
223             case AccessibleEventId::STATE_CHANGED:
224             case AccessibleEventId::BOUNDRECT_CHANGED:
225             case AccessibleEventId::INVALIDATE_ALL_CHILDREN:
226             case AccessibleEventId::SELECTION_CHANGED:
227             case AccessibleEventId::VISIBLE_DATA_CHANGED:
228             case AccessibleEventId::VALUE_CHANGED:
229             case AccessibleEventId::MEMBER_OF_RELATION_CHANGED:
230             case AccessibleEventId::CARET_CHANGED:
231             case AccessibleEventId::TEXT_CHANGED:
232             case AccessibleEventId::HYPERTEXT_CHANGED:
233             case AccessibleEventId::TABLE_CAPTION_CHANGED:
234 			case AccessibleEventId::TABLE_COLUMN_DESCRIPTION_CHANGED:
235 			case AccessibleEventId::TABLE_COLUMN_HEADER_CHANGED:
236 			case AccessibleEventId::TABLE_MODEL_CHANGED:
237 			case AccessibleEventId::TABLE_ROW_DESCRIPTION_CHANGED:
238 			case AccessibleEventId::TABLE_ROW_HEADER_CHANGED:
239 			case AccessibleEventId::TABLE_SUMMARY_CHANGED:
240             // --> PB 2006-03-21 #130798# EventId TEXT_SELECTION_CHANGED was missed
241             // these Ids are also missed: SUB_WINDOW_OF_RELATION_CHANGED & TEXT_ATTRIBUTE_CHANGED
242             case AccessibleEventId::TEXT_SELECTION_CHANGED:
243             // <--
244                 // nothing to translate
245 				break;
246 
247 			default:
248 				OSL_ENSURE( sal_False, "OWrappedAccessibleChildrenManager::translateAccessibleEvent: unknown (or unexpected) event id!" );
249 				break;
250 		}
251 	}
252 
253 	//-------------------------------------------------------------------------
254 	void OWrappedAccessibleChildrenManager::handleChildNotification( const AccessibleEventObject& _rEvent )
255 	{
256 		if ( AccessibleEventId::INVALIDATE_ALL_CHILDREN == _rEvent.EventId )
257 		{	// clear our child map
258 			invalidateAll( );
259 		}
260 		else if ( AccessibleEventId::CHILD == _rEvent.EventId )
261 		{
262 			// check if the removed or replaced element is cached
263 			Reference< XAccessible > xRemoved;
264 			if ( _rEvent.OldValue >>= xRemoved )
265 				removeFromCache( xRemoved );
266 		}
267 	}
268 
269 	//--------------------------------------------------------------------
270 	void SAL_CALL OWrappedAccessibleChildrenManager::disposing( const EventObject& _rSource ) throw (RuntimeException)
271 	{
272 		// this should come from one of the inner XAccessible's of our children
273 		Reference< XAccessible > xSource( _rSource.Source, UNO_QUERY );
274 		AccessibleMap::iterator aDisposedPos = m_aChildrenMap.find( xSource );
275 #if OSL_DEBUG_LEVEL > 0
276         if ( m_aChildrenMap.end() == aDisposedPos )
277         {
278        		OSL_ENSURE( sal_False,
279 	    		"OWrappedAccessibleChildrenManager::disposing: where did this come from?" );
280             // helper for dignostics
281             Reference< XAccessible > xOwningAccessible( m_aOwningAccessible );
282             Reference< XAccessibleContext > xContext;
283             try
284             {
285                 if ( xOwningAccessible.is() )
286                     xContext = xOwningAccessible->getAccessibleContext();
287                 if ( xContext.is() )
288                 {
289                     ::rtl::OUString sName = xContext->getAccessibleName();
290                     ::rtl::OUString sDescription = xContext->getAccessibleDescription();
291 //                  sal_Int32 nPlaceYourBreakpointHere = 0;
292                 }
293             }
294             catch( const Exception& /*e*/ )
295             {
296                 // silent this, it's only diagnostics which failed
297             }
298         }
299 #endif
300 		if ( m_aChildrenMap.end() != aDisposedPos )
301 		{
302 			m_aChildrenMap.erase( aDisposedPos );
303 		}
304 	}
305 
306 	//=========================================================================
307 	//= OAccessibleWrapper (implementation)
308 	//=========================================================================
309 	//-------------------------------------------------------------------------
310 	OAccessibleWrapper::OAccessibleWrapper( const Reference< XMultiServiceFactory >& _rxORB,
311 			const Reference< XAccessible >& _rxInnerAccessible, const Reference< XAccessible >& _rxParentAccessible )
312 		:OAccessibleWrapper_Base( )
313 		,OComponentProxyAggregation( _rxORB, Reference< XComponent >( _rxInnerAccessible, UNO_QUERY ) )
314 		,m_xParentAccessible( _rxParentAccessible )
315 		,m_xInnerAccessible( _rxInnerAccessible )
316 	{
317 	}
318 
319 	//--------------------------------------------------------------------
320 	OAccessibleWrapper::~OAccessibleWrapper( )
321 	{
322 		if ( !m_rBHelper.bDisposed )
323 		{
324 			acquire();	// to prevent duplicate dtor calls
325 			dispose();
326 		}
327 	}
328 
329 	//--------------------------------------------------------------------
330 	IMPLEMENT_FORWARD_XTYPEPROVIDER2( OAccessibleWrapper, OComponentProxyAggregation, OAccessibleWrapper_Base )
331     IMPLEMENT_FORWARD_REFCOUNT( OAccessibleWrapper, OComponentProxyAggregation )
332 
333 	//--------------------------------------------------------------------
334 	Any OAccessibleWrapper::queryInterface( const Type& _rType ) throw (RuntimeException)
335 	{
336         // #111089# instead of the inner XAccessible the proxy XAccessible must be returned
337 		Any aReturn = OAccessibleWrapper_Base::queryInterface( _rType );
338 		if ( !aReturn.hasValue() )
339 			aReturn = OComponentProxyAggregation::queryInterface( _rType );
340 
341 		return aReturn;
342 	}
343 
344 	//--------------------------------------------------------------------
345 	Reference< XAccessibleContext > OAccessibleWrapper::getContextNoCreate( ) const
346 	{
347 		return (Reference< XAccessibleContext >)m_aContext;
348 	}
349 
350 	//--------------------------------------------------------------------
351 	OAccessibleContextWrapper* OAccessibleWrapper::createAccessibleContext( const Reference< XAccessibleContext >& _rxInnerContext )
352 	{
353 		return new OAccessibleContextWrapper( getORB(), _rxInnerContext, this, m_xParentAccessible );
354 	}
355 
356 	//--------------------------------------------------------------------
357 	Reference< XAccessibleContext > SAL_CALL OAccessibleWrapper::getAccessibleContext(  ) throw (RuntimeException)
358 	{
359 		// see if the context is still alive (we cache it)
360 		Reference< XAccessibleContext > xContext = (Reference< XAccessibleContext >)m_aContext;
361 		if ( !xContext.is() )
362 		{
363 			// create a new context
364 			Reference< XAccessibleContext > xInnerContext = m_xInnerAccessible->getAccessibleContext( );
365 			if ( xInnerContext.is() )
366 			{
367 				xContext = createAccessibleContext( xInnerContext );
368 				// cache it
369 				m_aContext = WeakReference< XAccessibleContext >( xContext );
370 			}
371 		}
372 
373 		return xContext;
374 	}
375 
376 	//=========================================================================
377 	//= OAccessibleWrapper (implementation)
378 	//=========================================================================
379 	//-------------------------------------------------------------------------
380 	OAccessibleContextWrapperHelper::OAccessibleContextWrapperHelper(
381 				const Reference< XMultiServiceFactory >& _rxORB,
382 				::cppu::OBroadcastHelper& _rBHelper,
383 				const Reference< XAccessibleContext >& _rxInnerAccessibleContext,
384 				const Reference< XAccessible >& _rxOwningAccessible,
385 				const Reference< XAccessible >& _rxParentAccessible )
386 		:OComponentProxyAggregationHelper( _rxORB, _rBHelper )
387 		,m_xInnerContext( _rxInnerAccessibleContext )
388 		,m_xOwningAccessible( _rxOwningAccessible )
389 		,m_xParentAccessible( _rxParentAccessible )
390 		,m_pChildMapper( NULL )
391 	{
392 		// initialize the mapper for our children
393 		m_pChildMapper = new OWrappedAccessibleChildrenManager( getORB() );
394 		m_pChildMapper->acquire();
395 
396 		// determine if we're allowed to cache children
397 		Reference< XAccessibleStateSet > xStates( m_xInnerContext->getAccessibleStateSet( ) );
398 		OSL_ENSURE( xStates.is(), "OAccessibleContextWrapperHelper::OAccessibleContextWrapperHelper: no inner state set!" );
399 		m_pChildMapper->setTransientChildren( !xStates.is() || xStates->contains( AccessibleStateType::MANAGES_DESCENDANTS) );
400 
401 		m_pChildMapper->setOwningAccessible( m_xOwningAccessible );
402 	}
403 
404 	//--------------------------------------------------------------------
405 	void OAccessibleContextWrapperHelper::aggregateProxy( oslInterlockedCount& _rRefCount, ::cppu::OWeakObject& _rDelegator )
406 	{
407 		Reference< XComponent > xInnerComponent( m_xInnerContext, UNO_QUERY );
408 		OSL_ENSURE( xInnerComponent.is(), "OComponentProxyAggregation::aggregateProxy: accessible is no XComponent!" );
409         if ( xInnerComponent.is() )
410 		    componentAggregateProxyFor( xInnerComponent, _rRefCount, _rDelegator );
411 
412 		// add as event listener to the inner context, because we want to multiplex the AccessibleEvents
413 		osl_incrementInterlockedCount( &_rRefCount );
414 		{
415 			Reference< XAccessibleEventBroadcaster > xBroadcaster( m_xInner, UNO_QUERY );
416 			if ( xBroadcaster.is() )
417 				xBroadcaster->addEventListener( this );
418 		}
419 		osl_decrementInterlockedCount( &_rRefCount );
420 	}
421 
422 	//--------------------------------------------------------------------
423 	OAccessibleContextWrapperHelper::~OAccessibleContextWrapperHelper( )
424 	{
425 		OSL_ENSURE( m_rBHelper.bDisposed, "OAccessibleContextWrapperHelper::~OAccessibleContextWrapperHelper: you should ensure (in your dtor) that the object is disposed!" );
426 
427 		m_pChildMapper->release();
428 		m_pChildMapper = NULL;
429 	}
430 
431 	//--------------------------------------------------------------------
432 	Any SAL_CALL OAccessibleContextWrapperHelper::queryInterface( const Type& _rType ) throw (RuntimeException)
433 	{
434 		Any aReturn = OComponentProxyAggregationHelper::queryInterface( _rType );
435 		if ( !aReturn.hasValue() )
436 			aReturn = OAccessibleContextWrapperHelper_Base::queryInterface( _rType );
437 		return aReturn;
438 	}
439 
440 	//--------------------------------------------------------------------
441 	IMPLEMENT_FORWARD_XTYPEPROVIDER2( OAccessibleContextWrapperHelper, OComponentProxyAggregationHelper, OAccessibleContextWrapperHelper_Base )
442 
443 	//--------------------------------------------------------------------
444 	sal_Int32 SAL_CALL OAccessibleContextWrapperHelper::getAccessibleChildCount(  ) throw (RuntimeException)
445 	{
446 		return m_xInnerContext->getAccessibleChildCount();
447 	}
448 
449 	//--------------------------------------------------------------------
450 	Reference< XAccessible > SAL_CALL OAccessibleContextWrapperHelper::getAccessibleChild( sal_Int32 i ) throw (IndexOutOfBoundsException, RuntimeException)
451 	{
452 		// get the child of the wrapped component
453 		Reference< XAccessible > xInnerChild = m_xInnerContext->getAccessibleChild( i );
454 		return m_pChildMapper->getAccessibleWrapperFor( xInnerChild );
455 	}
456 
457 	//--------------------------------------------------------------------
458 	Reference< XAccessibleRelationSet > SAL_CALL OAccessibleContextWrapperHelper::getAccessibleRelationSet(  ) throw (RuntimeException)
459 	{
460 		return m_xInnerContext->getAccessibleRelationSet();
461 			// TODO: if this relation set would contain relations to siblings, we would normally need
462 			// to wrap them, too ....
463 	}
464 
465 	//--------------------------------------------------------------------
466 	void SAL_CALL OAccessibleContextWrapperHelper::notifyEvent( const AccessibleEventObject& _rEvent ) throw (RuntimeException)
467 	{
468 #if OSL_DEBUG_LEVEL > 0
469 		if ( AccessibleEventId::STATE_CHANGED == _rEvent.EventId )
470 		{
471 			sal_Bool bChildTransienceChanged = sal_False;
472 			sal_Int16 nChangeState = 0;
473 			if ( _rEvent.OldValue >>= nChangeState )
474 				bChildTransienceChanged = bChildTransienceChanged || AccessibleStateType::MANAGES_DESCENDANTS == nChangeState;
475 			if ( _rEvent.NewValue >>= nChangeState )
476 				bChildTransienceChanged = bChildTransienceChanged || AccessibleStateType::MANAGES_DESCENDANTS == nChangeState;
477 			OSL_ENSURE( !bChildTransienceChanged, "OAccessibleContextWrapperHelper::notifyEvent: MANAGES_DESCENDANTS is not expected to change during runtime!" );
478 				// if this asserts, then we would need to update our m_bTransientChildren flag here,
479 				// as well as (potentially) our child cache
480 		}
481 #endif
482 		AccessibleEventObject aTranslatedEvent( _rEvent );
483 
484 		{
485 			::osl::MutexGuard aGuard( m_rBHelper.rMutex );
486 
487             // translate the event
488 			queryInterface( ::getCppuType( static_cast< Reference< XInterface >* >( NULL ) ) ) >>= aTranslatedEvent.Source;
489 			m_pChildMapper->translateAccessibleEvent( _rEvent, aTranslatedEvent );
490 
491             // see if any of these notifications affect our child manager
492 			m_pChildMapper->handleChildNotification( _rEvent );
493 
494             if ( aTranslatedEvent.NewValue == m_xInner )
495 				aTranslatedEvent.NewValue = makeAny(aTranslatedEvent.Source);
496 			if ( aTranslatedEvent.OldValue == m_xInner )
497 				aTranslatedEvent.OldValue = makeAny(aTranslatedEvent.Source);
498 		}
499 
500 		notifyTranslatedEvent( aTranslatedEvent );
501 	}
502 
503 	//--------------------------------------------------------------------
504 	void SAL_CALL OAccessibleContextWrapperHelper::dispose() throw( RuntimeException )
505 	{
506 		::osl::MutexGuard aGuard( m_rBHelper.rMutex );
507 
508 		// stop multiplexing events
509 		Reference< XAccessibleEventBroadcaster > xBroadcaster( m_xInner, UNO_QUERY );
510 		OSL_ENSURE( xBroadcaster.is(), "OAccessibleContextWrapperHelper::disposing(): inner context is no broadcaster!" );
511 		if ( xBroadcaster.is() )
512 			xBroadcaster->removeEventListener( this );
513 
514 		// dispose the child cache/map
515 		m_pChildMapper->dispose();
516 
517 		// let the base class dispose the inner component
518 		OComponentProxyAggregationHelper::dispose();
519 	}
520 
521 	//--------------------------------------------------------------------
522 	void SAL_CALL OAccessibleContextWrapperHelper::disposing( const EventObject& _rEvent )  throw (RuntimeException)
523 	{
524 		// simply disambiguate this
525 		OComponentProxyAggregationHelper::disposing( _rEvent );
526 	}
527 
528 	//====================================================================
529 	//= OAccessibleContextWrapper
530 	//====================================================================
531 	//--------------------------------------------------------------------
532 	IMPLEMENT_FORWARD_XINTERFACE2( OAccessibleContextWrapper, OAccessibleContextWrapper_CBase, OAccessibleContextWrapperHelper )
533 
534 	//--------------------------------------------------------------------
535 	IMPLEMENT_FORWARD_XTYPEPROVIDER2( OAccessibleContextWrapper, OAccessibleContextWrapper_CBase, OAccessibleContextWrapperHelper )
536 
537 	//--------------------------------------------------------------------
538 	OAccessibleContextWrapper::OAccessibleContextWrapper( const Reference< XMultiServiceFactory >& _rxORB,
539 			const Reference< XAccessibleContext >& _rxInnerAccessibleContext, const Reference< XAccessible >& _rxOwningAccessible,
540 			const Reference< XAccessible >& _rxParentAccessible )
541 		:OAccessibleContextWrapper_CBase( m_aMutex )
542 		,OAccessibleContextWrapperHelper( _rxORB, rBHelper, _rxInnerAccessibleContext, _rxOwningAccessible, _rxParentAccessible )
543 		,m_nNotifierClient( 0 )
544 	{
545 		aggregateProxy( m_refCount, *this );
546 	}
547 
548 	//--------------------------------------------------------------------
549 	OAccessibleContextWrapper::~OAccessibleContextWrapper()
550 	{
551 	}
552 
553 	//--------------------------------------------------------------------
554 	sal_Int32 SAL_CALL OAccessibleContextWrapper::getAccessibleChildCount(  ) throw (RuntimeException)
555 	{
556 		return OAccessibleContextWrapperHelper::getAccessibleChildCount();
557 	}
558 
559 	//--------------------------------------------------------------------
560 	Reference< XAccessible > SAL_CALL OAccessibleContextWrapper::getAccessibleChild( sal_Int32 i ) throw (IndexOutOfBoundsException, RuntimeException)
561 	{
562 		return OAccessibleContextWrapperHelper::getAccessibleChild( i );
563 	}
564 
565 	//--------------------------------------------------------------------
566 	Reference< XAccessible > SAL_CALL OAccessibleContextWrapper::getAccessibleParent(  ) throw (RuntimeException)
567 	{
568 		return m_xParentAccessible;
569 	}
570 
571 	//--------------------------------------------------------------------
572 	sal_Int32 SAL_CALL OAccessibleContextWrapper::getAccessibleIndexInParent(  ) throw (RuntimeException)
573 	{
574 		return m_xInnerContext->getAccessibleIndexInParent();
575 	}
576 
577 	//--------------------------------------------------------------------
578 	sal_Int16 SAL_CALL OAccessibleContextWrapper::getAccessibleRole(  ) throw (RuntimeException)
579 	{
580 		return m_xInnerContext->getAccessibleRole();
581 	}
582 
583 	//--------------------------------------------------------------------
584 	::rtl::OUString SAL_CALL OAccessibleContextWrapper::getAccessibleDescription(  ) throw (RuntimeException)
585 	{
586 		return m_xInnerContext->getAccessibleDescription();
587 	}
588 
589 	//--------------------------------------------------------------------
590 	::rtl::OUString SAL_CALL OAccessibleContextWrapper::getAccessibleName(  ) throw (RuntimeException)
591 	{
592 		return m_xInnerContext->getAccessibleName();
593 	}
594 
595 	//--------------------------------------------------------------------
596 	Reference< XAccessibleRelationSet > SAL_CALL OAccessibleContextWrapper::getAccessibleRelationSet(  ) throw (RuntimeException)
597 	{
598 		return OAccessibleContextWrapperHelper::getAccessibleRelationSet();
599 	}
600 
601 	//--------------------------------------------------------------------
602 	Reference< XAccessibleStateSet > SAL_CALL OAccessibleContextWrapper::getAccessibleStateSet(  ) throw (RuntimeException)
603 	{
604 		return m_xInnerContext->getAccessibleStateSet();
605 	}
606 
607 	//--------------------------------------------------------------------
608 	Locale SAL_CALL OAccessibleContextWrapper::getLocale(  ) throw (IllegalAccessibleComponentStateException, RuntimeException)
609 	{
610 		return m_xInnerContext->getLocale();
611 	}
612 
613 	//--------------------------------------------------------------------
614 	void OAccessibleContextWrapper::notifyTranslatedEvent( const AccessibleEventObject& _rEvent ) throw (RuntimeException)
615 	{
616 		if ( m_nNotifierClient )
617 			AccessibleEventNotifier::addEvent( m_nNotifierClient, _rEvent );
618 	}
619 
620 	//--------------------------------------------------------------------
621 	void SAL_CALL OAccessibleContextWrapper::addEventListener( const Reference< XAccessibleEventListener >& _rxListener ) throw (RuntimeException)
622 	{
623 		::osl::MutexGuard aGuard( m_aMutex );
624 		if ( !m_nNotifierClient )
625 			m_nNotifierClient = AccessibleEventNotifier::registerClient( );
626 		AccessibleEventNotifier::addEventListener( m_nNotifierClient, _rxListener );
627 	}
628 
629 	//--------------------------------------------------------------------
630 	void SAL_CALL OAccessibleContextWrapper::removeEventListener( const Reference< XAccessibleEventListener >& _rxListener ) throw (RuntimeException)
631 	{
632 		::osl::MutexGuard aGuard( m_aMutex );
633 		if ( m_nNotifierClient )
634 		{
635 			if ( 0 == AccessibleEventNotifier::removeEventListener( m_nNotifierClient, _rxListener ) )
636 			{
637 				AccessibleEventNotifier::TClientId nId( m_nNotifierClient );
638 				m_nNotifierClient = 0;
639 				AccessibleEventNotifier::revokeClient( nId );
640 			}
641 		}
642 	}
643 
644 	//--------------------------------------------------------------------
645 	void SAL_CALL OAccessibleContextWrapper::disposing()  throw (RuntimeException)
646 	{
647 		AccessibleEventNotifier::TClientId nClientId( 0 );
648 
649 		// --- <mutex lock> -----------------------------------------
650 		{
651 			::osl::MutexGuard aGuard( m_aMutex );
652 
653 			// prepare notifying our AccessibleListeners
654 			if ( m_nNotifierClient )
655 			{
656 				nClientId = m_nNotifierClient;
657 				m_nNotifierClient = 0;
658 			}
659 		}
660 		// --- </mutex lock> -----------------------------------------
661 
662 		// let the base class do
663 		OAccessibleContextWrapperHelper::dispose();
664 
665 		// notify the disposal
666 		if ( nClientId )
667 			AccessibleEventNotifier::revokeClientNotifyDisposing( nClientId, *this );
668 	}
669 
670 	//--------------------------------------------------------------------
671 	void SAL_CALL OAccessibleContextWrapper::dispose() throw( RuntimeException )
672 	{
673 		// simply disambiguate
674 		OComponentProxyAggregation_CBase::dispose();
675 	}
676 
677 //.............................................................................
678 }	// namespace accessibility
679 //.............................................................................
680