/* -*- 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 "EventThread.hxx" #include #include #include #include namespace frm { using namespace ::com::sun::star::uno; using namespace ::com::sun::star::awt; using namespace ::com::sun::star::lang; OComponentEventThread::OComponentEventThread( ::cppu::OComponentHelper* pCompImpl ) : m_pCompImpl( pCompImpl ) { osl_atomic_increment(&m_refCount); // Hold a reference of the Control { InterfaceRef xIFace(static_cast(pCompImpl)); m_xComp.set(xIFace, css::uno::UNO_QUERY); } // and add us at the Control { Reference xEvtLstnr = static_cast(this); m_xComp->addEventListener( xEvtLstnr ); } osl_atomic_decrement(&m_refCount); } OComponentEventThread::~OComponentEventThread() { DBG_ASSERT( m_aEvents.empty(), "OComponentEventThread::~OComponentEventThread: Didn't call dispose?" ); impl_clearEventQueue(); } Any SAL_CALL OComponentEventThread::queryInterface(const Type& _rType) throw (RuntimeException, std::exception) { Any aReturn; aReturn = OWeakObject::queryInterface(_rType); if (!aReturn.hasValue()) aReturn = ::cppu::queryInterface(_rType, static_cast(this) ); return aReturn; } void OComponentEventThread::impl_clearEventQueue() { while ( m_aEvents.size() ) { delete *m_aEvents.begin(); m_aEvents.erase( m_aEvents.begin() ); } m_aControls.erase( m_aControls.begin(), m_aControls.end() ); m_aFlags.erase( m_aFlags.begin(), m_aFlags.end() ); } void OComponentEventThread::disposing( const EventObject& evt ) throw ( ::com::sun::star::uno::RuntimeException, std::exception) { if( evt.Source == m_xComp ) { ::osl::MutexGuard aGuard( m_aMutex ); // Remove EventListener Reference xEvtLstnr = static_cast(this); m_xComp->removeEventListener( xEvtLstnr ); // Clear EventQueue impl_clearEventQueue(); // Free the Control and set pCompImpl to 0, // so that the thread knows, that it should terminate. m_xComp = 0; m_pCompImpl = 0; // Wake up the thread and terminate m_aCond.set(); terminate(); } } void OComponentEventThread::addEvent( const EventObject* _pEvt, bool bFlag ) { Reference xTmp; addEvent( _pEvt, xTmp, bFlag ); } void OComponentEventThread::addEvent( const EventObject* _pEvt, const Reference& rControl, bool bFlag ) { ::osl::MutexGuard aGuard( m_aMutex ); // Put data into the queue m_aEvents.push_back( cloneEvent( _pEvt ) ); Reference xWeakControl(rControl, UNO_QUERY); Reference xControlAdapter = xWeakControl.is() ? xWeakControl->queryAdapter() : Reference(); m_aControls.push_back( xControlAdapter ); m_aFlags.push_back( bFlag ); // Wake up thread m_aCond.set(); } void OComponentEventThread::implStarted( ) { acquire( ); } void OComponentEventThread::implTerminated( ) { release( ); } void SAL_CALL OComponentEventThread::onTerminated() { OComponentEventThread_TBASE::onTerminated(); implTerminated( ); } void OComponentEventThread::run() { osl_setThreadName("frm::OComponentEventThread"); implStarted( ); // Hold on to ourselves, so that we're not deleted if a dispose is called at some point in time InterfaceRef xThis(static_cast(this)); do { ::osl::MutexGuard aGuard(m_aMutex); while( m_aEvents.size() > 0 ) { // Get the Control and hold on to it so that it cannot be deleted during actionPerformed Reference xComp = m_xComp; ::cppu::OComponentHelper *pCompImpl = m_pCompImpl; ThreadEvents::iterator firstEvent( m_aEvents.begin() ); boost::scoped_ptr pEvt(*firstEvent); m_aEvents.erase( firstEvent ); ThreadObjects::iterator firstControl( m_aControls.begin() ); Reference xControlAdapter = *firstControl; m_aControls.erase( firstControl ); ThreadBools::iterator firstFlag( m_aFlags.begin() ); bool bFlag = *firstFlag; m_aFlags.erase( firstFlag ); { MutexRelease aReleaseOnce(m_aMutex); // Because a queryHardRef can throw an Exception, it should not be called when // the mutex is locked. Reference xControl; if ( xControlAdapter.is() ) xControl.set( xControlAdapter->queryAdapted(), css::uno::UNO_QUERY); if( xComp.is() ) processEvent( pCompImpl, pEvt.get(), xControl, bFlag ); } } // After a Dispose, we do not know the Control anymore. // Thus, we must not wait either. if( !m_xComp.is() ) return; // Reset waiting condition m_aCond.reset(); { MutexRelease aReleaseOnce(m_aMutex); // And wait ... if, in the meantime, an Event came in after all m_aCond.wait(); } } while( true ); } } // namespace frm /* vim:set shiftwidth=4 softtabstop=4 expandtab: */