b6299c9e56
com::sun:⭐:accessibility::XAccessibleContext::getAccessibleChild() can throw an IndexOutOfBoundsException exception even when fetching with an index that is positive and less than the value returned by a call to the accessible context's getAccessibleChildCount() method so put every getAccessibleChild() call in a try/catch block. Note: this is actually expected behavior even though it is rare. For example, accessibility::AccessibleTextHelper_Impl::getAccessibleChild() uses the following code snippet to throw such an exception: if( 0 > i || i >= getAccessibleChildCount() || GetTextForwarder().GetParagraphCount() <= i ) In the case of tdf#146626, getAccessibleChildCount() returns 22 but getAccessibleChild(1) throws such an exception due to the last condition in the above code snippet. Change-Id: If974afb7b9178faa99b91dcd79eb5f169bbfe13e Reviewed-on: https://gerrit.libreoffice.org/c/core/+/153160 Tested-by: Jenkins Reviewed-by: Patrick Luby <plubius@neooffice.org>
230 lines
7.4 KiB
C++
230 lines
7.4 KiB
C++
/* -*- 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 "documentfocuslistener.hxx"
|
|
|
|
#include <com/sun/star/accessibility/XAccessibleEventBroadcaster.hpp>
|
|
#include <com/sun/star/accessibility/AccessibleEventId.hpp>
|
|
#include <com/sun/star/accessibility/AccessibleStateType.hpp>
|
|
#include <com/sun/star/lang/IndexOutOfBoundsException.hpp>
|
|
#include <sal/log.hxx>
|
|
|
|
using namespace ::com::sun::star::accessibility;
|
|
using namespace ::com::sun::star::lang;
|
|
using namespace ::com::sun::star::uno;
|
|
|
|
DocumentFocusListener::DocumentFocusListener(AquaA11yFocusTracker& rTracker) :
|
|
m_aFocusTracker(rTracker)
|
|
{
|
|
}
|
|
|
|
void SAL_CALL
|
|
DocumentFocusListener::disposing( const EventObject& aEvent )
|
|
{
|
|
// Unref the object here, but do not remove as listener since the object
|
|
// might no longer be in a state that safely allows this.
|
|
if( aEvent.Source.is() )
|
|
m_aRefList.erase(aEvent.Source);
|
|
}
|
|
|
|
void SAL_CALL
|
|
DocumentFocusListener::notifyEvent( const AccessibleEventObject& aEvent )
|
|
{
|
|
try {
|
|
switch( aEvent.EventId )
|
|
{
|
|
case AccessibleEventId::STATE_CHANGED:
|
|
{
|
|
sal_Int64 nState = AccessibleStateType::INVALID;
|
|
aEvent.NewValue >>= nState;
|
|
|
|
if( AccessibleStateType::FOCUSED == nState )
|
|
m_aFocusTracker.setFocusedObject( getAccessible(aEvent) );
|
|
}
|
|
break;
|
|
|
|
case AccessibleEventId::CHILD:
|
|
{
|
|
Reference< XAccessible > xChild;
|
|
if( (aEvent.OldValue >>= xChild) && xChild.is() )
|
|
detachRecursive(xChild);
|
|
|
|
if( (aEvent.NewValue >>= xChild) && xChild.is() )
|
|
attachRecursive(xChild);
|
|
}
|
|
break;
|
|
|
|
case AccessibleEventId::INVALIDATE_ALL_CHILDREN:
|
|
{
|
|
Reference< XAccessible > xAccessible( getAccessible(aEvent) );
|
|
detachRecursive(xAccessible);
|
|
attachRecursive(xAccessible);
|
|
SAL_INFO("vcl", "Invalidate all children called" );
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
catch (const IndexOutOfBoundsException&)
|
|
{
|
|
SAL_WARN("vcl", "Focused object has invalid index in parent");
|
|
}
|
|
}
|
|
|
|
Reference< XAccessible > DocumentFocusListener::getAccessible(const EventObject& aEvent )
|
|
{
|
|
Reference< XAccessible > xAccessible(aEvent.Source, UNO_QUERY);
|
|
|
|
if( xAccessible.is() )
|
|
return xAccessible;
|
|
|
|
Reference< XAccessibleContext > xContext(aEvent.Source, UNO_QUERY);
|
|
|
|
if( xContext.is() )
|
|
{
|
|
Reference< XAccessible > xParent( xContext->getAccessibleParent() );
|
|
if( xParent.is() )
|
|
{
|
|
Reference< XAccessibleContext > xParentContext( xParent->getAccessibleContext() );
|
|
if( xParentContext.is() )
|
|
{
|
|
try {
|
|
return xParentContext->getAccessibleChild( xContext->getAccessibleIndexInParent() );
|
|
}
|
|
catch (const IndexOutOfBoundsException&)
|
|
{
|
|
SAL_WARN("vcl", "Accessible object has invalid index in parent");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return Reference< XAccessible >();
|
|
}
|
|
|
|
void DocumentFocusListener::attachRecursive(const Reference< XAccessible >& xAccessible)
|
|
{
|
|
Reference< XAccessibleContext > xContext = xAccessible->getAccessibleContext();
|
|
|
|
if( xContext.is() )
|
|
attachRecursive(xAccessible, xContext);
|
|
}
|
|
|
|
void DocumentFocusListener::attachRecursive(
|
|
const Reference< XAccessible >& xAccessible,
|
|
const Reference< XAccessibleContext >& xContext
|
|
)
|
|
{
|
|
if( xContext.is() )
|
|
{
|
|
sal_Int64 nStateSet = xContext->getAccessibleStateSet();
|
|
|
|
attachRecursive(xAccessible, xContext, nStateSet);
|
|
}
|
|
}
|
|
|
|
void DocumentFocusListener::attachRecursive(
|
|
const Reference< XAccessible >& xAccessible,
|
|
const Reference< XAccessibleContext >& xContext,
|
|
sal_Int64 nStateSet
|
|
)
|
|
{
|
|
if( nStateSet & AccessibleStateType::FOCUSED )
|
|
m_aFocusTracker.setFocusedObject( xAccessible );
|
|
|
|
Reference< XAccessibleEventBroadcaster > xBroadcaster(xContext, UNO_QUERY);
|
|
|
|
// If not already done, add the broadcaster to the list and attach as listener.
|
|
if( xBroadcaster.is() && m_aRefList.insert(xBroadcaster).second )
|
|
{
|
|
xBroadcaster->addAccessibleEventListener(static_cast< XAccessibleEventListener *>(this));
|
|
|
|
if( ! (nStateSet & AccessibleStateType::MANAGES_DESCENDANTS) )
|
|
{
|
|
try {
|
|
sal_Int64 n, nmax = xContext->getAccessibleChildCount();
|
|
for( n = 0; n < nmax; n++ )
|
|
{
|
|
Reference< XAccessible > xChild( xContext->getAccessibleChild( n ) );
|
|
|
|
if( xChild.is() )
|
|
attachRecursive(xChild);
|
|
}
|
|
}
|
|
catch (const IndexOutOfBoundsException&)
|
|
{
|
|
SAL_WARN("vcl", "Accessible object index does not exist in parent");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void DocumentFocusListener::detachRecursive(const Reference< XAccessible >& xAccessible)
|
|
{
|
|
Reference< XAccessibleContext > xContext = xAccessible->getAccessibleContext();
|
|
|
|
if( xContext.is() )
|
|
detachRecursive(xAccessible, xContext);
|
|
}
|
|
|
|
void DocumentFocusListener::detachRecursive(
|
|
const Reference< XAccessible >& xAccessible,
|
|
const Reference< XAccessibleContext >& xContext
|
|
)
|
|
{
|
|
sal_Int64 nStateSet = xContext->getAccessibleStateSet();
|
|
|
|
detachRecursive(xAccessible, xContext, nStateSet);
|
|
}
|
|
|
|
void DocumentFocusListener::detachRecursive(
|
|
const Reference< XAccessible >&,
|
|
const Reference< XAccessibleContext >& xContext,
|
|
sal_Int64 nStateSet
|
|
)
|
|
{
|
|
Reference< XAccessibleEventBroadcaster > xBroadcaster(xContext, UNO_QUERY);
|
|
|
|
if( xBroadcaster.is() && 0 < m_aRefList.erase(xBroadcaster) )
|
|
{
|
|
xBroadcaster->removeAccessibleEventListener(static_cast< XAccessibleEventListener *>(this));
|
|
|
|
if( ! (nStateSet & AccessibleStateType::MANAGES_DESCENDANTS) )
|
|
{
|
|
try {
|
|
sal_Int64 n, nmax = xContext->getAccessibleChildCount();
|
|
for( n = 0; n < nmax; n++ )
|
|
{
|
|
Reference< XAccessible > xChild( xContext->getAccessibleChild( n ) );
|
|
|
|
if( xChild.is() )
|
|
detachRecursive(xChild);
|
|
}
|
|
}
|
|
catch (const IndexOutOfBoundsException&)
|
|
{
|
|
SAL_WARN("vcl", "Accessible object index does not exist in parent");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|