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_accessibility.hxx"
26 
27 // includes --------------------------------------------------------------
28 #include <accessibility/standard/vclxaccessibletoolbox.hxx>
29 #include <accessibility/standard/vclxaccessibletoolboxitem.hxx>
30 #include <toolkit/helper/convert.hxx>
31 
32 #include <unotools/accessiblestatesethelper.hxx>
33 #include <com/sun/star/accessibility/AccessibleEventId.hpp>
34 #include <com/sun/star/accessibility/AccessibleStateType.hpp>
35 #include <com/sun/star/lang/XUnoTunnel.hpp>
36 #include <com/sun/star/lang/XUnoTunnel.hpp>
37 #include <tools/debug.hxx>
38 #include <vcl/toolbox.hxx>
39 #include <comphelper/accessiblewrapper.hxx>
40 #include <comphelper/processfactory.hxx>
41 
42 using namespace ::comphelper;
43 using namespace ::com::sun::star;
44 using namespace ::com::sun::star::uno;
45 using namespace ::com::sun::star::lang;
46 using namespace ::com::sun::star::accessibility;
47 
48 namespace
49 {
50 	// =========================================================================
51 	// = OToolBoxWindowItemContext
52 	// =========================================================================
53 	/** XAccessibleContext implementation for a toolbox item which is represented by a VCL Window
54 	*/
55 	class OToolBoxWindowItemContext : public OAccessibleContextWrapper
56 	{
57 		sal_Int32 m_nIndexInParent;
58 	public:
OToolBoxWindowItemContext(sal_Int32 _nIndexInParent,const::com::sun::star::uno::Reference<::com::sun::star::lang::XMultiServiceFactory> & _rxORB,const::com::sun::star::uno::Reference<::com::sun::star::accessibility::XAccessibleContext> & _rxInnerAccessibleContext,const::com::sun::star::uno::Reference<::com::sun::star::accessibility::XAccessible> & _rxOwningAccessible,const::com::sun::star::uno::Reference<::com::sun::star::accessibility::XAccessible> & _rxParentAccessible)59 		OToolBoxWindowItemContext(sal_Int32 _nIndexInParent,
60 			const ::com::sun::star::uno::Reference< ::com::sun::star::lang::XMultiServiceFactory >& _rxORB,
61 			const ::com::sun::star::uno::Reference< ::com::sun::star::accessibility::XAccessibleContext >& _rxInnerAccessibleContext,
62 			const ::com::sun::star::uno::Reference< ::com::sun::star::accessibility::XAccessible >& _rxOwningAccessible,
63 			const ::com::sun::star::uno::Reference< ::com::sun::star::accessibility::XAccessible >& _rxParentAccessible
64 			) : OAccessibleContextWrapper(
65 			_rxORB,
66 			_rxInnerAccessibleContext,
67 			_rxOwningAccessible,
68 			_rxParentAccessible)
69 			,m_nIndexInParent(_nIndexInParent)
70 		{
71 		}
72 		virtual sal_Int32 SAL_CALL getAccessibleIndexInParent(  ) throw (::com::sun::star::uno::RuntimeException);
73 	};
74 
75 	// -------------------------------------------------------------------------
getAccessibleIndexInParent()76 	sal_Int32 SAL_CALL OToolBoxWindowItemContext::getAccessibleIndexInParent(  ) throw (::com::sun::star::uno::RuntimeException)
77 	{
78 		::osl::MutexGuard aGuard( m_aMutex );
79 		return m_nIndexInParent;
80 	}
81 
82 	// =========================================================================
83 	// = OToolBoxWindowItem
84 	// =========================================================================
85 	typedef ::cppu::ImplHelper1	<	XUnoTunnel
86 								>	OToolBoxWindowItem_Base;
87 
88 	/** XAccessible implementation for a toolbox item which is represented by a VCL Window
89 	*/
90 	class OToolBoxWindowItem
91 			:public OAccessibleWrapper
92 			,public OToolBoxWindowItem_Base
93 	{
94 	private:
95 		sal_Int32 m_nIndexInParent;
96 
97 	public:
getIndexInParent() const98 		inline sal_Int32	getIndexInParent() const					{ return m_nIndexInParent; }
setIndexInParent(sal_Int32 _nNewIndex)99 		inline void			setIndexInParent( sal_Int32 _nNewIndex )	{ m_nIndexInParent = _nNewIndex; }
100 
101 		static sal_Bool isWindowItem( const Reference< XAccessible >& _rxAcc, OToolBoxWindowItem** /* [out] */ _ppImplementation = NULL );
102 
103 	public:
OToolBoxWindowItem(sal_Int32 _nIndexInParent,const::com::sun::star::uno::Reference<::com::sun::star::lang::XMultiServiceFactory> & _rxORB,const::com::sun::star::uno::Reference<::com::sun::star::accessibility::XAccessible> & _rxInnerAccessible,const::com::sun::star::uno::Reference<::com::sun::star::accessibility::XAccessible> & _rxParentAccessible)104 		OToolBoxWindowItem(sal_Int32 _nIndexInParent,
105 			const ::com::sun::star::uno::Reference< ::com::sun::star::lang::XMultiServiceFactory >& _rxORB,
106 			const ::com::sun::star::uno::Reference< ::com::sun::star::accessibility::XAccessible >& _rxInnerAccessible,
107 			const ::com::sun::star::uno::Reference< ::com::sun::star::accessibility::XAccessible >& _rxParentAccessible
108 			) : OAccessibleWrapper(
109 			_rxORB,
110 			_rxInnerAccessible,
111 			_rxParentAccessible)
112 			,m_nIndexInParent(_nIndexInParent)
113 		{
114 		}
115 
116 	protected:
117 		// XInterface
118 		DECLARE_XINTERFACE( )
119 		DECLARE_XTYPEPROVIDER( )
120 
121 		// OAccessibleWrapper
122 		virtual OAccessibleContextWrapper* createAccessibleContext(
123 				const ::com::sun::star::uno::Reference< ::com::sun::star::accessibility::XAccessibleContext >& _rxInnerContext
124 			);
125 
126 		// XUnoTunnel
127 		virtual sal_Int64 SAL_CALL getSomething( const Sequence< sal_Int8 >& aIdentifier ) throw (RuntimeException);
128 		static Sequence< sal_Int8 > getUnoTunnelImplementationId();
129 	};
130 
131 	// -------------------------------------------------------------------------
IMPLEMENT_FORWARD_XINTERFACE2(OToolBoxWindowItem,OAccessibleWrapper,OToolBoxWindowItem_Base)132 	IMPLEMENT_FORWARD_XINTERFACE2( OToolBoxWindowItem, OAccessibleWrapper, OToolBoxWindowItem_Base )
133 	IMPLEMENT_FORWARD_XTYPEPROVIDER2( OToolBoxWindowItem, OAccessibleWrapper, OToolBoxWindowItem_Base )
134 
135 	// -------------------------------------------------------------------------
136 	OAccessibleContextWrapper* OToolBoxWindowItem::createAccessibleContext(
137 			const Reference< XAccessibleContext >& _rxInnerContext )
138 	{
139 		return new OToolBoxWindowItemContext( m_nIndexInParent,getORB(), _rxInnerContext, this, getParent() );
140 	}
141 
142 	//--------------------------------------------------------------------
isWindowItem(const Reference<XAccessible> & _rxAcc,OToolBoxWindowItem ** _ppImplementation)143 	sal_Bool OToolBoxWindowItem::isWindowItem( const Reference< XAccessible >& _rxAcc, OToolBoxWindowItem** /* [out] */ _ppImplementation )
144 	{
145 		OToolBoxWindowItem* pImplementation = NULL;
146 
147 		Reference< XUnoTunnel > xTunnel( _rxAcc, UNO_QUERY );
148 		if ( xTunnel.is() )
149 			pImplementation = reinterpret_cast< OToolBoxWindowItem* >( xTunnel->getSomething( getUnoTunnelImplementationId() ) );
150 
151 		if ( _ppImplementation )
152 			*_ppImplementation = pImplementation;
153 
154 		return NULL != pImplementation;
155 	}
156 
157 	//--------------------------------------------------------------------
getUnoTunnelImplementationId()158 	Sequence< sal_Int8 > OToolBoxWindowItem::getUnoTunnelImplementationId()
159 	{
160 		static ::cppu::OImplementationId * pId = 0;
161 		if (! pId)
162 		{
163 			::osl::MutexGuard aGuard( ::osl::Mutex::getGlobalMutex() );
164 			if (! pId)
165 			{
166 				static ::cppu::OImplementationId aId;
167 				pId = &aId;
168 			}
169 		}
170 		return pId->getImplementationId();
171 	}
172 
173 	//--------------------------------------------------------------------
getSomething(const Sequence<sal_Int8> & _rId)174 	sal_Int64 SAL_CALL OToolBoxWindowItem::getSomething( const Sequence< sal_Int8 >& _rId ) throw (RuntimeException)
175 	{
176 		if	(	( 16 == _rId.getLength() )
177 			&&	( 0 == rtl_compareMemory( getUnoTunnelImplementationId().getConstArray(), _rId.getConstArray(), 16 ) )
178 			)
179 			return reinterpret_cast< sal_Int64>( this );
180 
181 		return 0;
182 	}
183 }
184 
DBG_NAME(VCLXAccessibleToolBox)185 DBG_NAME(VCLXAccessibleToolBox)
186 
187 // -----------------------------------------------------------------------------
188 // VCLXAccessibleToolBox
189 // -----------------------------------------------------------------------------
190 VCLXAccessibleToolBox::VCLXAccessibleToolBox( VCLXWindow* pVCLXWindow ) :
191 
192 	VCLXAccessibleComponent( pVCLXWindow )
193 
194 {
195 	DBG_CTOR(VCLXAccessibleToolBox,NULL);
196 }
197 // -----------------------------------------------------------------------------
~VCLXAccessibleToolBox()198 VCLXAccessibleToolBox::~VCLXAccessibleToolBox()
199 {
200 	DBG_DTOR(VCLXAccessibleToolBox,NULL);
201 }
202 // -----------------------------------------------------------------------------
GetItem_Impl(sal_Int32 _nPos,bool _bMustHaveFocus)203 VCLXAccessibleToolBoxItem* VCLXAccessibleToolBox::GetItem_Impl( sal_Int32 _nPos, bool _bMustHaveFocus )
204 {
205 	VCLXAccessibleToolBoxItem* pItem = NULL;
206 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
207 	if ( pToolBox && ( !_bMustHaveFocus || pToolBox->HasFocus() ) )
208 	{
209 		ToolBoxItemsMap::iterator aIter = m_aAccessibleChildren.find( _nPos );
210 		// returns only toolbox buttons, not windows
211 		if ( aIter != m_aAccessibleChildren.end() && aIter->second.is())
212 			pItem = static_cast< VCLXAccessibleToolBoxItem* >( aIter->second.get() );
213 	}
214 
215 	return pItem;
216 }
217 // -----------------------------------------------------------------------------
218 
UpdateFocus_Impl()219 void VCLXAccessibleToolBox::UpdateFocus_Impl()
220 {
221 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
222 	if( !pToolBox )
223 		return;
224 
225 	// submit events only if toolbox has the focus to avoid sending events due to mouse move
226 	sal_Bool bHasFocus = sal_False;
227 	if ( pToolBox->HasFocus() )
228 		bHasFocus = sal_True;
229 	else
230 	{
231 		// check for subtoolbar, i.e. check if our parent is a toolbar
232 		ToolBox* pToolBoxParent = dynamic_cast< ToolBox* >( pToolBox->GetParent() );
233 		// subtoolbars never get the focus as key input is just forwarded, so check if the parent toolbar has it
234 		if ( pToolBoxParent && pToolBoxParent->HasFocus() )
235 			bHasFocus = sal_True;
236 	}
237 
238 	if ( bHasFocus )
239 	{
240 		sal_uInt16 nHighlightItemId = pToolBox->GetHighlightItemId();
241 		sal_uInt16 nFocusCount = 0;
242 		for ( ToolBoxItemsMap::iterator aIter = m_aAccessibleChildren.begin();
243 			aIter != m_aAccessibleChildren.end(); ++aIter )
244 		{
245 			sal_uInt16 nItemId = pToolBox->GetItemId( (sal_uInt16)aIter->first );
246 
247 			if ( aIter->second.is() )
248 			{
249 				VCLXAccessibleToolBoxItem* pItem =
250 					static_cast< VCLXAccessibleToolBoxItem* >( aIter->second.get() );
251 				if ( pItem->HasFocus() && nItemId != nHighlightItemId )
252 				{
253 					// reset the old focused item
254 					pItem->SetFocus( sal_False );
255 					nFocusCount++;
256 				}
257 				if ( nItemId == nHighlightItemId )
258 				{
259 					// set the new focused item
260 					pItem->SetFocus( sal_True );
261 					nFocusCount++;
262 				}
263 			}
264 		// both items changed?
265 		if ( nFocusCount > 1 )
266 			break;
267 		}
268 	}
269 }
270 // -----------------------------------------------------------------------------
ReleaseFocus_Impl(sal_Int32 _nPos)271 void VCLXAccessibleToolBox::ReleaseFocus_Impl( sal_Int32 _nPos )
272 {
273 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
274 	if ( pToolBox ) // #107124#, do not check for focus because this message is also handled in losefocus
275 	{
276 		ToolBoxItemsMap::iterator aIter = m_aAccessibleChildren.find( _nPos );
277 		if ( aIter != m_aAccessibleChildren.end() && aIter->second.is() )
278 		{
279 			VCLXAccessibleToolBoxItem* pItem =
280 				static_cast< VCLXAccessibleToolBoxItem* >( aIter->second.get() );
281 			if ( pItem->HasFocus() )
282 				pItem->SetFocus( sal_False );
283 		}
284 	}
285 }
286 // -----------------------------------------------------------------------------
UpdateChecked_Impl(sal_Int32 _nPos)287 void VCLXAccessibleToolBox::UpdateChecked_Impl( sal_Int32 _nPos )
288 {
289 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
290 	if ( pToolBox )
291 	{
292 		sal_uInt16 nFocusId = pToolBox->GetItemId( (sal_uInt16)_nPos );
293 		VCLXAccessibleToolBoxItem* pFocusItem = NULL;
294 
295 		for ( ToolBoxItemsMap::iterator aIter = m_aAccessibleChildren.begin();
296 			aIter != m_aAccessibleChildren.end(); ++aIter )
297 		{
298 				sal_uInt16 nItemId = pToolBox->GetItemId( (sal_uInt16)aIter->first );
299 
300 				VCLXAccessibleToolBoxItem* pItem =
301 					static_cast< VCLXAccessibleToolBoxItem* >( aIter->second.get() );
302 				pItem->SetChecked( pToolBox->IsItemChecked( nItemId ) );
303 				if ( nItemId == nFocusId )
304 					pFocusItem = pItem;
305 		}
306 		// Solution: If the position is not a child item, the focus should not be called
307 		if ( pFocusItem && (sal_uInt16)_nPos != TOOLBOX_ITEM_NOTFOUND )
308 			pFocusItem->SetFocus( sal_True );
309 	}
310 }
311 // -----------------------------------------------------------------------------
UpdateIndeterminate_Impl(sal_Int32 _nPos)312 void VCLXAccessibleToolBox::UpdateIndeterminate_Impl( sal_Int32 _nPos )
313 {
314 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
315 	if ( pToolBox )
316 	{
317 		sal_uInt16 nItemId = pToolBox->GetItemId( (sal_uInt16)_nPos );
318 
319 		ToolBoxItemsMap::iterator aIter = m_aAccessibleChildren.find( _nPos );
320 		if ( aIter != m_aAccessibleChildren.end() && aIter->second.is() )
321 		{
322 			VCLXAccessibleToolBoxItem* pItem =
323 				static_cast< VCLXAccessibleToolBoxItem* >( aIter->second.get() );
324 			if ( pItem )
325 				pItem->SetIndeterminate( pToolBox->GetItemState( nItemId ) == STATE_DONTKNOW );
326 		}
327 	}
328 }
329 // -----------------------------------------------------------------------------
implReleaseToolboxItem(ToolBoxItemsMap::iterator & _rMapPos,bool _bNotifyRemoval,bool _bDispose)330 void VCLXAccessibleToolBox::implReleaseToolboxItem( ToolBoxItemsMap::iterator& _rMapPos,
331 		bool _bNotifyRemoval, bool _bDispose )
332 {
333 	Reference< XAccessible > xItemAcc( _rMapPos->second );
334 	if ( !xItemAcc.is() )
335 		return;
336 
337 	if ( _bNotifyRemoval )
338 	{
339 		NotifyAccessibleEvent( AccessibleEventId::CHILD, makeAny( xItemAcc ), Any() );
340 	}
341 
342 	OToolBoxWindowItem* pWindowItem = NULL;
343 	if ( !OToolBoxWindowItem::isWindowItem( xItemAcc, &pWindowItem ) )
344 	{
345 		static_cast< VCLXAccessibleToolBoxItem* >( xItemAcc.get() )->ReleaseToolBox();
346 		if ( _bDispose )
347 			::comphelper::disposeComponent( xItemAcc );
348 	}
349 	else
350 	{
351 		if ( _bDispose )
352 		{
353 			if ( pWindowItem )
354 			{
355 				Reference< XAccessibleContext > xContext( pWindowItem->getContextNoCreate() );
356 				::comphelper::disposeComponent( xContext );
357 			}
358 		}
359 	}
360 }
361 
362 // -----------------------------------------------------------------------------
UpdateItem_Impl(sal_Int32 _nPos,sal_Bool _bItemAdded)363 void VCLXAccessibleToolBox::UpdateItem_Impl( sal_Int32 _nPos, sal_Bool _bItemAdded )
364 {
365 	if ( _nPos < sal_Int32( m_aAccessibleChildren.size() ) )
366 	{
367 		UpdateAllItems_Impl();
368 		return;
369 	}
370 
371 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
372 	if ( pToolBox )
373 	{
374 		if ( !_bItemAdded )
375 		{	// the item was removed
376 			// -> destroy the old item
377 			ToolBoxItemsMap::iterator aItemPos = m_aAccessibleChildren.find( _nPos );
378 			if ( m_aAccessibleChildren.end() != aItemPos )
379 			{
380 				implReleaseToolboxItem( aItemPos, true, true );
381 				m_aAccessibleChildren.erase( aItemPos );
382 			}
383 		}
384 
385 		// adjust the "index-in-parent"s
386 		ToolBoxItemsMap::iterator aIndexAdjust = m_aAccessibleChildren.upper_bound( _nPos );
387 		while ( m_aAccessibleChildren.end() != aIndexAdjust )
388 		{
389 			Reference< XAccessible > xItemAcc( aIndexAdjust->second );
390 
391 			OToolBoxWindowItem* pWindowItem = NULL;
392 			if ( !OToolBoxWindowItem::isWindowItem( xItemAcc, &pWindowItem ) )
393 			{
394 				VCLXAccessibleToolBoxItem* pItem = static_cast< VCLXAccessibleToolBoxItem* >( xItemAcc.get() );
395 				if ( pItem )
396 				{
397 					sal_Int32 nIndex = pItem->getIndexInParent( );
398 					nIndex += _bItemAdded ? +1 : -1;
399 					pItem->setIndexInParent( nIndex );
400 				}
401 			}
402 			else
403 			{
404 				if ( pWindowItem )
405 				{
406 					sal_Int32 nIndex = pWindowItem->getIndexInParent( );
407 					nIndex += _bItemAdded ? +1 : -1;
408 					pWindowItem->setIndexInParent( nIndex );
409 				}
410 			}
411 
412 			++aIndexAdjust;
413 		}
414 
415 		if ( _bItemAdded )
416 		{
417 			// TODO: we should make this dependent on the existence of event listeners
418 			// with the current implementation, we always create accessible object
419 			Any aNewChild = makeAny( getAccessibleChild( (sal_Int32)_nPos ) );
420 			NotifyAccessibleEvent( AccessibleEventId::CHILD, Any(), aNewChild );
421 		}
422 	}
423 }
424 // -----------------------------------------------------------------------------
UpdateAllItems_Impl()425 void VCLXAccessibleToolBox::UpdateAllItems_Impl()
426 {
427 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
428 	if ( pToolBox )
429 	{
430 		// deregister the old items
431 		for ( ToolBoxItemsMap::iterator aIter = m_aAccessibleChildren.begin();
432 			aIter != m_aAccessibleChildren.end(); ++aIter )
433 		{
434 			implReleaseToolboxItem( aIter, true, true );
435 		}
436 		m_aAccessibleChildren.clear();
437 
438 		// register the new items
439 		sal_uInt16 i, nCount = pToolBox->GetItemCount();
440 		for ( i = 0; i < nCount; ++i )
441 		{
442 			Any aNewValue;
443 			aNewValue <<= getAccessibleChild( (sal_Int32)i );;
444 			NotifyAccessibleEvent( AccessibleEventId::CHILD, Any(), aNewValue );
445 		}
446 	}
447 }
448 
449 // -----------------------------------------------------------------------------
450 
UpdateCustomPopupItemp_Impl(Window * pWindow,bool bOpen)451 void VCLXAccessibleToolBox::UpdateCustomPopupItemp_Impl( Window* pWindow, bool bOpen )
452 {
453 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
454 	if( pWindow && pToolBox )
455 	{
456 		Reference< XAccessible > xChild( pWindow->GetAccessible() );
457 		if( xChild.is() )
458 		{
459 			Reference< XAccessible > xChildItem( getAccessibleChild( static_cast< sal_Int32 >( pToolBox->GetItemPos( pToolBox->GetDownItemId() ) ) ) );
460 			VCLXAccessibleToolBoxItem* pItem = static_cast< VCLXAccessibleToolBoxItem* >( xChildItem.get() );
461 
462 			pItem->SetChild( xChild );
463 			pItem->NotifyChildEvent( xChild, bOpen );
464 		}
465 	}
466 }
467 
468 // -----------------------------------------------------------------------------
UpdateItemName_Impl(sal_Int32 _nPos)469 void VCLXAccessibleToolBox::UpdateItemName_Impl( sal_Int32 _nPos )
470 {
471 	VCLXAccessibleToolBoxItem* pItem = GetItem_Impl( _nPos, false );
472 	if ( pItem )
473 		pItem->NameChanged();
474 }
475 // -----------------------------------------------------------------------------
UpdateItemEnabled_Impl(sal_Int32 _nPos)476 void VCLXAccessibleToolBox::UpdateItemEnabled_Impl( sal_Int32 _nPos )
477 {
478 	VCLXAccessibleToolBoxItem* pItem = GetItem_Impl( _nPos, false );
479 	if ( pItem )
480 		pItem->ToggleEnableState();
481 }
482 // -----------------------------------------------------------------------------
HandleSubToolBarEvent(const VclWindowEvent & rVclWindowEvent,bool _bShow)483 void VCLXAccessibleToolBox::HandleSubToolBarEvent( const VclWindowEvent& rVclWindowEvent, bool _bShow )
484 {
485 	Window* pChildWindow = (Window *) rVclWindowEvent.GetData();
486 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
487 	if ( pChildWindow
488 		&& pToolBox
489 		&& pToolBox == pChildWindow->GetParent()
490 		&& pChildWindow->GetType() == WINDOW_TOOLBOX )
491 	{
492 		sal_Int32 nIndex = pToolBox->GetItemPos( pToolBox->GetCurItemId() );
493 		Reference< XAccessible > xItem = getAccessibleChild( nIndex );
494 		if ( xItem.is() )
495 		{
496 			Reference< XAccessible > xChild = pChildWindow->GetAccessible();
497 			VCLXAccessibleToolBoxItem* pItem =
498 				static_cast< VCLXAccessibleToolBoxItem* >( xItem.get() );
499 			pItem->SetChild( xChild );
500 			pItem->NotifyChildEvent( xChild, _bShow );
501 		}
502 	}
503 }
504 // -----------------------------------------------------------------------------
ReleaseSubToolBox(ToolBox * _pSubToolBox)505 void VCLXAccessibleToolBox::ReleaseSubToolBox( ToolBox* _pSubToolBox )
506 {
507 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
508 	if ( pToolBox )
509 	{
510 		sal_Int32 nIndex = pToolBox->GetItemPos( pToolBox->GetCurItemId() );
511 		Reference< XAccessible > xItem = getAccessibleChild( nIndex );
512 		if ( xItem.is() )
513 		{
514 			Reference< XAccessible > xChild = _pSubToolBox->GetAccessible();
515 			VCLXAccessibleToolBoxItem* pItem =
516 				static_cast< VCLXAccessibleToolBoxItem* >( xItem.get() );
517 			if ( pItem->GetChild() == xChild )
518 			{
519 				pItem->SetChild( Reference< XAccessible >() );
520 				pItem->NotifyChildEvent( xChild, false );
521 			}
522 		}
523 	}
524 }
525 // -----------------------------------------------------------------------------
FillAccessibleStateSet(utl::AccessibleStateSetHelper & rStateSet)526 void VCLXAccessibleToolBox::FillAccessibleStateSet( utl::AccessibleStateSetHelper& rStateSet )
527 {
528 	VCLXAccessibleComponent::FillAccessibleStateSet( rStateSet );
529 
530 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
531 	if ( pToolBox )
532 	{
533 		rStateSet.AddState( AccessibleStateType::FOCUSABLE );
534 		if ( pToolBox->IsHorizontal() )
535 			rStateSet.AddState( AccessibleStateType::HORIZONTAL );
536 		else
537 			rStateSet.AddState( AccessibleStateType::VERTICAL );
538 	}
539 }
540 // -----------------------------------------------------------------------------
ProcessWindowEvent(const VclWindowEvent & rVclWindowEvent)541 void VCLXAccessibleToolBox::ProcessWindowEvent( const VclWindowEvent& rVclWindowEvent )
542 {
543 	// to prevent an early release of the toolbox (VCLEVENT_OBJECT_DYING)
544 	Reference< XAccessibleContext > xTemp = this;
545 
546 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
547 	switch ( rVclWindowEvent.GetId() )
548 	{
549 		case VCLEVENT_TOOLBOX_CLICK:
550 		case VCLEVENT_TOOLBOX_SELECT:
551 		{
552 			if ( rVclWindowEvent.GetData() )
553 			{
554 				UpdateChecked_Impl( (sal_Int32)(sal_IntPtr)rVclWindowEvent.GetData() );
555 				UpdateIndeterminate_Impl( (sal_Int32)(sal_IntPtr)rVclWindowEvent.GetData() );
556 			}
557 			else if( pToolBox->GetItemPos(pToolBox->GetCurItemId()) != TOOLBOX_ITEM_NOTFOUND )
558 			{
559 				UpdateChecked_Impl( pToolBox->GetItemPos(pToolBox->GetCurItemId()) );
560 				UpdateIndeterminate_Impl( pToolBox->GetItemPos(pToolBox->GetCurItemId()) );
561 			}
562 			break;
563 		}
564 		case VCLEVENT_TOOLBOX_DOUBLECLICK:
565 		case VCLEVENT_TOOLBOX_ACTIVATE:
566 		case VCLEVENT_TOOLBOX_DEACTIVATE:
567 		//case VCLEVENT_TOOLBOX_SELECT:
568 			break;
569 		// IA2 CWS. MT: Still using VCLEVENT_TOOLBOX_CLICK, see comment in vcl/source/window/toolbox2.cxx
570 		/*
571 		case VCLEVENT_TOOLBOX_ITEMUPDATED:
572 		{
573 			if ( rVclWindowEvent.GetData() )
574 			{
575 				UpdateChecked_Impl( TOOLBOX_ITEM_NOTFOUND );
576 				UpdateIndeterminate_Impl( (sal_Int32)rVclWindowEvent.GetData() );
577 			}
578 		break;
579 		}
580 		*/
581 		case VCLEVENT_TOOLBOX_HIGHLIGHT:
582 			UpdateFocus_Impl();
583 			break;
584 
585 		case VCLEVENT_TOOLBOX_HIGHLIGHTOFF:
586 			ReleaseFocus_Impl( (sal_Int32)(sal_IntPtr)rVclWindowEvent.GetData() );
587 			break;
588 
589 		case VCLEVENT_TOOLBOX_ITEMADDED :
590 //			UpdateItem_Impl( (sal_Int32)(sal_IntPtr)rVclWindowEvent.GetData(), VCLEVENT_TOOLBOX_ITEMADDED == rVclWindowEvent.GetId() );
591 			UpdateItem_Impl( (sal_Int32)(sal_IntPtr)rVclWindowEvent.GetData(), sal_True );
592 			break;
593 
594 		case VCLEVENT_TOOLBOX_ITEMREMOVED :
595 		case VCLEVENT_TOOLBOX_ALLITEMSCHANGED :
596 		{
597 			UpdateAllItems_Impl();
598 			break;
599 		}
600 
601 		case VCLEVENT_TOOLBOX_ITEMWINDOWCHANGED:
602 		{
603 			sal_Int32 nPos = (sal_Int32)(sal_IntPtr)rVclWindowEvent.GetData();
604 			ToolBoxItemsMap::iterator aAccessiblePos( m_aAccessibleChildren.find( nPos ) );
605 			if ( m_aAccessibleChildren.end() != aAccessiblePos )
606 			{
607 				implReleaseToolboxItem( aAccessiblePos, false, true );
608 				m_aAccessibleChildren.erase (aAccessiblePos);
609 			}
610 
611 			Any aNewValue;
612 			aNewValue <<= getAccessibleChild(nPos);
613 			NotifyAccessibleEvent( AccessibleEventId::CHILD, Any(), aNewValue );
614 			break;
615 		}
616 		case VCLEVENT_TOOLBOX_ITEMTEXTCHANGED :
617 			UpdateItemName_Impl( (sal_Int32)(sal_IntPtr)rVclWindowEvent.GetData() );
618 			break;
619 
620 		case VCLEVENT_TOOLBOX_ITEMENABLED :
621 		case VCLEVENT_TOOLBOX_ITEMDISABLED :
622 		{
623 			UpdateItemEnabled_Impl( (sal_Int32)(sal_IntPtr)rVclWindowEvent.GetData() );
624 			break;
625 		}
626 
627 		case VCLEVENT_DROPDOWN_OPEN:
628 		case VCLEVENT_DROPDOWN_CLOSE:
629 		{
630 			UpdateCustomPopupItemp_Impl( static_cast< Window* >( rVclWindowEvent.GetData() ), rVclWindowEvent.GetId() == VCLEVENT_DROPDOWN_OPEN );
631 			break;
632 		}
633 
634 		case VCLEVENT_OBJECT_DYING :
635 		{
636 			// if this toolbox is a subtoolbox, we have to release it from its parent
637 			ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
638 			if ( pToolBox && pToolBox->GetParent() &&
639 				pToolBox->GetParent()->GetType() == WINDOW_TOOLBOX )
640 			{
641 				VCLXAccessibleToolBox* pParent = static_cast< VCLXAccessibleToolBox* >(
642 					pToolBox->GetParent()->GetAccessible()->getAccessibleContext().get() );
643 				if ( pParent )
644 					pParent->ReleaseSubToolBox( pToolBox );
645 			}
646 
647 			// dispose all items
648 			for ( ToolBoxItemsMap::iterator aIter = m_aAccessibleChildren.begin();
649 				aIter != m_aAccessibleChildren.end(); ++aIter )
650 			{
651 				implReleaseToolboxItem( aIter, false, true );
652 			}
653 			m_aAccessibleChildren.clear();
654 
655 			// !!! no break to call base class
656 		}
657 
658 		default:
659 			VCLXAccessibleComponent::ProcessWindowEvent( rVclWindowEvent );
660 	}
661 }
662 // -----------------------------------------------------------------------------
ProcessWindowChildEvent(const VclWindowEvent & rVclWindowEvent)663 void VCLXAccessibleToolBox::ProcessWindowChildEvent( const VclWindowEvent& rVclWindowEvent )
664 {
665 	switch ( rVclWindowEvent.GetId() )
666 	{
667 		case VCLEVENT_WINDOW_SHOW: // send create on show for direct accessible children
668 		{
669 			Reference< XAccessible > xReturn = GetItemWindowAccessible(rVclWindowEvent);
670 			if ( xReturn.is() )
671 				NotifyAccessibleEvent( AccessibleEventId::CHILD, Any(), makeAny(xReturn) );
672 			else
673 				HandleSubToolBarEvent( rVclWindowEvent, true );
674 		}
675 		break;
676 
677 		default:
678 			VCLXAccessibleComponent::ProcessWindowChildEvent( rVclWindowEvent );
679 
680 	}
681 }
682 // -----------------------------------------------------------------------------
683 // XInterface
684 // -----------------------------------------------------------------------------
IMPLEMENT_FORWARD_XINTERFACE2(VCLXAccessibleToolBox,VCLXAccessibleComponent,VCLXAccessibleToolBox_BASE)685 IMPLEMENT_FORWARD_XINTERFACE2( VCLXAccessibleToolBox, VCLXAccessibleComponent, VCLXAccessibleToolBox_BASE )
686 // -----------------------------------------------------------------------------
687 // XTypeProvider
688 // -----------------------------------------------------------------------------
689 IMPLEMENT_FORWARD_XTYPEPROVIDER2( VCLXAccessibleToolBox, VCLXAccessibleComponent, VCLXAccessibleToolBox_BASE )
690 // -----------------------------------------------------------------------------
691 // XComponent
692 // -----------------------------------------------------------------------------
693 void SAL_CALL VCLXAccessibleToolBox::disposing()
694 {
695 	VCLXAccessibleComponent::disposing();
696 
697 	// release the items
698 	for ( ToolBoxItemsMap::iterator aIter = m_aAccessibleChildren.begin();
699 		aIter != m_aAccessibleChildren.end(); ++aIter )
700 	{
701 		implReleaseToolboxItem( aIter, false, true );
702 	}
703 	m_aAccessibleChildren.clear();
704 }
705 // -----------------------------------------------------------------------------
706 // XServiceInfo
707 // -----------------------------------------------------------------------------
getImplementationName()708 ::rtl::OUString VCLXAccessibleToolBox::getImplementationName() throw (RuntimeException)
709 {
710 	return ::rtl::OUString::createFromAscii( "com.sun.star.comp.toolkit.AccessibleToolBox" );
711 }
712 // -----------------------------------------------------------------------------
getSupportedServiceNames()713 Sequence< ::rtl::OUString > VCLXAccessibleToolBox::getSupportedServiceNames() throw (RuntimeException)
714 {
715 	Sequence< ::rtl::OUString > aNames = VCLXAccessibleComponent::getSupportedServiceNames();
716 	sal_Int32 nLength = aNames.getLength();
717 	aNames.realloc( nLength + 1 );
718 	aNames[nLength] = ::rtl::OUString::createFromAscii( "com.sun.star.accessibility.AccessibleToolBox" );
719 	return aNames;
720 }
721 // -----------------------------------------------------------------------------
722 // XAccessibleContext
723 // -----------------------------------------------------------------------------
getAccessibleChildCount()724 sal_Int32 SAL_CALL VCLXAccessibleToolBox::getAccessibleChildCount(  ) throw (RuntimeException)
725 {
726 	comphelper::OExternalLockGuard aGuard( this );
727 
728 	sal_Int32 nCount = 0;
729 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
730 	if ( pToolBox )
731 	nCount = pToolBox->GetItemCount();
732 
733 	return nCount;
734 }
735 // -----------------------------------------------------------------------------
getAccessibleChild(sal_Int32 i)736 Reference< XAccessible > SAL_CALL VCLXAccessibleToolBox::getAccessibleChild( sal_Int32 i ) throw (IndexOutOfBoundsException, RuntimeException)
737 {
738 	if ( i < 0 || i >= getAccessibleChildCount() )
739 		throw IndexOutOfBoundsException();
740 
741 	comphelper::OExternalLockGuard aGuard( this );
742 
743 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
744 	if ( pToolBox )
745 	{
746 		Reference< XAccessible > xChild;
747 		// search for the child
748 		ToolBoxItemsMap::iterator aIter = m_aAccessibleChildren.find(i);
749 		if ( m_aAccessibleChildren.end() == aIter )
750 		{
751 			sal_uInt16 nItemId = pToolBox->GetItemId( (sal_uInt16)i );
752 			sal_uInt16 nHighlightItemId = pToolBox->GetHighlightItemId();
753 			Window* pItemWindow = pToolBox->GetItemWindow( nItemId );
754 			// not found -> create a new child
755 			VCLXAccessibleToolBoxItem* pChild = new VCLXAccessibleToolBoxItem( pToolBox, i );
756 			Reference< XAccessible> xParent = pChild;
757 			if ( pItemWindow )
758 			{
759 				xChild = new OToolBoxWindowItem(0,::comphelper::getProcessServiceFactory(),pItemWindow->GetAccessible(),xParent);
760 				pItemWindow->SetAccessible(xChild);
761 				pChild->SetChild( xChild );
762 			}
763 			xChild = pChild;
764 			if ( nHighlightItemId > 0 && nItemId == nHighlightItemId )
765 				pChild->SetFocus( sal_True );
766 			if ( pToolBox->IsItemChecked( nItemId ) )
767 				pChild->SetChecked( sal_True );
768 			if ( pToolBox->GetItemState( nItemId ) == STATE_DONTKNOW )
769 				pChild->SetIndeterminate( true );
770 			m_aAccessibleChildren.insert( ToolBoxItemsMap::value_type( i, xChild ) );
771 		}
772 		else
773 		{
774 			// found it
775 			xChild = aIter->second;
776 		}
777 		return xChild;
778 	}
779 
780 	return NULL;
781 }
782 // -----------------------------------------------------------------------------
getAccessibleAtPoint(const awt::Point & _rPoint)783 Reference< XAccessible > SAL_CALL VCLXAccessibleToolBox::getAccessibleAtPoint( const awt::Point& _rPoint ) throw (RuntimeException)
784 {
785 	comphelper::OExternalLockGuard aGuard( this );
786 
787 	Reference< XAccessible > xAccessible;
788 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
789 	if ( pToolBox )
790 	{
791 		sal_uInt16 nItemPos = pToolBox->GetItemPos( VCLPoint( _rPoint ) );
792 		if ( nItemPos != TOOLBOX_ITEM_NOTFOUND )
793 			xAccessible = getAccessibleChild( nItemPos );
794 	}
795 
796 	return xAccessible;
797 }
798 // -----------------------------------------------------------------------------
GetItemWindowAccessible(const VclWindowEvent & rVclWindowEvent)799 Reference< XAccessible > VCLXAccessibleToolBox::GetItemWindowAccessible( const VclWindowEvent& rVclWindowEvent )
800 {
801 	Reference< XAccessible > xReturn;
802 	Window* pChildWindow = (Window *) rVclWindowEvent.GetData();
803 	ToolBox* pToolBox = static_cast< ToolBox* >( GetWindow() );
804 	if ( pChildWindow && pToolBox )
805 	{
806 		sal_uInt16 nCount = pToolBox->GetItemCount();
807 		for (sal_uInt16 i = 0 ; i < nCount && !xReturn.is() ; ++i)
808 		{
809 			sal_uInt16 nItemId = pToolBox->GetItemId( i );
810 			Window* pItemWindow = pToolBox->GetItemWindow( nItemId );
811 			if ( pItemWindow == pChildWindow )
812 				xReturn = getAccessibleChild(i);
813 		}
814 	}
815 	return xReturn;
816 }
817 // -----------------------------------------------------------------------------
GetChildAccessible(const VclWindowEvent & rVclWindowEvent)818 Reference< XAccessible > VCLXAccessibleToolBox::GetChildAccessible( const VclWindowEvent& rVclWindowEvent )
819 {
820 	Reference< XAccessible > xReturn = GetItemWindowAccessible(rVclWindowEvent);
821 
822 	if ( !xReturn.is() )
823 		xReturn = VCLXAccessibleComponent::GetChildAccessible(rVclWindowEvent);
824 	return xReturn;
825 }
826 // -----------------------------------------------------------------------------
827 // XAccessibleSelection
828 // -----------------------------------------------------------------------------
selectAccessibleChild(sal_Int32 nChildIndex)829 void VCLXAccessibleToolBox::selectAccessibleChild( sal_Int32 nChildIndex ) throw (IndexOutOfBoundsException, RuntimeException)
830 {
831 	OExternalLockGuard aGuard( this );
832 	if ( nChildIndex < 0 || nChildIndex >= getAccessibleChildCount() )
833 		throw IndexOutOfBoundsException();
834 	ToolBox * pToolBox = static_cast < ToolBox * > ( GetWindow() );
835 	sal_uInt16 nPos = static_cast < sal_uInt16 > (nChildIndex);
836 	pToolBox->ChangeHighlight( nPos );
837 }
838 // -----------------------------------------------------------------------------
isAccessibleChildSelected(sal_Int32 nChildIndex)839 sal_Bool VCLXAccessibleToolBox::isAccessibleChildSelected( sal_Int32 nChildIndex ) throw (IndexOutOfBoundsException, RuntimeException)
840 {
841 	OExternalLockGuard aGuard( this );
842 	if ( nChildIndex < 0 || nChildIndex >= getAccessibleChildCount() )
843 		throw IndexOutOfBoundsException();
844 	ToolBox * pToolBox = static_cast < ToolBox * > ( GetWindow() );
845 	sal_uInt16 nPos = static_cast < sal_uInt16 > (nChildIndex);
846 	if ( pToolBox != NULL && pToolBox->GetHighlightItemId() == pToolBox->GetItemId( nPos ) )
847 		return sal_True;
848 	else
849 		return sal_False;
850 }
851 // -----------------------------------------------------------------------------
clearAccessibleSelection()852 void VCLXAccessibleToolBox::clearAccessibleSelection(  ) throw (RuntimeException)
853 {
854 	OExternalLockGuard aGuard( this );
855 	ToolBox * pToolBox = static_cast < ToolBox * > ( GetWindow() );
856 	pToolBox -> LoseFocus();
857 }
858 // -----------------------------------------------------------------------------
selectAllAccessibleChildren()859 void VCLXAccessibleToolBox::selectAllAccessibleChildren(  ) throw (RuntimeException)
860 {
861 	OExternalLockGuard aGuard( this );
862 	// intentionally empty. makes no sense for a toolbox
863 }
864 // -----------------------------------------------------------------------------
getSelectedAccessibleChildCount()865 sal_Int32 VCLXAccessibleToolBox::getSelectedAccessibleChildCount(  ) throw (RuntimeException)
866 {
867 	OExternalLockGuard aGuard( this );
868 	sal_Int32 nRet = 0;
869 	for ( sal_Int32 i = 0, nCount = getAccessibleChildCount(); i < nCount; i++ )
870 	{
871 		if ( isAccessibleChildSelected( i ) )
872 		{
873 			nRet = 1;
874 			break; // a toolbox can only have (n)one selected child
875 		}
876 	}
877 	return nRet;
878 }
879 // -----------------------------------------------------------------------------
getSelectedAccessibleChild(sal_Int32 nSelectedChildIndex)880 Reference< XAccessible > VCLXAccessibleToolBox::getSelectedAccessibleChild( sal_Int32 nSelectedChildIndex ) throw (IndexOutOfBoundsException, RuntimeException)
881 {
882 	OExternalLockGuard aGuard( this );
883 	if ( nSelectedChildIndex < 0 || nSelectedChildIndex >= getSelectedAccessibleChildCount() )
884 		throw IndexOutOfBoundsException();
885 	Reference< XAccessible > xChild;
886 	for ( sal_Int32 i = 0, j = 0, nCount = getAccessibleChildCount(); i < nCount; i++ )
887 	{
888 		if ( isAccessibleChildSelected( i ) && ( j++ == nSelectedChildIndex ) )
889 		{
890 			xChild = getAccessibleChild( i );
891 			break;
892 		}
893 	}
894 	return xChild;
895 }
896 // -----------------------------------------------------------------------------
deselectAccessibleChild(sal_Int32 nChildIndex)897 void VCLXAccessibleToolBox::deselectAccessibleChild( sal_Int32 nChildIndex ) throw (IndexOutOfBoundsException, RuntimeException)
898 {
899 	OExternalLockGuard aGuard( this );
900 	if ( nChildIndex < 0 || nChildIndex >= getAccessibleChildCount() )
901 		throw IndexOutOfBoundsException();
902 	clearAccessibleSelection(); // a toolbox can only have (n)one selected child
903 }
904 // -----------------------------------------------------------------------------
905