/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace accessibility { // class AccessibleListBox ----------------------------------------------------- using namespace ::com::sun::star::accessibility; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::lang; using namespace ::com::sun::star; // Ctor() and Dtor() AccessibleListBox::AccessibleListBox( SvTreeListBox& _rListBox, const Reference< XAccessible >& _xParent ) : VCLXAccessibleComponent( _rListBox.GetWindowPeer() ), m_xParent( _xParent ) { } AccessibleListBox::~AccessibleListBox() { if ( isAlive() ) { // increment ref count to prevent double call of Dtor osl_atomic_increment( &m_refCount ); dispose(); } } IMPLEMENT_FORWARD_XINTERFACE2(AccessibleListBox, VCLXAccessibleComponent, AccessibleListBox_BASE) IMPLEMENT_FORWARD_XTYPEPROVIDER2(AccessibleListBox, VCLXAccessibleComponent, AccessibleListBox_BASE) void AccessibleListBox::ProcessWindowEvent( const VclWindowEvent& rVclWindowEvent ) { if ( isAlive() ) { switch ( rVclWindowEvent.GetId() ) { case VCLEVENT_CHECKBOX_TOGGLE : { if ( !getListBox() || !getListBox()->HasFocus() ) { return; } AccessibleListBoxEntry* pCurOpEntry = GetCurEventEntry(rVclWindowEvent); if(!pCurOpEntry) { return ; } uno::Any aValue; aValue <<= AccessibleStateType::CHECKED; if ( getListBox()->GetCheckButtonState( pCurOpEntry->GetSvLBoxEntry() ) == SV_BUTTON_CHECKED ) { pCurOpEntry->NotifyAccessibleEvent( AccessibleEventId::STATE_CHANGED, uno::Any(), aValue ); } else { pCurOpEntry->NotifyAccessibleEvent( AccessibleEventId::STATE_CHANGED, aValue,uno::Any() ); } break; } case VCLEVENT_LISTBOX_SELECT : { OSL_FAIL("Debug: Treelist shouldn't use VCLEVENT_LISTBOX_SELECT"); break; } case VCLEVENT_LISTBOX_TREESELECT: { if ( getListBox() && getListBox()->HasFocus() ) { AccessibleListBoxEntry* pEntry =static_cast< AccessibleListBoxEntry* >(m_xFocusedChild.get()); if (pEntry) { pEntry->NotifyAccessibleEvent( AccessibleEventId::SELECTION_CHANGED, Any(), Any() ); } } } break; case VCLEVENT_LISTBOX_TREEFOCUS: { SvTreeListBox* pBox = getListBox(); bool bNeedFocus = false; if (pBox) { vcl::Window* pParent = ((vcl::Window*)pBox)->GetParent(); if (pParent && pParent->GetType() == WINDOW_FLOATINGWINDOW) { // MT: ImplGetAppSVData shouldn't be exported from VCL. // In which scenario is this needed? // If needed, we need to find an other solution /* ImplSVData* pSVData = ImplGetAppSVData(); if (pSVData && pSVData->maWinData.mpFirstFloat == (FloatingWindow*)pParent) bNeedFocus = true; */ } } if( pBox && (pBox->HasFocus() || bNeedFocus) ) { uno::Any aOldValue, aNewValue; SvTreeListEntry* pEntry = static_cast< SvTreeListEntry* >( rVclWindowEvent.GetData() ); if ( pEntry ) { AccessibleListBoxEntry* pEntryFocus =static_cast< AccessibleListBoxEntry* >(m_xFocusedChild.get()); if (pEntryFocus && pEntryFocus->GetSvLBoxEntry() == pEntry) { aOldValue <<= uno::Any(); aNewValue <<= m_xFocusedChild; NotifyAccessibleEvent( AccessibleEventId::ACTIVE_DESCENDANT_CHANGED, aOldValue, aNewValue ); return ; } aOldValue <<= m_xFocusedChild; MAP_ENTRY::iterator mi = m_mapEntry.find(pEntry); if(mi != m_mapEntry.end()) { OSL_ASSERT(mi->second.get() != NULL); m_xFocusedChild = mi->second; } else { AccessibleListBoxEntry *pEntNew = new AccessibleListBoxEntry( *getListBox(), pEntry, NULL ); m_xFocusedChild = pEntNew; m_mapEntry.insert(MAP_ENTRY::value_type(pEntry,pEntNew)); } aNewValue <<= m_xFocusedChild; NotifyAccessibleEvent( AccessibleEventId::ACTIVE_DESCENDANT_CHANGED, aOldValue, aNewValue ); } else { aOldValue <<= uno::Any(); aNewValue <<= AccessibleStateType::FOCUSED; NotifyAccessibleEvent( AccessibleEventId::STATE_CHANGED, aOldValue, aNewValue ); } } } break; case VCLEVENT_LISTBOX_ITEMREMOVED: { SvTreeListEntry* pEntry = static_cast< SvTreeListEntry* >( rVclWindowEvent.GetData() ); if ( pEntry ) { RemoveChildEntries(pEntry); } else { // NULL means Clear() MAP_ENTRY::iterator mi = m_mapEntry.begin(); for ( ; mi != m_mapEntry.end() ; ++mi) { uno::Any aNewValue; uno::Any aOldValue; aOldValue <<= mi->second; NotifyAccessibleEvent( AccessibleEventId::CHILD, aOldValue, aNewValue ); } m_mapEntry.clear(); } } break; // #i92103# case VCLEVENT_ITEM_EXPANDED : case VCLEVENT_ITEM_COLLAPSED : { SvTreeListEntry* pEntry = static_cast< SvTreeListEntry* >( rVclWindowEvent.GetData() ); if ( pEntry ) { AccessibleListBoxEntry* pAccListBoxEntry = new AccessibleListBoxEntry( *getListBox(), pEntry, this ); Reference< XAccessible > xChild = pAccListBoxEntry; const short nAccEvent = ( rVclWindowEvent.GetId() == VCLEVENT_ITEM_EXPANDED ) ? AccessibleEventId::LISTBOX_ENTRY_EXPANDED : AccessibleEventId::LISTBOX_ENTRY_COLLAPSED; uno::Any aListBoxEntry; aListBoxEntry <<= xChild; NotifyAccessibleEvent( nAccEvent, Any(), aListBoxEntry ); if ( getListBox() && getListBox()->HasFocus() ) { NotifyAccessibleEvent( AccessibleEventId::ACTIVE_DESCENDANT_CHANGED, Any(), aListBoxEntry ); } } } break; default: VCLXAccessibleComponent::ProcessWindowEvent (rVclWindowEvent); } } } AccessibleListBoxEntry* AccessibleListBox::GetCurEventEntry( const VclWindowEvent& rVclWindowEvent ) { SvTreeListEntry* pEntry = static_cast< SvTreeListEntry* >( rVclWindowEvent.GetData() ); if ( !pEntry ) pEntry = getListBox()->GetCurEntry(); AccessibleListBoxEntry* pEntryFocus =static_cast< AccessibleListBoxEntry* >(m_xFocusedChild.get()); if (pEntryFocus && pEntry && pEntry != pEntryFocus->GetSvLBoxEntry()) { AccessibleListBoxEntry *pAccCurOptionEntry =NULL; MAP_ENTRY::iterator mi = m_mapEntry.find(pEntry); if (mi != m_mapEntry.end()) { pAccCurOptionEntry= static_cast< AccessibleListBoxEntry* >(mi->second.get()); } else { pAccCurOptionEntry =new AccessibleListBoxEntry( *getListBox(), pEntry, NULL ); std::pair pairMi = m_mapEntry.insert(MAP_ENTRY::value_type(pAccCurOptionEntry->GetSvLBoxEntry(),pAccCurOptionEntry)); mi = pairMi.first; } uno::Any aNewValue; aNewValue <<= mi->second;//xAcc NotifyAccessibleEvent( AccessibleEventId::CHILD, uno::Any(), aNewValue );//Add return pAccCurOptionEntry; } else { return pEntryFocus; } } void AccessibleListBox::RemoveChildEntries(SvTreeListEntry* pEntry) { MAP_ENTRY::iterator mi = m_mapEntry.find(pEntry); if ( mi != m_mapEntry.end() ) { uno::Any aNewValue; uno::Any aOldValue; aOldValue <<= mi->second; NotifyAccessibleEvent( AccessibleEventId::CHILD, aOldValue, aNewValue ); m_mapEntry.erase(mi); } SvTreeListBox* pBox = getListBox(); SvTreeListEntry* pEntryChild = pBox->FirstChild(pEntry); while (pEntryChild) { RemoveChildEntries(pEntryChild); pEntryChild = SvTreeListBox::NextSibling(pEntryChild); } } void AccessibleListBox::ProcessWindowChildEvent( const VclWindowEvent& rVclWindowEvent ) { switch ( rVclWindowEvent.GetId() ) { case VCLEVENT_WINDOW_SHOW: case VCLEVENT_WINDOW_HIDE: { } break; default: { VCLXAccessibleComponent::ProcessWindowChildEvent( rVclWindowEvent ); } break; } } // XComponent void SAL_CALL AccessibleListBox::disposing() { ::osl::MutexGuard aGuard( m_aMutex ); m_mapEntry.clear(); VCLXAccessibleComponent::disposing(); m_xParent = NULL; } // XServiceInfo OUString SAL_CALL AccessibleListBox::getImplementationName() throw(RuntimeException, std::exception) { return getImplementationName_Static(); } Sequence< OUString > SAL_CALL AccessibleListBox::getSupportedServiceNames() throw(RuntimeException, std::exception) { return getSupportedServiceNames_Static(); } sal_Bool SAL_CALL AccessibleListBox::supportsService( const OUString& _rServiceName ) throw (RuntimeException, std::exception) { return cppu::supportsService(this, _rServiceName); } // XServiceInfo - static methods Sequence< OUString > AccessibleListBox::getSupportedServiceNames_Static() throw( RuntimeException ) { Sequence< OUString > aSupported(3); aSupported[0] = "com.sun.star.accessibility.AccessibleContext"; aSupported[1] = "com.sun.star.accessibility.AccessibleComponent"; aSupported[2] = "com.sun.star.awt.AccessibleTreeListBox"; return aSupported; } OUString AccessibleListBox::getImplementationName_Static() throw( RuntimeException ) { return OUString( "com.sun.star.comp.svtools.AccessibleTreeListBox" ); } // XAccessible Reference< XAccessibleContext > SAL_CALL AccessibleListBox::getAccessibleContext( ) throw (RuntimeException, std::exception) { ensureAlive(); return this; } // XAccessibleContext sal_Int32 SAL_CALL AccessibleListBox::getAccessibleChildCount( ) throw (RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); sal_Int32 nCount = 0; SvTreeListBox* pSvTreeListBox = getListBox(); if ( pSvTreeListBox ) nCount = pSvTreeListBox->GetLevelChildCount( NULL ); return nCount; } Reference< XAccessible > SAL_CALL AccessibleListBox::getAccessibleChild( sal_Int32 i ) throw (IndexOutOfBoundsException,RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); SvTreeListEntry* pEntry = getListBox()->GetEntry(i); if ( !pEntry ) throw IndexOutOfBoundsException(); // Solution: Set the parameter of the parent to null to let entry determine the parent by itself //return new AccessibleListBoxEntry( *getListBox(), pEntry, this ); return new AccessibleListBoxEntry( *getListBox(), pEntry, NULL ); } Reference< XAccessible > SAL_CALL AccessibleListBox::getAccessibleParent( ) throw (RuntimeException, std::exception) { ::osl::MutexGuard aGuard( m_aMutex ); ensureAlive(); return m_xParent; } sal_Int32 AccessibleListBox::GetRoleType() { sal_Int32 nCase = 0; SvTreeListEntry* pEntry = getListBox()->GetEntry(0); if ( pEntry ) { if( pEntry->HasChildrenOnDemand() || getListBox()->GetChildCount(pEntry) > 0 ) { nCase = 1; return nCase; } } bool bHasButtons = (getListBox()->GetStyle() & WB_HASBUTTONS)!=0; if( !(getListBox()->GetTreeFlags() & SvTreeFlags::CHKBTN) ) { if( bHasButtons ) nCase = 1; } else { if( bHasButtons ) nCase = 2; else nCase = 3; } return nCase; } sal_Int16 SAL_CALL AccessibleListBox::getAccessibleRole() throw (RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); if(getListBox()) { SvTreeAccRoleType nType = getListBox()->GetAllEntriesAccessibleRoleType(); if( nType == SvTreeAccRoleType::TREE) return AccessibleRole::TREE; else if( nType == SvTreeAccRoleType::LIST) return AccessibleRole::LIST; } //o is: return AccessibleRole::TREE; bool bHasButtons = (getListBox()->GetStyle() & WB_HASBUTTONS)!=0; if(!bHasButtons && (getListBox()->GetTreeFlags() & SvTreeFlags::CHKBTN)) return AccessibleRole::LIST; else if (GetRoleType() == 0) return AccessibleRole::LIST; else return AccessibleRole::TREE; } OUString SAL_CALL AccessibleListBox::getAccessibleDescription( ) throw (RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); return getListBox()->GetAccessibleDescription(); } OUString SAL_CALL AccessibleListBox::getAccessibleName( ) throw (RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); return getListBox()->GetAccessibleName(); } // XAccessibleSelection void SAL_CALL AccessibleListBox::selectAccessibleChild( sal_Int32 nChildIndex ) throw (IndexOutOfBoundsException, RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); SvTreeListEntry* pEntry = getListBox()->GetEntry( nChildIndex ); if ( !pEntry ) throw IndexOutOfBoundsException(); getListBox()->Select( pEntry, true ); } sal_Bool SAL_CALL AccessibleListBox::isAccessibleChildSelected( sal_Int32 nChildIndex ) throw (IndexOutOfBoundsException, RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); SvTreeListEntry* pEntry = getListBox()->GetEntry( nChildIndex ); if ( !pEntry ) throw IndexOutOfBoundsException(); return getListBox()->IsSelected( pEntry ); } void SAL_CALL AccessibleListBox::clearAccessibleSelection( ) throw (RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); sal_Int32 nCount = getListBox()->GetLevelChildCount( NULL ); for ( sal_Int32 i = 0; i < nCount; ++i ) { SvTreeListEntry* pEntry = getListBox()->GetEntry( i ); if ( getListBox()->IsSelected( pEntry ) ) getListBox()->Select( pEntry, false ); } } void SAL_CALL AccessibleListBox::selectAllAccessibleChildren( ) throw (RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); sal_Int32 nCount = getListBox()->GetLevelChildCount( NULL ); for ( sal_Int32 i = 0; i < nCount; ++i ) { SvTreeListEntry* pEntry = getListBox()->GetEntry( i ); if ( !getListBox()->IsSelected( pEntry ) ) getListBox()->Select( pEntry, true ); } } sal_Int32 SAL_CALL AccessibleListBox::getSelectedAccessibleChildCount( ) throw (RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); return getListBox()->GetSelectionCount(); } Reference< XAccessible > SAL_CALL AccessibleListBox::getSelectedAccessibleChild( sal_Int32 nSelectedChildIndex ) throw (IndexOutOfBoundsException, RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); if ( nSelectedChildIndex < 0 || nSelectedChildIndex >= getSelectedAccessibleChildCount() ) throw IndexOutOfBoundsException(); Reference< XAccessible > xChild; sal_Int32 nSelCount= 0; sal_Int32 nCount = getListBox()->GetLevelChildCount( NULL ); for ( sal_Int32 i = 0; i < nCount; ++i ) { SvTreeListEntry* pEntry = getListBox()->GetEntry( i ); if ( getListBox()->IsSelected( pEntry ) ) ++nSelCount; if ( nSelCount == ( nSelectedChildIndex + 1 ) ) { // Solution: Set the parameter of the parent to null to let entry determine the parent by itself //xChild = new AccessibleListBoxEntry( *getListBox(), pEntry, this ); xChild = new AccessibleListBoxEntry( *getListBox(), pEntry, NULL ); break; } } return xChild; } void SAL_CALL AccessibleListBox::deselectAccessibleChild( sal_Int32 nSelectedChildIndex ) throw (IndexOutOfBoundsException, RuntimeException, std::exception) { ::comphelper::OExternalLockGuard aGuard( this ); ensureAlive(); SvTreeListEntry* pEntry = getListBox()->GetEntry( nSelectedChildIndex ); if ( !pEntry ) throw IndexOutOfBoundsException(); getListBox()->Select( pEntry, false ); } void AccessibleListBox::FillAccessibleStateSet( utl::AccessibleStateSetHelper& rStateSet ) { VCLXAccessibleComponent::FillAccessibleStateSet( rStateSet ); if ( getListBox() && isAlive() ) { rStateSet.AddState( AccessibleStateType::FOCUSABLE ); rStateSet.AddState( AccessibleStateType::MANAGES_DESCENDANTS ); if ( getListBox()->GetSelectionMode() == MULTIPLE_SELECTION ) rStateSet.AddState( AccessibleStateType::MULTI_SELECTABLE ); } } VclPtr< SvTreeListBox > AccessibleListBox::getListBox() const { return GetAs< SvTreeListBox >(); } }// namespace accessibility /* vim:set shiftwidth=4 softtabstop=4 expandtab: */