LCOV - code coverage report
Current view: top level - comphelper/source/misc - accessiblecontexthelper.cxx (source / functions) Hit Total Coverage
Test: commit 10e77ab3ff6f4314137acd6e2702a6e5c1ce1fae Lines: 82 115 71.3 %
Date: 2014-11-03 Functions: 21 27 77.8 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include <comphelper/accessiblecontexthelper.hxx>
      21             : #include <osl/diagnose.h>
      22             : #include <cppuhelper/weakref.hxx>
      23             : #include <com/sun/star/accessibility/AccessibleEventId.hpp>
      24             : #include <com/sun/star/accessibility/AccessibleStateType.hpp>
      25             : #include <comphelper/accessibleeventnotifier.hxx>
      26             : 
      27             : 
      28             : namespace comphelper
      29             : {
      30             : 
      31             : 
      32             :     using namespace ::com::sun::star::uno;
      33             :     using namespace ::com::sun::star::lang;
      34             :     using namespace ::com::sun::star::accessibility;
      35             : 
      36        5374 :     IMutex::~IMutex() {}
      37             : 
      38             : 
      39             :     //= OContextHelper_Impl
      40             : 
      41             :     /** implementation class for OAccessibleContextHelper. No own thread safety!
      42             :     */
      43        5432 :     class OContextHelper_Impl
      44             :     {
      45             :     private:
      46             :         IMutex*                             m_pExternalLock;    // the optional additional external lock
      47             : 
      48             :         WeakReference< XAccessible >        m_aCreator;         // the XAccessible which created our XAccessibleContext
      49             : 
      50             :         AccessibleEventNotifier::TClientId  m_nClientId;
      51             : 
      52             :     public:
      53           0 :         inline  Reference< XAccessible >    getCreator( ) const                 { return m_aCreator; }
      54             :         inline  void                        setCreator( const Reference< XAccessible >& _rAcc );
      55             : 
      56      103132 :         inline  IMutex*                     getExternalLock( )                  { return m_pExternalLock; }
      57       10946 :         inline  void                        setExternalLock( IMutex* _pLock )   { m_pExternalLock = _pLock; }
      58             : 
      59             :         inline  AccessibleEventNotifier::TClientId
      60        8480 :                                             getClientId() const                 { return m_nClientId; }
      61         152 :         inline  void                        setClientId( const AccessibleEventNotifier::TClientId _nId )
      62         152 :                                                                                 { m_nClientId = _nId; }
      63             : 
      64             :     public:
      65        5514 :         OContextHelper_Impl()
      66             :             :m_pExternalLock( NULL )
      67        5514 :             ,m_nClientId( 0 )
      68             :         {
      69        5514 :         }
      70             :     };
      71             : 
      72             : 
      73         718 :     inline  void OContextHelper_Impl::setCreator( const Reference< XAccessible >& _rAcc )
      74             :     {
      75         718 :         m_aCreator = _rAcc;
      76         718 :     }
      77             : 
      78             : 
      79             :     //= OAccessibleContextHelper
      80             : 
      81             : 
      82        5514 :     OAccessibleContextHelper::OAccessibleContextHelper( IMutex* _pExternalLock )
      83        5514 :         :OAccessibleContextHelper_Base( GetMutex() )
      84       11028 :         ,m_pImpl( NULL )
      85             :     {
      86             :         assert(_pExternalLock);
      87        5514 :         m_pImpl = new OContextHelper_Impl();
      88        5514 :         m_pImpl->setExternalLock( _pExternalLock );
      89        5514 :     }
      90             : 
      91             : 
      92        5432 :     void OAccessibleContextHelper::forgetExternalLock()
      93             :     {
      94        5432 :         m_pImpl->setExternalLock( NULL );
      95        5432 :     }
      96             : 
      97             : 
      98       10864 :     OAccessibleContextHelper::~OAccessibleContextHelper( )
      99             :     {
     100        5432 :         forgetExternalLock();
     101             :             // this ensures that the lock, which may be already destroyed as part of the derivee,
     102             :             // is not used anymore
     103             : 
     104        5432 :         ensureDisposed();
     105             : 
     106        5432 :         delete m_pImpl;
     107        5432 :         m_pImpl = NULL;
     108        5432 :     }
     109             : 
     110             : 
     111      103132 :     IMutex* OAccessibleContextHelper::getExternalLock( )
     112             :     {
     113      103132 :         return m_pImpl->getExternalLock();
     114             :     }
     115             : 
     116             : 
     117        5562 :     void SAL_CALL OAccessibleContextHelper::disposing()
     118             :     {
     119             :         // rhbz#1001768: de facto this class is locked by SolarMutex;
     120             :         // do not lock m_Mutex because it may cause deadlock
     121        5562 :         OMutexGuard aGuard( getExternalLock() );
     122             : 
     123        5562 :         if ( m_pImpl->getClientId( ) )
     124             :         {
     125          48 :             AccessibleEventNotifier::revokeClientNotifyDisposing( m_pImpl->getClientId( ), *this );
     126          48 :             m_pImpl->setClientId( 0 );
     127        5562 :         }
     128        5562 :     }
     129             : 
     130             : 
     131          74 :     void SAL_CALL OAccessibleContextHelper::addAccessibleEventListener( const Reference< XAccessibleEventListener >& _rxListener ) throw (RuntimeException, std::exception)
     132             :     {
     133          74 :         OMutexGuard aGuard( getExternalLock() );
     134             :             // don't use the OContextEntryGuard - it will throw an exception if we're not alive
     135             :             // anymore, while the most recent specification for XComponent states that we should
     136             :             // silently ignore the call in such a situation
     137          74 :         if ( !isAlive() )
     138             :         {
     139           0 :             if ( _rxListener.is() )
     140           0 :                 _rxListener->disposing( EventObject( *this ) );
     141          74 :             return;
     142             :         }
     143             : 
     144          74 :         if ( _rxListener.is() )
     145             :         {
     146          74 :             if ( !m_pImpl->getClientId( ) )
     147          74 :                 m_pImpl->setClientId( AccessibleEventNotifier::registerClient( ) );
     148             : 
     149          74 :             AccessibleEventNotifier::addEventListener( m_pImpl->getClientId( ), _rxListener );
     150          74 :         }
     151             :     }
     152             : 
     153             : 
     154          78 :     void SAL_CALL OAccessibleContextHelper::removeAccessibleEventListener( const Reference< XAccessibleEventListener >& _rxListener ) throw (RuntimeException, std::exception)
     155             :     {
     156          78 :         OMutexGuard aGuard( getExternalLock() );
     157             :             // don't use the OContextEntryGuard - it will throw an exception if we're not alive
     158             :             // anymore, while the most recent specification for XComponent states that we should
     159             :             // silently ignore the call in such a situation
     160          78 :         if ( !isAlive() )
     161         126 :             return;
     162             : 
     163          30 :         if ( _rxListener.is() )
     164             :         {
     165          30 :             sal_Int32 nListenerCount = AccessibleEventNotifier::removeEventListener( m_pImpl->getClientId( ), _rxListener );
     166          30 :             if ( !nListenerCount )
     167             :             {
     168             :                 // no listeners anymore
     169             :                 // -> revoke ourself. This may lead to the notifier thread dying (if we were the last client),
     170             :                 // and at least to us not firing any events anymore, in case somebody calls
     171             :                 // NotifyAccessibleEvent, again
     172          30 :                 AccessibleEventNotifier::revokeClient( m_pImpl->getClientId( ) );
     173          30 :                 m_pImpl->setClientId( 0 );
     174             :             }
     175          30 :         }
     176             :     }
     177             : 
     178             : 
     179        2547 :     void OAccessibleContextHelper::NotifyAccessibleEvent( const sal_Int16 _nEventId,
     180             :         const Any& _rOldValue, const Any& _rNewValue )
     181             :     {
     182        2547 :         if ( !m_pImpl->getClientId( ) )
     183             :             // if we don't have a client id for the notifier, then we don't have listeners, then
     184             :             // we don't need to notify anything
     185        4979 :             return;
     186             : 
     187             :         // build an event object
     188         115 :         AccessibleEventObject aEvent;
     189         115 :         aEvent.Source = *this;
     190         115 :         aEvent.EventId = _nEventId;
     191         115 :         aEvent.OldValue = _rOldValue;
     192         115 :         aEvent.NewValue = _rNewValue;
     193             : 
     194             :         // let the notifier handle this event
     195         115 :         AccessibleEventNotifier::addEvent( m_pImpl->getClientId( ), aEvent );
     196             :     }
     197             : 
     198             : 
     199       93834 :     bool OAccessibleContextHelper::isAlive() const
     200             :     {
     201       93834 :         return !GetBroadcastHelper().bDisposed && !GetBroadcastHelper().bInDispose;
     202             :     }
     203             : 
     204             : 
     205       93646 :     void OAccessibleContextHelper::ensureAlive() const
     206             :     {
     207       93646 :         if( !isAlive() )
     208           0 :             throw DisposedException();
     209       93646 :     }
     210             : 
     211             : 
     212        6092 :     void OAccessibleContextHelper::ensureDisposed( )
     213             :     {
     214        6092 :         if ( !GetBroadcastHelper().bDisposed )
     215             :         {
     216             :             OSL_ENSURE( 0 == m_refCount, "OAccessibleContextHelper::ensureDisposed: this method _has_ to be called from without your dtor only!" );
     217           0 :             acquire();
     218           0 :             dispose();
     219             :         }
     220        6092 :     }
     221             : 
     222             : 
     223         718 :     void OAccessibleContextHelper::lateInit( const Reference< XAccessible >& _rxAccessible )
     224             :     {
     225         718 :         m_pImpl->setCreator( _rxAccessible );
     226         718 :     }
     227             : 
     228             : 
     229           0 :     Reference< XAccessible > OAccessibleContextHelper::getAccessibleCreator( ) const
     230             :     {
     231           0 :         return m_pImpl->getCreator();
     232             :     }
     233             : 
     234             : 
     235           0 :     sal_Int32 SAL_CALL OAccessibleContextHelper::getAccessibleIndexInParent(  ) throw (RuntimeException, std::exception)
     236             :     {
     237           0 :         OExternalLockGuard aGuard( this );
     238             : 
     239             :         // -1 for child not found/no parent (according to specification)
     240           0 :         sal_Int32 nRet = -1;
     241             : 
     242             :         try
     243             :         {
     244             : 
     245           0 :             Reference< XAccessibleContext > xParentContext( implGetParentContext() );
     246             : 
     247             :             //  iterate over parent's children and search for this object
     248           0 :             if ( xParentContext.is() )
     249             :             {
     250             :                 // our own XAccessible for comparing with the children of our parent
     251           0 :                 Reference< XAccessible > xCreator( m_pImpl->getCreator() );
     252             : 
     253             :                 OSL_ENSURE( xCreator.is(), "OAccessibleContextHelper::getAccessibleIndexInParent: invalid creator!" );
     254             :                     // two ideas why this could be NULL:
     255             :                     // * nobody called our late ctor (init), so we never had a creator at all -> bad
     256             :                     // * the creator is already dead. In this case, we should have been disposed, and
     257             :                     //   never survived the above OContextEntryGuard.
     258             :                     // in all other situations the creator should be non-NULL
     259             : 
     260           0 :                 if ( xCreator.is() )
     261             :                 {
     262           0 :                     sal_Int32 nChildCount = xParentContext->getAccessibleChildCount();
     263           0 :                     for ( sal_Int32 nChild = 0; ( nChild < nChildCount ) && ( -1 == nRet ); ++nChild )
     264             :                     {
     265           0 :                         Reference< XAccessible > xChild( xParentContext->getAccessibleChild( nChild ) );
     266           0 :                         if ( xChild.get() == xCreator.get() )
     267           0 :                             nRet = nChild;
     268           0 :                     }
     269           0 :                 }
     270           0 :             }
     271             :         }
     272           0 :         catch( const Exception& )
     273             :         {
     274             :             OSL_FAIL( "OAccessibleContextHelper::getAccessibleIndexInParent: caught an exception!" );
     275             :         }
     276             : 
     277           0 :         return nRet;
     278             :     }
     279             : 
     280             : 
     281           0 :     Locale SAL_CALL OAccessibleContextHelper::getLocale(  ) throw (IllegalAccessibleComponentStateException, RuntimeException, std::exception)
     282             :     {
     283             :         // simply ask the parent
     284           0 :         Reference< XAccessible > xParent = getAccessibleParent();
     285           0 :         Reference< XAccessibleContext > xParentContext;
     286           0 :         if ( xParent.is() )
     287           0 :             xParentContext = xParent->getAccessibleContext();
     288             : 
     289           0 :         if ( !xParentContext.is() )
     290           0 :             throw IllegalAccessibleComponentStateException( OUString(), *this );
     291             : 
     292           0 :         return xParentContext->getLocale();
     293             :     }
     294             : 
     295             : 
     296        1430 :     Reference< XAccessibleContext > OAccessibleContextHelper::implGetParentContext()
     297             :     {
     298        1430 :         Reference< XAccessible > xParent = getAccessibleParent();
     299        1430 :         Reference< XAccessibleContext > xParentContext;
     300        1430 :         if ( xParent.is() )
     301        1430 :             xParentContext = xParent->getAccessibleContext();
     302        1430 :         return xParentContext;
     303             :     }
     304             : 
     305             : 
     306             : }   // namespace comphelper
     307             : 
     308             : 
     309             : 
     310             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.10