1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
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/.
9 * This file incorporates work covered by the following license notice:
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 .
20 #include <sal/config.h>
24 #include <comphelper/proxyaggregation.hxx>
25 #include <com/sun/star/reflection/ProxyFactory.hpp>
32 using namespace ::com::sun::star::uno
;
33 using namespace ::com::sun::star::lang
;
34 using namespace ::com::sun::star::reflection
;
36 OProxyAggregation::OProxyAggregation( const Reference
< XComponentContext
>& _rxContext
)
37 :m_xContext( _rxContext
)
42 void OProxyAggregation::baseAggregateProxyFor( const Reference
< XInterface
>& _rxComponent
, oslInterlockedCount
& _rRefCount
,
43 ::cppu::OWeakObject
& _rDelegator
)
45 // first a factory for the proxy
46 Reference
< XProxyFactory
> xFactory
= ProxyFactory::create( m_xContext
);
48 // then the proxy itself
49 { // i36686 OJ: achieve the destruction of the temporary -> otherwise it leads to _rRefCount -= 2
50 m_xProxyAggregate
= xFactory
->createProxy( _rxComponent
);
52 if ( m_xProxyAggregate
.is() )
53 m_xProxyAggregate
->queryAggregation( cppu::UnoType
<decltype(m_xProxyTypeAccess
)>::get() ) >>= m_xProxyTypeAccess
;
55 // aggregate the proxy
56 osl_atomic_increment( &_rRefCount
);
57 if ( m_xProxyAggregate
.is() )
59 // At this point in time, the proxy has a ref count of exactly two - in m_xControlContextProxy,
60 // and in m_xProxyTypeAccess.
61 // Remember to _not_ reset these members unless the delegator of the proxy has been reset, too!
62 m_xProxyAggregate
->setDelegator( _rDelegator
);
64 osl_atomic_decrement( &_rRefCount
);
68 Any SAL_CALL
OProxyAggregation::queryAggregation( const Type
& _rType
) throw (RuntimeException
)
70 return m_xProxyAggregate
.is() ? m_xProxyAggregate
->queryAggregation( _rType
) : Any();
74 Sequence
< Type
> SAL_CALL
OProxyAggregation::getTypes( ) throw (RuntimeException
)
76 Sequence
< Type
> aTypes
;
77 if ( m_xProxyAggregate
.is() )
79 if ( m_xProxyTypeAccess
.is() )
80 aTypes
= m_xProxyTypeAccess
->getTypes();
86 OProxyAggregation::~OProxyAggregation()
88 if ( m_xProxyAggregate
.is() )
89 m_xProxyAggregate
->setDelegator( NULL
);
90 m_xProxyAggregate
.clear();
91 m_xProxyTypeAccess
.clear();
92 // this should remove the _two_only_ "real" references (means not delegated to
93 // ourself) to this proxy, and thus delete it
96 OComponentProxyAggregationHelper::OComponentProxyAggregationHelper( const Reference
< XComponentContext
>& _rxContext
,
97 ::cppu::OBroadcastHelper
& _rBHelper
)
98 :OProxyAggregation( _rxContext
)
99 ,m_rBHelper( _rBHelper
)
101 OSL_ENSURE( _rxContext
.is(), "OComponentProxyAggregationHelper::OComponentProxyAggregationHelper: invalid arguments!" );
105 void OComponentProxyAggregationHelper::componentAggregateProxyFor(
106 const Reference
< XComponent
>& _rxComponent
, oslInterlockedCount
& _rRefCount
,
107 ::cppu::OWeakObject
& _rDelegator
)
109 OSL_ENSURE( _rxComponent
.is(), "OComponentProxyAggregationHelper::componentAggregateProxyFor: invalid inner component!" );
110 m_xInner
= _rxComponent
;
112 // aggregate a proxy for the object
113 baseAggregateProxyFor( m_xInner
.get(), _rRefCount
, _rDelegator
);
115 // add as event listener to the inner context, because we want to be notified of disposals
116 osl_atomic_increment( &_rRefCount
);
119 m_xInner
->addEventListener( this );
121 osl_atomic_decrement( &_rRefCount
);
125 Any SAL_CALL
OComponentProxyAggregationHelper::queryInterface( const Type
& _rType
) throw (RuntimeException
, std::exception
)
127 Any
aReturn( BASE::queryInterface( _rType
) );
128 if ( !aReturn
.hasValue() )
129 aReturn
= OProxyAggregation::queryAggregation( _rType
);
134 IMPLEMENT_FORWARD_XTYPEPROVIDER2( OComponentProxyAggregationHelper
, BASE
, OProxyAggregation
)
137 OComponentProxyAggregationHelper::~OComponentProxyAggregationHelper( )
139 OSL_ENSURE( m_rBHelper
.bDisposed
, "OComponentProxyAggregationHelper::~OComponentProxyAggregationHelper: you should dispose your derived class in the dtor, if necessary!" );
140 // if this asserts, add the following to your derived class dtor:
142 // if ( !m_rBHelper.bDisposed )
144 // acquire(); // to prevent duplicate dtor calls
152 void SAL_CALL
OComponentProxyAggregationHelper::disposing( const EventObject
& _rSource
) throw (RuntimeException
, std::exception
)
154 if ( _rSource
.Source
== m_xInner
)
155 { // it's our inner context which is dying -> dispose ourself
156 if ( !m_rBHelper
.bDisposed
&& !m_rBHelper
.bInDispose
)
157 { // (if necessary only, of course)
164 void SAL_CALL
OComponentProxyAggregationHelper::dispose() throw( RuntimeException
, std::exception
)
166 ::osl::MutexGuard
aGuard( m_rBHelper
.rMutex
);
168 // dispose our inner context
169 // before we do this, remove ourself as listener - else in disposing( EventObject ), we
170 // would dispose ourself a second time
171 Reference
< XComponent
> xComp( m_xInner
, UNO_QUERY
);
174 xComp
->removeEventListener( this );
180 OComponentProxyAggregation::OComponentProxyAggregation( const Reference
< XComponentContext
>& _rxContext
,
181 const Reference
< XComponent
>& _rxComponent
)
182 :WeakComponentImplHelperBase( m_aMutex
)
183 ,OComponentProxyAggregationHelper( _rxContext
, rBHelper
)
185 OSL_ENSURE( _rxComponent
.is(), "OComponentProxyAggregation::OComponentProxyAggregation: accessible is no XComponent!" );
186 if ( _rxComponent
.is() )
187 componentAggregateProxyFor( _rxComponent
, m_refCount
, *this );
191 OComponentProxyAggregation::~OComponentProxyAggregation()
193 implEnsureDisposeInDtor( );
197 IMPLEMENT_FORWARD_XINTERFACE2( OComponentProxyAggregation
, WeakComponentImplHelperBase
, OComponentProxyAggregationHelper
)
200 IMPLEMENT_GET_IMPLEMENTATION_ID( OComponentProxyAggregation
)
203 Sequence
< Type
> SAL_CALL
OComponentProxyAggregation::getTypes( ) throw (RuntimeException
, std::exception
)
205 Sequence
< Type
> aTypes( OComponentProxyAggregationHelper::getTypes() );
207 // append XComponent, coming from WeakComponentImplHelperBase
208 sal_Int32 nLen
= aTypes
.getLength();
209 aTypes
.realloc( nLen
+ 1 );
210 aTypes
[ nLen
] = cppu::UnoType
<XComponent
>::get();
216 void OComponentProxyAggregation::implEnsureDisposeInDtor( )
218 if ( !rBHelper
.bDisposed
)
220 acquire(); // to prevent duplicate dtor calls
226 void SAL_CALL
OComponentProxyAggregation::disposing( const EventObject
& _rSource
) throw (RuntimeException
, std::exception
)
228 // Simply disambiguate---this is necessary for MSVC to distinguish
229 // "disposing(EventObject)" from "disposing()"; but it is also a good
230 // place to check for recursive calls that would be caused by an object
231 // being registered as an XEventListener at itself (cf. rhbz#928568):
232 assert(_rSource
.Source
!= static_cast< cppu::OWeakObject
* >(this));
233 OComponentProxyAggregationHelper::disposing( _rSource
);
237 void SAL_CALL
OComponentProxyAggregation::disposing() throw (RuntimeException
)
239 // call the dispose-functionality of the base, which will dispose our aggregated component
240 OComponentProxyAggregationHelper::dispose();
244 void SAL_CALL
OComponentProxyAggregation::dispose() throw( RuntimeException
, std::exception
)
246 // simply disambiguate
247 WeakComponentImplHelperBase::dispose();
252 } // namespace comphelper
255 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */