1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_cppuhelper.hxx"
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski #include <cppuhelper/interfacecontainer.hxx>
28*b1cdbd2cSJim Jagielski #include <cppuhelper/queryinterface.hxx>
29*b1cdbd2cSJim Jagielski #include <cppuhelper/propshlp.hxx>
30*b1cdbd2cSJim Jagielski 
31*b1cdbd2cSJim Jagielski #include <osl/diagnose.h>
32*b1cdbd2cSJim Jagielski #include <osl/mutex.hxx>
33*b1cdbd2cSJim Jagielski 
34*b1cdbd2cSJim Jagielski #include <hash_map>
35*b1cdbd2cSJim Jagielski 
36*b1cdbd2cSJim Jagielski #include <com/sun/star/lang/XEventListener.hpp>
37*b1cdbd2cSJim Jagielski 
38*b1cdbd2cSJim Jagielski 
39*b1cdbd2cSJim Jagielski using namespace osl;
40*b1cdbd2cSJim Jagielski using namespace com::sun::star::uno;
41*b1cdbd2cSJim Jagielski using namespace com::sun::star::lang;
42*b1cdbd2cSJim Jagielski 
43*b1cdbd2cSJim Jagielski namespace cppu
44*b1cdbd2cSJim Jagielski {
45*b1cdbd2cSJim Jagielski 
46*b1cdbd2cSJim Jagielski //===================================================================
47*b1cdbd2cSJim Jagielski //===================================================================
48*b1cdbd2cSJim Jagielski //===================================================================
49*b1cdbd2cSJim Jagielski /**
50*b1cdbd2cSJim Jagielski  * Reallocate the sequence.
51*b1cdbd2cSJim Jagielski  */
realloc(Sequence<Reference<XInterface>> & rSeq,sal_Int32 nNewLen)52*b1cdbd2cSJim Jagielski static void realloc( Sequence< Reference< XInterface > > & rSeq, sal_Int32 nNewLen )
53*b1cdbd2cSJim Jagielski 	SAL_THROW( () )
54*b1cdbd2cSJim Jagielski {
55*b1cdbd2cSJim Jagielski 	rSeq.realloc( nNewLen );
56*b1cdbd2cSJim Jagielski }
57*b1cdbd2cSJim Jagielski 
58*b1cdbd2cSJim Jagielski /**
59*b1cdbd2cSJim Jagielski  * Remove an element from an interface sequence.
60*b1cdbd2cSJim Jagielski  */
sequenceRemoveElementAt(Sequence<Reference<XInterface>> & rSeq,sal_Int32 index)61*b1cdbd2cSJim Jagielski static void sequenceRemoveElementAt( Sequence< Reference< XInterface > > & rSeq, sal_Int32 index )
62*b1cdbd2cSJim Jagielski 	SAL_THROW( () )
63*b1cdbd2cSJim Jagielski {
64*b1cdbd2cSJim Jagielski 	sal_Int32 nNewLen = rSeq.getLength() - 1;
65*b1cdbd2cSJim Jagielski 
66*b1cdbd2cSJim Jagielski 	Sequence< Reference< XInterface > > aDestSeq( rSeq.getLength() - 1 );
67*b1cdbd2cSJim Jagielski 	// getArray on a const sequence is faster
68*b1cdbd2cSJim Jagielski 	const Reference< XInterface > * pSource = ((const Sequence< Reference< XInterface > > &)rSeq).getConstArray();
69*b1cdbd2cSJim Jagielski 	Reference< XInterface > * pDest = aDestSeq.getArray();
70*b1cdbd2cSJim Jagielski 	sal_Int32 i = 0;
71*b1cdbd2cSJim Jagielski 	for( ; i < index; i++ )
72*b1cdbd2cSJim Jagielski 		pDest[i] = pSource[i];
73*b1cdbd2cSJim Jagielski 	for( sal_Int32 j = i ; j < nNewLen; j++ )
74*b1cdbd2cSJim Jagielski 		pDest[j] = pSource[j+1];
75*b1cdbd2cSJim Jagielski 	rSeq = aDestSeq;
76*b1cdbd2cSJim Jagielski }
77*b1cdbd2cSJim Jagielski 
78*b1cdbd2cSJim Jagielski 
79*b1cdbd2cSJim Jagielski //-----------------------------------------------------------------------------
80*b1cdbd2cSJim Jagielski //-----------------------------------------------------------------------------
81*b1cdbd2cSJim Jagielski 
82*b1cdbd2cSJim Jagielski #ifdef _MSC_VER
83*b1cdbd2cSJim Jagielski #pragma warning( disable: 4786 )
84*b1cdbd2cSJim Jagielski #endif
85*b1cdbd2cSJim Jagielski 
86*b1cdbd2cSJim Jagielski //===================================================================
87*b1cdbd2cSJim Jagielski //===================================================================
88*b1cdbd2cSJim Jagielski //===================================================================
OInterfaceIteratorHelper(OInterfaceContainerHelper & rCont_)89*b1cdbd2cSJim Jagielski OInterfaceIteratorHelper::OInterfaceIteratorHelper( OInterfaceContainerHelper & rCont_ )
90*b1cdbd2cSJim Jagielski 	SAL_THROW( () )
91*b1cdbd2cSJim Jagielski 	: rCont( rCont_ )
92*b1cdbd2cSJim Jagielski {
93*b1cdbd2cSJim Jagielski 	MutexGuard aGuard( rCont.rMutex );
94*b1cdbd2cSJim Jagielski 	if( rCont.bInUse )
95*b1cdbd2cSJim Jagielski 		// worst case, two iterators at the same time
96*b1cdbd2cSJim Jagielski 		rCont.copyAndResetInUse();
97*b1cdbd2cSJim Jagielski 	bIsList = rCont_.bIsList;
98*b1cdbd2cSJim Jagielski 	aData = rCont_.aData;
99*b1cdbd2cSJim Jagielski 	if( bIsList )
100*b1cdbd2cSJim Jagielski 	{
101*b1cdbd2cSJim Jagielski 		rCont.bInUse = sal_True;
102*b1cdbd2cSJim Jagielski 		nRemain = aData.pAsSequence->getLength();
103*b1cdbd2cSJim Jagielski 	}
104*b1cdbd2cSJim Jagielski 	else if( aData.pAsInterface )
105*b1cdbd2cSJim Jagielski 	{
106*b1cdbd2cSJim Jagielski 		aData.pAsInterface->acquire();
107*b1cdbd2cSJim Jagielski 		nRemain = 1;
108*b1cdbd2cSJim Jagielski 	}
109*b1cdbd2cSJim Jagielski 	else
110*b1cdbd2cSJim Jagielski 		nRemain = 0;
111*b1cdbd2cSJim Jagielski }
112*b1cdbd2cSJim Jagielski 
~OInterfaceIteratorHelper()113*b1cdbd2cSJim Jagielski OInterfaceIteratorHelper::~OInterfaceIteratorHelper() SAL_THROW( () )
114*b1cdbd2cSJim Jagielski {
115*b1cdbd2cSJim Jagielski 	sal_Bool bShared;
116*b1cdbd2cSJim Jagielski 	{
117*b1cdbd2cSJim Jagielski 	MutexGuard aGuard( rCont.rMutex );
118*b1cdbd2cSJim Jagielski 	// bResetInUse protect the iterator against recursion
119*b1cdbd2cSJim Jagielski 	bShared = aData.pAsSequence == rCont.aData.pAsSequence && rCont.bIsList;
120*b1cdbd2cSJim Jagielski 	if( bShared )
121*b1cdbd2cSJim Jagielski 	{
122*b1cdbd2cSJim Jagielski 		OSL_ENSURE( rCont.bInUse, "OInterfaceContainerHelper must be in use" );
123*b1cdbd2cSJim Jagielski 		rCont.bInUse = sal_False;
124*b1cdbd2cSJim Jagielski 	}
125*b1cdbd2cSJim Jagielski 	}
126*b1cdbd2cSJim Jagielski 
127*b1cdbd2cSJim Jagielski 	if( !bShared )
128*b1cdbd2cSJim Jagielski 	{
129*b1cdbd2cSJim Jagielski 		if( bIsList )
130*b1cdbd2cSJim Jagielski 			// Sequence owned by the iterator
131*b1cdbd2cSJim Jagielski 			delete aData.pAsSequence;
132*b1cdbd2cSJim Jagielski 		else if( aData.pAsInterface )
133*b1cdbd2cSJim Jagielski 			// Interface is acquired by the iterator
134*b1cdbd2cSJim Jagielski 			aData.pAsInterface->release();
135*b1cdbd2cSJim Jagielski 	}
136*b1cdbd2cSJim Jagielski }
137*b1cdbd2cSJim Jagielski 
next()138*b1cdbd2cSJim Jagielski XInterface * OInterfaceIteratorHelper::next() SAL_THROW( () )
139*b1cdbd2cSJim Jagielski {
140*b1cdbd2cSJim Jagielski 	if( nRemain )
141*b1cdbd2cSJim Jagielski 	{
142*b1cdbd2cSJim Jagielski 		nRemain--;
143*b1cdbd2cSJim Jagielski 		if( bIsList )
144*b1cdbd2cSJim Jagielski 			// typecase to const,so the getArray method is faster
145*b1cdbd2cSJim Jagielski 			return aData.pAsSequence->getConstArray()[nRemain].get();
146*b1cdbd2cSJim Jagielski 		else if( aData.pAsInterface )
147*b1cdbd2cSJim Jagielski 			return aData.pAsInterface;
148*b1cdbd2cSJim Jagielski 	}
149*b1cdbd2cSJim Jagielski 	// exception
150*b1cdbd2cSJim Jagielski 	return 0;
151*b1cdbd2cSJim Jagielski }
152*b1cdbd2cSJim Jagielski 
remove()153*b1cdbd2cSJim Jagielski void OInterfaceIteratorHelper::remove() SAL_THROW( () )
154*b1cdbd2cSJim Jagielski {
155*b1cdbd2cSJim Jagielski 	if( bIsList )
156*b1cdbd2cSJim Jagielski 	{
157*b1cdbd2cSJim Jagielski 		OSL_ASSERT( nRemain >= 0 &&
158*b1cdbd2cSJim Jagielski 					nRemain < aData.pAsSequence->getLength() );
159*b1cdbd2cSJim Jagielski         XInterface * p = aData.pAsSequence->getConstArray()[nRemain].get();
160*b1cdbd2cSJim Jagielski 		rCont.removeInterface( * reinterpret_cast< const Reference< XInterface > * >( &p ) );
161*b1cdbd2cSJim Jagielski 	}
162*b1cdbd2cSJim Jagielski 	else
163*b1cdbd2cSJim Jagielski 	{
164*b1cdbd2cSJim Jagielski 		OSL_ASSERT( 0 == nRemain );
165*b1cdbd2cSJim Jagielski 		rCont.removeInterface( * reinterpret_cast< const Reference< XInterface > * >(&aData.pAsInterface));
166*b1cdbd2cSJim Jagielski 	}
167*b1cdbd2cSJim Jagielski }
168*b1cdbd2cSJim Jagielski 
169*b1cdbd2cSJim Jagielski //===================================================================
170*b1cdbd2cSJim Jagielski //===================================================================
171*b1cdbd2cSJim Jagielski //===================================================================
172*b1cdbd2cSJim Jagielski 
173*b1cdbd2cSJim Jagielski 
OInterfaceContainerHelper(Mutex & rMutex_)174*b1cdbd2cSJim Jagielski OInterfaceContainerHelper::OInterfaceContainerHelper( Mutex & rMutex_ ) SAL_THROW( () )
175*b1cdbd2cSJim Jagielski 	: rMutex( rMutex_ )
176*b1cdbd2cSJim Jagielski 	, bInUse( sal_False )
177*b1cdbd2cSJim Jagielski 	, bIsList( sal_False )
178*b1cdbd2cSJim Jagielski {
179*b1cdbd2cSJim Jagielski }
180*b1cdbd2cSJim Jagielski 
~OInterfaceContainerHelper()181*b1cdbd2cSJim Jagielski OInterfaceContainerHelper::~OInterfaceContainerHelper() SAL_THROW( () )
182*b1cdbd2cSJim Jagielski {
183*b1cdbd2cSJim Jagielski 	OSL_ENSURE( !bInUse, "~OInterfaceContainerHelper but is in use" );
184*b1cdbd2cSJim Jagielski 	if( bIsList )
185*b1cdbd2cSJim Jagielski 		delete aData.pAsSequence;
186*b1cdbd2cSJim Jagielski 	else if( aData.pAsInterface )
187*b1cdbd2cSJim Jagielski 		aData.pAsInterface->release();
188*b1cdbd2cSJim Jagielski }
189*b1cdbd2cSJim Jagielski 
getLength() const190*b1cdbd2cSJim Jagielski sal_Int32 OInterfaceContainerHelper::getLength() const SAL_THROW( () )
191*b1cdbd2cSJim Jagielski {
192*b1cdbd2cSJim Jagielski 	MutexGuard aGuard( rMutex );
193*b1cdbd2cSJim Jagielski 	if( bIsList )
194*b1cdbd2cSJim Jagielski 		return aData.pAsSequence->getLength();
195*b1cdbd2cSJim Jagielski 	else if( aData.pAsInterface )
196*b1cdbd2cSJim Jagielski 		return 1;
197*b1cdbd2cSJim Jagielski 	return 0;
198*b1cdbd2cSJim Jagielski }
199*b1cdbd2cSJim Jagielski 
getElements() const200*b1cdbd2cSJim Jagielski Sequence< Reference<XInterface> > OInterfaceContainerHelper::getElements() const SAL_THROW( () )
201*b1cdbd2cSJim Jagielski {
202*b1cdbd2cSJim Jagielski 	MutexGuard aGuard( rMutex );
203*b1cdbd2cSJim Jagielski 	if( bIsList )
204*b1cdbd2cSJim Jagielski 		return *aData.pAsSequence;
205*b1cdbd2cSJim Jagielski 	else if( aData.pAsInterface )
206*b1cdbd2cSJim Jagielski 	{
207*b1cdbd2cSJim Jagielski 		Reference<XInterface> x( aData.pAsInterface );
208*b1cdbd2cSJim Jagielski 		return Sequence< Reference< XInterface > >( &x, 1 );
209*b1cdbd2cSJim Jagielski 	}
210*b1cdbd2cSJim Jagielski 	return Sequence< Reference< XInterface > >();
211*b1cdbd2cSJim Jagielski }
212*b1cdbd2cSJim Jagielski 
copyAndResetInUse()213*b1cdbd2cSJim Jagielski void OInterfaceContainerHelper::copyAndResetInUse() SAL_THROW( () )
214*b1cdbd2cSJim Jagielski {
215*b1cdbd2cSJim Jagielski 	OSL_ENSURE( bInUse, "OInterfaceContainerHelper not in use" );
216*b1cdbd2cSJim Jagielski 	if( bInUse )
217*b1cdbd2cSJim Jagielski 	{
218*b1cdbd2cSJim Jagielski 		// this should be the worst case. If a iterator is active
219*b1cdbd2cSJim Jagielski 		// and a new Listener is added.
220*b1cdbd2cSJim Jagielski 		if( bIsList )
221*b1cdbd2cSJim Jagielski 			aData.pAsSequence = new Sequence< Reference< XInterface > >( *aData.pAsSequence );
222*b1cdbd2cSJim Jagielski 		else if( aData.pAsInterface )
223*b1cdbd2cSJim Jagielski 			aData.pAsInterface->acquire();
224*b1cdbd2cSJim Jagielski 
225*b1cdbd2cSJim Jagielski 		bInUse = sal_False;
226*b1cdbd2cSJim Jagielski 	}
227*b1cdbd2cSJim Jagielski }
228*b1cdbd2cSJim Jagielski 
addInterface(const Reference<XInterface> & rListener)229*b1cdbd2cSJim Jagielski sal_Int32 OInterfaceContainerHelper::addInterface( const Reference<XInterface> & rListener ) SAL_THROW( () )
230*b1cdbd2cSJim Jagielski {
231*b1cdbd2cSJim Jagielski 	OSL_ASSERT( rListener.is() );
232*b1cdbd2cSJim Jagielski 	MutexGuard aGuard( rMutex );
233*b1cdbd2cSJim Jagielski 	if( bInUse )
234*b1cdbd2cSJim Jagielski 		copyAndResetInUse();
235*b1cdbd2cSJim Jagielski 
236*b1cdbd2cSJim Jagielski 	if( bIsList )
237*b1cdbd2cSJim Jagielski 	{
238*b1cdbd2cSJim Jagielski 		sal_Int32 nLen = aData.pAsSequence->getLength();
239*b1cdbd2cSJim Jagielski 		realloc( *aData.pAsSequence, nLen +1 );
240*b1cdbd2cSJim Jagielski 		aData.pAsSequence->getArray()[ nLen ] = rListener;
241*b1cdbd2cSJim Jagielski 		return nLen +1;
242*b1cdbd2cSJim Jagielski 	}
243*b1cdbd2cSJim Jagielski 	else if( aData.pAsInterface )
244*b1cdbd2cSJim Jagielski 	{
245*b1cdbd2cSJim Jagielski 		Sequence< Reference< XInterface > > * pSeq = new Sequence< Reference< XInterface > >( 2 );
246*b1cdbd2cSJim Jagielski 		Reference<XInterface> * pArray = pSeq->getArray();
247*b1cdbd2cSJim Jagielski 		pArray[0] = aData.pAsInterface;
248*b1cdbd2cSJim Jagielski 		pArray[1] = rListener;
249*b1cdbd2cSJim Jagielski 		aData.pAsInterface->release();
250*b1cdbd2cSJim Jagielski 		aData.pAsSequence = pSeq;
251*b1cdbd2cSJim Jagielski 		bIsList = sal_True;
252*b1cdbd2cSJim Jagielski 		return 2;
253*b1cdbd2cSJim Jagielski 	}
254*b1cdbd2cSJim Jagielski 	else
255*b1cdbd2cSJim Jagielski 	{
256*b1cdbd2cSJim Jagielski 		aData.pAsInterface = rListener.get();
257*b1cdbd2cSJim Jagielski 		if( rListener.is() )
258*b1cdbd2cSJim Jagielski 			rListener->acquire();
259*b1cdbd2cSJim Jagielski 		return 1;
260*b1cdbd2cSJim Jagielski 	}
261*b1cdbd2cSJim Jagielski }
262*b1cdbd2cSJim Jagielski 
removeInterface(const Reference<XInterface> & rListener)263*b1cdbd2cSJim Jagielski sal_Int32 OInterfaceContainerHelper::removeInterface( const Reference<XInterface> & rListener ) SAL_THROW( () )
264*b1cdbd2cSJim Jagielski {
265*b1cdbd2cSJim Jagielski 	OSL_ASSERT( rListener.is() );
266*b1cdbd2cSJim Jagielski 	MutexGuard aGuard( rMutex );
267*b1cdbd2cSJim Jagielski 	if( bInUse )
268*b1cdbd2cSJim Jagielski 		copyAndResetInUse();
269*b1cdbd2cSJim Jagielski 
270*b1cdbd2cSJim Jagielski 	if( bIsList )
271*b1cdbd2cSJim Jagielski 	{
272*b1cdbd2cSJim Jagielski 		const Reference<XInterface> * pL = aData.pAsSequence->getConstArray();
273*b1cdbd2cSJim Jagielski 		sal_Int32 nLen = aData.pAsSequence->getLength();
274*b1cdbd2cSJim Jagielski 		sal_Int32 i;
275*b1cdbd2cSJim Jagielski 		for( i = 0; i < nLen; i++ )
276*b1cdbd2cSJim Jagielski 		{
277*b1cdbd2cSJim Jagielski 			// It is not valid to compare the Pointer direkt, but is is is much
278*b1cdbd2cSJim Jagielski 			// more faster.
279*b1cdbd2cSJim Jagielski 			if( pL[i].get() == rListener.get() )
280*b1cdbd2cSJim Jagielski 			{
281*b1cdbd2cSJim Jagielski 				sequenceRemoveElementAt( *aData.pAsSequence, i );
282*b1cdbd2cSJim Jagielski 				break;
283*b1cdbd2cSJim Jagielski 			}
284*b1cdbd2cSJim Jagielski 		}
285*b1cdbd2cSJim Jagielski 
286*b1cdbd2cSJim Jagielski 		if( i == nLen )
287*b1cdbd2cSJim Jagielski 		{
288*b1cdbd2cSJim Jagielski 			// interface not found, use the correct compare method
289*b1cdbd2cSJim Jagielski 			for( i = 0; i < nLen; i++ )
290*b1cdbd2cSJim Jagielski 			{
291*b1cdbd2cSJim Jagielski 				if( pL[i] == rListener )
292*b1cdbd2cSJim Jagielski 				{
293*b1cdbd2cSJim Jagielski 					sequenceRemoveElementAt(*aData.pAsSequence, i );
294*b1cdbd2cSJim Jagielski 					break;
295*b1cdbd2cSJim Jagielski 				}
296*b1cdbd2cSJim Jagielski 			}
297*b1cdbd2cSJim Jagielski 		}
298*b1cdbd2cSJim Jagielski 
299*b1cdbd2cSJim Jagielski 		if( aData.pAsSequence->getLength() == 1 )
300*b1cdbd2cSJim Jagielski 		{
301*b1cdbd2cSJim Jagielski 			XInterface * p = aData.pAsSequence->getConstArray()[0].get();
302*b1cdbd2cSJim Jagielski 			p->acquire();
303*b1cdbd2cSJim Jagielski 			delete aData.pAsSequence;
304*b1cdbd2cSJim Jagielski 			aData.pAsInterface = p;
305*b1cdbd2cSJim Jagielski 			bIsList = sal_False;
306*b1cdbd2cSJim Jagielski 			return 1;
307*b1cdbd2cSJim Jagielski 		}
308*b1cdbd2cSJim Jagielski 		else
309*b1cdbd2cSJim Jagielski 			return aData.pAsSequence->getLength();
310*b1cdbd2cSJim Jagielski 	}
311*b1cdbd2cSJim Jagielski 	else if( aData.pAsInterface && Reference<XInterface>( aData.pAsInterface ) == rListener )
312*b1cdbd2cSJim Jagielski 	{
313*b1cdbd2cSJim Jagielski 		aData.pAsInterface->release();
314*b1cdbd2cSJim Jagielski 		aData.pAsInterface = 0;
315*b1cdbd2cSJim Jagielski 	}
316*b1cdbd2cSJim Jagielski 	return aData.pAsInterface ? 1 : 0;
317*b1cdbd2cSJim Jagielski }
318*b1cdbd2cSJim Jagielski 
disposeAndClear(const EventObject & rEvt)319*b1cdbd2cSJim Jagielski void OInterfaceContainerHelper::disposeAndClear( const EventObject & rEvt ) SAL_THROW( () )
320*b1cdbd2cSJim Jagielski {
321*b1cdbd2cSJim Jagielski 	ClearableMutexGuard aGuard( rMutex );
322*b1cdbd2cSJim Jagielski 	OInterfaceIteratorHelper aIt( *this );
323*b1cdbd2cSJim Jagielski 	// Container freigeben, falls im disposing neue Eintr�ge kommen
324*b1cdbd2cSJim Jagielski 	OSL_ENSURE( !bIsList || bInUse, "OInterfaceContainerHelper not in use" );
325*b1cdbd2cSJim Jagielski 	if( !bIsList && aData.pAsInterface )
326*b1cdbd2cSJim Jagielski 		aData.pAsInterface->release();
327*b1cdbd2cSJim Jagielski 	// set the member to null, the iterator delete the values
328*b1cdbd2cSJim Jagielski 	aData.pAsInterface = NULL;
329*b1cdbd2cSJim Jagielski 	bIsList = sal_False;
330*b1cdbd2cSJim Jagielski 	bInUse = sal_False;
331*b1cdbd2cSJim Jagielski 	aGuard.clear();
332*b1cdbd2cSJim Jagielski 	while( aIt.hasMoreElements() )
333*b1cdbd2cSJim Jagielski 	{
334*b1cdbd2cSJim Jagielski 		try
335*b1cdbd2cSJim Jagielski 		{
336*b1cdbd2cSJim Jagielski 			Reference<XEventListener > xLst( aIt.next(), UNO_QUERY );
337*b1cdbd2cSJim Jagielski 			if( xLst.is() )
338*b1cdbd2cSJim Jagielski 				xLst->disposing( rEvt );
339*b1cdbd2cSJim Jagielski 		}
340*b1cdbd2cSJim Jagielski 		catch ( RuntimeException & )
341*b1cdbd2cSJim Jagielski 		{
342*b1cdbd2cSJim Jagielski 			// be robust, if e.g. a remote bridge has disposed already.
343*b1cdbd2cSJim Jagielski 			// there is no way, to delegate the error to the caller :o(.
344*b1cdbd2cSJim Jagielski 		}
345*b1cdbd2cSJim Jagielski 	}
346*b1cdbd2cSJim Jagielski }
347*b1cdbd2cSJim Jagielski 
348*b1cdbd2cSJim Jagielski 
clear()349*b1cdbd2cSJim Jagielski void OInterfaceContainerHelper::clear() SAL_THROW( () )
350*b1cdbd2cSJim Jagielski {
351*b1cdbd2cSJim Jagielski 	ClearableMutexGuard aGuard( rMutex );
352*b1cdbd2cSJim Jagielski 	OInterfaceIteratorHelper aIt( *this );
353*b1cdbd2cSJim Jagielski 	// Container freigeben, falls im disposing neue Eintr�ge kommen
354*b1cdbd2cSJim Jagielski 	OSL_ENSURE( !bIsList || bInUse, "OInterfaceContainerHelper not in use" );
355*b1cdbd2cSJim Jagielski 	if( !bIsList && aData.pAsInterface )
356*b1cdbd2cSJim Jagielski 		aData.pAsInterface->release();
357*b1cdbd2cSJim Jagielski 	// set the member to null, the iterator delete the values
358*b1cdbd2cSJim Jagielski 	aData.pAsInterface = 0;
359*b1cdbd2cSJim Jagielski 	bIsList = sal_False;
360*b1cdbd2cSJim Jagielski 	bInUse = sal_False;
361*b1cdbd2cSJim Jagielski 	// release mutex before aIt destructor call
362*b1cdbd2cSJim Jagielski 	aGuard.clear();
363*b1cdbd2cSJim Jagielski }
364*b1cdbd2cSJim Jagielski 
365*b1cdbd2cSJim Jagielski //##################################################################################################
366*b1cdbd2cSJim Jagielski //##################################################################################################
367*b1cdbd2cSJim Jagielski //##################################################################################################
368*b1cdbd2cSJim Jagielski 
369*b1cdbd2cSJim Jagielski // specialized class for type
370*b1cdbd2cSJim Jagielski 
371*b1cdbd2cSJim Jagielski typedef ::std::vector< std::pair < Type , void* > > t_type2ptr;
372*b1cdbd2cSJim Jagielski 
OMultiTypeInterfaceContainerHelper(Mutex & rMutex_)373*b1cdbd2cSJim Jagielski OMultiTypeInterfaceContainerHelper::OMultiTypeInterfaceContainerHelper( Mutex & rMutex_ )
374*b1cdbd2cSJim Jagielski     SAL_THROW( () )
375*b1cdbd2cSJim Jagielski     : rMutex( rMutex_ )
376*b1cdbd2cSJim Jagielski {
377*b1cdbd2cSJim Jagielski 	m_pMap = new t_type2ptr();
378*b1cdbd2cSJim Jagielski }
~OMultiTypeInterfaceContainerHelper()379*b1cdbd2cSJim Jagielski OMultiTypeInterfaceContainerHelper::~OMultiTypeInterfaceContainerHelper()
380*b1cdbd2cSJim Jagielski     SAL_THROW( () )
381*b1cdbd2cSJim Jagielski {
382*b1cdbd2cSJim Jagielski     t_type2ptr * pMap = (t_type2ptr *)m_pMap;
383*b1cdbd2cSJim Jagielski 	t_type2ptr::iterator iter = pMap->begin();
384*b1cdbd2cSJim Jagielski 	t_type2ptr::iterator end = pMap->end();
385*b1cdbd2cSJim Jagielski 
386*b1cdbd2cSJim Jagielski 	while( iter != end )
387*b1cdbd2cSJim Jagielski 	{
388*b1cdbd2cSJim Jagielski 		delete (OInterfaceContainerHelper*)(*iter).second;
389*b1cdbd2cSJim Jagielski 		(*iter).second = 0;
390*b1cdbd2cSJim Jagielski 		++iter;
391*b1cdbd2cSJim Jagielski 	}
392*b1cdbd2cSJim Jagielski 	delete pMap;
393*b1cdbd2cSJim Jagielski }
getContainedTypes() const394*b1cdbd2cSJim Jagielski Sequence< Type > OMultiTypeInterfaceContainerHelper::getContainedTypes() const
395*b1cdbd2cSJim Jagielski     SAL_THROW( () )
396*b1cdbd2cSJim Jagielski {
397*b1cdbd2cSJim Jagielski     t_type2ptr * pMap = (t_type2ptr *)m_pMap;
398*b1cdbd2cSJim Jagielski 	t_type2ptr::size_type nSize;
399*b1cdbd2cSJim Jagielski 
400*b1cdbd2cSJim Jagielski 	::osl::MutexGuard aGuard( rMutex );
401*b1cdbd2cSJim Jagielski 	nSize = pMap->size();
402*b1cdbd2cSJim Jagielski 	if( nSize )
403*b1cdbd2cSJim Jagielski 	{
404*b1cdbd2cSJim Jagielski 		::com::sun::star::uno::Sequence< Type > aInterfaceTypes( nSize );
405*b1cdbd2cSJim Jagielski 		Type * pArray = aInterfaceTypes.getArray();
406*b1cdbd2cSJim Jagielski 
407*b1cdbd2cSJim Jagielski 		t_type2ptr::iterator iter = pMap->begin();
408*b1cdbd2cSJim Jagielski 		t_type2ptr::iterator end = pMap->end();
409*b1cdbd2cSJim Jagielski 
410*b1cdbd2cSJim Jagielski 		sal_Int32 i = 0;
411*b1cdbd2cSJim Jagielski 		while( iter != end )
412*b1cdbd2cSJim Jagielski 		{
413*b1cdbd2cSJim Jagielski 			// are interfaces added to this container?
414*b1cdbd2cSJim Jagielski 			if( ((OInterfaceContainerHelper*)(*iter).second)->getLength() )
415*b1cdbd2cSJim Jagielski 				// yes, put the type in the array
416*b1cdbd2cSJim Jagielski 				pArray[i++] = (*iter).first;
417*b1cdbd2cSJim Jagielski 			++iter;
418*b1cdbd2cSJim Jagielski 		}
419*b1cdbd2cSJim Jagielski 		if( (t_type2ptr::size_type)i != nSize ) {
420*b1cdbd2cSJim Jagielski 			// may be empty container, reduce the sequence to the right size
421*b1cdbd2cSJim Jagielski 			aInterfaceTypes = ::com::sun::star::uno::Sequence< Type >( pArray, i );
422*b1cdbd2cSJim Jagielski 		}
423*b1cdbd2cSJim Jagielski 		return aInterfaceTypes;
424*b1cdbd2cSJim Jagielski 	}
425*b1cdbd2cSJim Jagielski 	return ::com::sun::star::uno::Sequence< Type >();
426*b1cdbd2cSJim Jagielski }
427*b1cdbd2cSJim Jagielski 
findType(t_type2ptr * pMap,const Type & rKey)428*b1cdbd2cSJim Jagielski static t_type2ptr::iterator findType(t_type2ptr *pMap, const Type & rKey )
429*b1cdbd2cSJim Jagielski {
430*b1cdbd2cSJim Jagielski 	t_type2ptr::iterator iter = pMap->begin();
431*b1cdbd2cSJim Jagielski 	t_type2ptr::iterator end = pMap->end();
432*b1cdbd2cSJim Jagielski 
433*b1cdbd2cSJim Jagielski 	while( iter != end )
434*b1cdbd2cSJim Jagielski     {
435*b1cdbd2cSJim Jagielski         if (iter->first == rKey)
436*b1cdbd2cSJim Jagielski             break;
437*b1cdbd2cSJim Jagielski         iter++;
438*b1cdbd2cSJim Jagielski     }
439*b1cdbd2cSJim Jagielski     return iter;
440*b1cdbd2cSJim Jagielski }
441*b1cdbd2cSJim Jagielski 
getContainer(const Type & rKey) const442*b1cdbd2cSJim Jagielski OInterfaceContainerHelper * OMultiTypeInterfaceContainerHelper::getContainer( const Type & rKey ) const
443*b1cdbd2cSJim Jagielski     SAL_THROW( () )
444*b1cdbd2cSJim Jagielski {
445*b1cdbd2cSJim Jagielski 	::osl::MutexGuard aGuard( rMutex );
446*b1cdbd2cSJim Jagielski 
447*b1cdbd2cSJim Jagielski     t_type2ptr * pMap = (t_type2ptr *)m_pMap;
448*b1cdbd2cSJim Jagielski  	t_type2ptr::iterator iter = findType( pMap, rKey );
449*b1cdbd2cSJim Jagielski 	if( iter != pMap->end() )
450*b1cdbd2cSJim Jagielski 			return (OInterfaceContainerHelper*) (*iter).second;
451*b1cdbd2cSJim Jagielski 	return 0;
452*b1cdbd2cSJim Jagielski }
addInterface(const Type & rKey,const Reference<XInterface> & rListener)453*b1cdbd2cSJim Jagielski sal_Int32 OMultiTypeInterfaceContainerHelper::addInterface(
454*b1cdbd2cSJim Jagielski     const Type & rKey, const Reference< XInterface > & rListener )
455*b1cdbd2cSJim Jagielski     SAL_THROW( () )
456*b1cdbd2cSJim Jagielski {
457*b1cdbd2cSJim Jagielski 	::osl::MutexGuard aGuard( rMutex );
458*b1cdbd2cSJim Jagielski     t_type2ptr * pMap = (t_type2ptr *)m_pMap;
459*b1cdbd2cSJim Jagielski 	t_type2ptr::iterator iter = findType( pMap, rKey );
460*b1cdbd2cSJim Jagielski 	if( iter == pMap->end() )
461*b1cdbd2cSJim Jagielski 	{
462*b1cdbd2cSJim Jagielski 		OInterfaceContainerHelper * pLC = new OInterfaceContainerHelper( rMutex );
463*b1cdbd2cSJim Jagielski         pMap->push_back(std::pair<Type, void*>(rKey, pLC));
464*b1cdbd2cSJim Jagielski 		return pLC->addInterface( rListener );
465*b1cdbd2cSJim Jagielski 	}
466*b1cdbd2cSJim Jagielski 	else
467*b1cdbd2cSJim Jagielski 		return ((OInterfaceContainerHelper*)(*iter).second)->addInterface( rListener );
468*b1cdbd2cSJim Jagielski }
removeInterface(const Type & rKey,const Reference<XInterface> & rListener)469*b1cdbd2cSJim Jagielski sal_Int32 OMultiTypeInterfaceContainerHelper::removeInterface(
470*b1cdbd2cSJim Jagielski     const Type & rKey, const Reference< XInterface > & rListener )
471*b1cdbd2cSJim Jagielski     SAL_THROW( () )
472*b1cdbd2cSJim Jagielski {
473*b1cdbd2cSJim Jagielski 	::osl::MutexGuard aGuard( rMutex );
474*b1cdbd2cSJim Jagielski 
475*b1cdbd2cSJim Jagielski 	// search container with id nUik
476*b1cdbd2cSJim Jagielski     t_type2ptr * pMap = (t_type2ptr *)m_pMap;
477*b1cdbd2cSJim Jagielski 	t_type2ptr::iterator iter = findType( pMap, rKey );
478*b1cdbd2cSJim Jagielski 		// container found?
479*b1cdbd2cSJim Jagielski 	if( iter != pMap->end() )
480*b1cdbd2cSJim Jagielski         return ((OInterfaceContainerHelper*)(*iter).second)->removeInterface( rListener );
481*b1cdbd2cSJim Jagielski 
482*b1cdbd2cSJim Jagielski 	// no container with this id. Always return 0
483*b1cdbd2cSJim Jagielski 	return 0;
484*b1cdbd2cSJim Jagielski }
disposeAndClear(const EventObject & rEvt)485*b1cdbd2cSJim Jagielski void OMultiTypeInterfaceContainerHelper::disposeAndClear( const EventObject & rEvt )
486*b1cdbd2cSJim Jagielski     SAL_THROW( () )
487*b1cdbd2cSJim Jagielski {
488*b1cdbd2cSJim Jagielski 	t_type2ptr::size_type nSize = 0;
489*b1cdbd2cSJim Jagielski 	OInterfaceContainerHelper ** ppListenerContainers = NULL;
490*b1cdbd2cSJim Jagielski 	{
491*b1cdbd2cSJim Jagielski 		::osl::MutexGuard aGuard( rMutex );
492*b1cdbd2cSJim Jagielski         t_type2ptr * pMap = (t_type2ptr *)m_pMap;
493*b1cdbd2cSJim Jagielski 		nSize = pMap->size();
494*b1cdbd2cSJim Jagielski 		if( nSize )
495*b1cdbd2cSJim Jagielski 		{
496*b1cdbd2cSJim Jagielski 			typedef OInterfaceContainerHelper* ppp;
497*b1cdbd2cSJim Jagielski 			ppListenerContainers = new ppp[nSize];
498*b1cdbd2cSJim Jagielski 			//ppListenerContainers = new (ListenerContainer*)[nSize];
499*b1cdbd2cSJim Jagielski 
500*b1cdbd2cSJim Jagielski 			t_type2ptr::iterator iter = pMap->begin();
501*b1cdbd2cSJim Jagielski 			t_type2ptr::iterator end = pMap->end();
502*b1cdbd2cSJim Jagielski 
503*b1cdbd2cSJim Jagielski 			t_type2ptr::size_type i = 0;
504*b1cdbd2cSJim Jagielski 			while( iter != end )
505*b1cdbd2cSJim Jagielski 			{
506*b1cdbd2cSJim Jagielski 				ppListenerContainers[i++] = (OInterfaceContainerHelper*)(*iter).second;
507*b1cdbd2cSJim Jagielski 				++iter;
508*b1cdbd2cSJim Jagielski 			}
509*b1cdbd2cSJim Jagielski 		}
510*b1cdbd2cSJim Jagielski 	}
511*b1cdbd2cSJim Jagielski 
512*b1cdbd2cSJim Jagielski 	// create a copy, because do not fire event in a guarded section
513*b1cdbd2cSJim Jagielski 	for( t_type2ptr::size_type i = 0;
514*b1cdbd2cSJim Jagielski 			i < nSize; i++ )
515*b1cdbd2cSJim Jagielski 	{
516*b1cdbd2cSJim Jagielski 		if( ppListenerContainers[i] )
517*b1cdbd2cSJim Jagielski 			ppListenerContainers[i]->disposeAndClear( rEvt );
518*b1cdbd2cSJim Jagielski 	}
519*b1cdbd2cSJim Jagielski 
520*b1cdbd2cSJim Jagielski 	delete [] ppListenerContainers;
521*b1cdbd2cSJim Jagielski }
clear()522*b1cdbd2cSJim Jagielski void OMultiTypeInterfaceContainerHelper::clear()
523*b1cdbd2cSJim Jagielski     SAL_THROW( () )
524*b1cdbd2cSJim Jagielski {
525*b1cdbd2cSJim Jagielski 	::osl::MutexGuard aGuard( rMutex );
526*b1cdbd2cSJim Jagielski     t_type2ptr * pMap = (t_type2ptr *)m_pMap;
527*b1cdbd2cSJim Jagielski 	t_type2ptr::iterator iter = pMap->begin();
528*b1cdbd2cSJim Jagielski 	t_type2ptr::iterator end = pMap->end();
529*b1cdbd2cSJim Jagielski 
530*b1cdbd2cSJim Jagielski 	while( iter != end )
531*b1cdbd2cSJim Jagielski 	{
532*b1cdbd2cSJim Jagielski 		((OInterfaceContainerHelper*)(*iter).second)->clear();
533*b1cdbd2cSJim Jagielski 		++iter;
534*b1cdbd2cSJim Jagielski 	}
535*b1cdbd2cSJim Jagielski }
536*b1cdbd2cSJim Jagielski 
537*b1cdbd2cSJim Jagielski 
538*b1cdbd2cSJim Jagielski //##################################################################################################
539*b1cdbd2cSJim Jagielski //##################################################################################################
540*b1cdbd2cSJim Jagielski //##################################################################################################
541*b1cdbd2cSJim Jagielski 
542*b1cdbd2cSJim Jagielski // specialized class for long
543*b1cdbd2cSJim Jagielski 
544*b1cdbd2cSJim Jagielski typedef ::std::vector< std::pair < sal_Int32 , void* > > t_long2ptr;
545*b1cdbd2cSJim Jagielski 
findLong(t_long2ptr * pMap,sal_Int32 nKey)546*b1cdbd2cSJim Jagielski static t_long2ptr::iterator findLong(t_long2ptr *pMap, sal_Int32 nKey )
547*b1cdbd2cSJim Jagielski {
548*b1cdbd2cSJim Jagielski 	t_long2ptr::iterator iter = pMap->begin();
549*b1cdbd2cSJim Jagielski 	t_long2ptr::iterator end = pMap->end();
550*b1cdbd2cSJim Jagielski 
551*b1cdbd2cSJim Jagielski 	while( iter != end )
552*b1cdbd2cSJim Jagielski     {
553*b1cdbd2cSJim Jagielski         if (iter->first == nKey)
554*b1cdbd2cSJim Jagielski             break;
555*b1cdbd2cSJim Jagielski         iter++;
556*b1cdbd2cSJim Jagielski     }
557*b1cdbd2cSJim Jagielski     return iter;
558*b1cdbd2cSJim Jagielski }
559*b1cdbd2cSJim Jagielski 
OMultiTypeInterfaceContainerHelperInt32(Mutex & rMutex_)560*b1cdbd2cSJim Jagielski OMultiTypeInterfaceContainerHelperInt32::OMultiTypeInterfaceContainerHelperInt32( Mutex & rMutex_ )
561*b1cdbd2cSJim Jagielski     SAL_THROW( () )
562*b1cdbd2cSJim Jagielski     : m_pMap( NULL )
563*b1cdbd2cSJim Jagielski     , rMutex( rMutex_ )
564*b1cdbd2cSJim Jagielski {
565*b1cdbd2cSJim Jagielski     // delay pMap allocation until necessary.
566*b1cdbd2cSJim Jagielski }
~OMultiTypeInterfaceContainerHelperInt32()567*b1cdbd2cSJim Jagielski OMultiTypeInterfaceContainerHelperInt32::~OMultiTypeInterfaceContainerHelperInt32()
568*b1cdbd2cSJim Jagielski     SAL_THROW( () )
569*b1cdbd2cSJim Jagielski {
570*b1cdbd2cSJim Jagielski     if (!m_pMap)
571*b1cdbd2cSJim Jagielski         return;
572*b1cdbd2cSJim Jagielski 
573*b1cdbd2cSJim Jagielski     t_long2ptr * pMap = (t_long2ptr *)m_pMap;
574*b1cdbd2cSJim Jagielski 	t_long2ptr::iterator iter = pMap->begin();
575*b1cdbd2cSJim Jagielski 	t_long2ptr::iterator end = pMap->end();
576*b1cdbd2cSJim Jagielski 
577*b1cdbd2cSJim Jagielski 	while( iter != end )
578*b1cdbd2cSJim Jagielski 	{
579*b1cdbd2cSJim Jagielski 		delete (OInterfaceContainerHelper*)(*iter).second;
580*b1cdbd2cSJim Jagielski 		(*iter).second = 0;
581*b1cdbd2cSJim Jagielski 		++iter;
582*b1cdbd2cSJim Jagielski 	}
583*b1cdbd2cSJim Jagielski 	delete pMap;
584*b1cdbd2cSJim Jagielski }
getContainedTypes() const585*b1cdbd2cSJim Jagielski Sequence< sal_Int32 > OMultiTypeInterfaceContainerHelperInt32::getContainedTypes() const
586*b1cdbd2cSJim Jagielski     SAL_THROW( () )
587*b1cdbd2cSJim Jagielski {
588*b1cdbd2cSJim Jagielski     t_long2ptr * pMap = (t_long2ptr *)m_pMap;
589*b1cdbd2cSJim Jagielski 	t_long2ptr::size_type nSize;
590*b1cdbd2cSJim Jagielski 
591*b1cdbd2cSJim Jagielski 	::osl::MutexGuard aGuard( rMutex );
592*b1cdbd2cSJim Jagielski 	nSize = pMap ? pMap->size() : 0;
593*b1cdbd2cSJim Jagielski 	if( nSize )
594*b1cdbd2cSJim Jagielski 	{
595*b1cdbd2cSJim Jagielski 		::com::sun::star::uno::Sequence< sal_Int32 > aInterfaceTypes( nSize );
596*b1cdbd2cSJim Jagielski 		sal_Int32 * pArray = aInterfaceTypes.getArray();
597*b1cdbd2cSJim Jagielski 
598*b1cdbd2cSJim Jagielski 		t_long2ptr::iterator iter = pMap->begin();
599*b1cdbd2cSJim Jagielski 		t_long2ptr::iterator end = pMap->end();
600*b1cdbd2cSJim Jagielski 
601*b1cdbd2cSJim Jagielski 		sal_Int32 i = 0;
602*b1cdbd2cSJim Jagielski 		while( iter != end )
603*b1cdbd2cSJim Jagielski 		{
604*b1cdbd2cSJim Jagielski 			// are interfaces added to this container?
605*b1cdbd2cSJim Jagielski 			if( ((OInterfaceContainerHelper*)(*iter).second)->getLength() )
606*b1cdbd2cSJim Jagielski 				// yes, put the type in the array
607*b1cdbd2cSJim Jagielski 				pArray[i++] = (*iter).first;
608*b1cdbd2cSJim Jagielski 			++iter;
609*b1cdbd2cSJim Jagielski 		}
610*b1cdbd2cSJim Jagielski 		if( (t_long2ptr::size_type)i != nSize ) {
611*b1cdbd2cSJim Jagielski 			// may be empty container, reduce the sequence to the right size
612*b1cdbd2cSJim Jagielski 			aInterfaceTypes = ::com::sun::star::uno::Sequence< sal_Int32 >( pArray, i );
613*b1cdbd2cSJim Jagielski 		}
614*b1cdbd2cSJim Jagielski 		return aInterfaceTypes;
615*b1cdbd2cSJim Jagielski 	}
616*b1cdbd2cSJim Jagielski 	return ::com::sun::star::uno::Sequence< sal_Int32 >();
617*b1cdbd2cSJim Jagielski }
getContainer(const sal_Int32 & rKey) const618*b1cdbd2cSJim Jagielski OInterfaceContainerHelper * OMultiTypeInterfaceContainerHelperInt32::getContainer( const sal_Int32 & rKey ) const
619*b1cdbd2cSJim Jagielski     SAL_THROW( () )
620*b1cdbd2cSJim Jagielski {
621*b1cdbd2cSJim Jagielski 	::osl::MutexGuard aGuard( rMutex );
622*b1cdbd2cSJim Jagielski 
623*b1cdbd2cSJim Jagielski     if (!m_pMap)
624*b1cdbd2cSJim Jagielski         return 0;
625*b1cdbd2cSJim Jagielski     t_long2ptr * pMap = (t_long2ptr *)m_pMap;
626*b1cdbd2cSJim Jagielski  	t_long2ptr::iterator iter = findLong( pMap, rKey );
627*b1cdbd2cSJim Jagielski 	if( iter != pMap->end() )
628*b1cdbd2cSJim Jagielski 			return (OInterfaceContainerHelper*) (*iter).second;
629*b1cdbd2cSJim Jagielski 	return 0;
630*b1cdbd2cSJim Jagielski }
addInterface(const sal_Int32 & rKey,const Reference<XInterface> & rListener)631*b1cdbd2cSJim Jagielski sal_Int32 OMultiTypeInterfaceContainerHelperInt32::addInterface(
632*b1cdbd2cSJim Jagielski     const sal_Int32 & rKey, const Reference< XInterface > & rListener )
633*b1cdbd2cSJim Jagielski     SAL_THROW( () )
634*b1cdbd2cSJim Jagielski {
635*b1cdbd2cSJim Jagielski 	::osl::MutexGuard aGuard( rMutex );
636*b1cdbd2cSJim Jagielski     if (!m_pMap)
637*b1cdbd2cSJim Jagielski     	m_pMap = new t_long2ptr();
638*b1cdbd2cSJim Jagielski     t_long2ptr * pMap = (t_long2ptr *)m_pMap;
639*b1cdbd2cSJim Jagielski 	t_long2ptr::iterator iter = findLong( pMap, rKey );
640*b1cdbd2cSJim Jagielski  	if( iter == pMap->end() )
641*b1cdbd2cSJim Jagielski 	{
642*b1cdbd2cSJim Jagielski 		OInterfaceContainerHelper * pLC = new OInterfaceContainerHelper( rMutex );
643*b1cdbd2cSJim Jagielski         pMap->push_back(std::pair< sal_Int32, void* >(rKey, pLC));
644*b1cdbd2cSJim Jagielski 		return pLC->addInterface( rListener );
645*b1cdbd2cSJim Jagielski 	}
646*b1cdbd2cSJim Jagielski 	else
647*b1cdbd2cSJim Jagielski 		return ((OInterfaceContainerHelper*)(*iter).second)->addInterface( rListener );
648*b1cdbd2cSJim Jagielski }
removeInterface(const sal_Int32 & rKey,const Reference<XInterface> & rListener)649*b1cdbd2cSJim Jagielski sal_Int32 OMultiTypeInterfaceContainerHelperInt32::removeInterface(
650*b1cdbd2cSJim Jagielski     const sal_Int32 & rKey, const Reference< XInterface > & rListener )
651*b1cdbd2cSJim Jagielski     SAL_THROW( () )
652*b1cdbd2cSJim Jagielski {
653*b1cdbd2cSJim Jagielski 	::osl::MutexGuard aGuard( rMutex );
654*b1cdbd2cSJim Jagielski 
655*b1cdbd2cSJim Jagielski     if (!m_pMap)
656*b1cdbd2cSJim Jagielski         return 0;
657*b1cdbd2cSJim Jagielski 	// search container with id nUik
658*b1cdbd2cSJim Jagielski     t_long2ptr * pMap = (t_long2ptr *)m_pMap;
659*b1cdbd2cSJim Jagielski 	t_long2ptr::iterator iter = findLong( pMap, rKey );
660*b1cdbd2cSJim Jagielski 		// container found?
661*b1cdbd2cSJim Jagielski 	if( iter != pMap->end() )
662*b1cdbd2cSJim Jagielski         return ((OInterfaceContainerHelper*)(*iter).second)->removeInterface( rListener );
663*b1cdbd2cSJim Jagielski 
664*b1cdbd2cSJim Jagielski 	// no container with this id. Always return 0
665*b1cdbd2cSJim Jagielski 	return 0;
666*b1cdbd2cSJim Jagielski }
disposeAndClear(const EventObject & rEvt)667*b1cdbd2cSJim Jagielski void OMultiTypeInterfaceContainerHelperInt32::disposeAndClear( const EventObject & rEvt )
668*b1cdbd2cSJim Jagielski     SAL_THROW( () )
669*b1cdbd2cSJim Jagielski {
670*b1cdbd2cSJim Jagielski 	t_long2ptr::size_type nSize = 0;
671*b1cdbd2cSJim Jagielski 	OInterfaceContainerHelper ** ppListenerContainers = NULL;
672*b1cdbd2cSJim Jagielski 	{
673*b1cdbd2cSJim Jagielski 		::osl::MutexGuard aGuard( rMutex );
674*b1cdbd2cSJim Jagielski         if (!m_pMap)
675*b1cdbd2cSJim Jagielski             return;
676*b1cdbd2cSJim Jagielski 
677*b1cdbd2cSJim Jagielski         t_long2ptr * pMap = (t_long2ptr *)m_pMap;
678*b1cdbd2cSJim Jagielski 		nSize = pMap->size();
679*b1cdbd2cSJim Jagielski 		if( nSize )
680*b1cdbd2cSJim Jagielski 		{
681*b1cdbd2cSJim Jagielski 			typedef OInterfaceContainerHelper* ppp;
682*b1cdbd2cSJim Jagielski 			ppListenerContainers = new ppp[nSize];
683*b1cdbd2cSJim Jagielski 			//ppListenerContainers = new (ListenerContainer*)[nSize];
684*b1cdbd2cSJim Jagielski 
685*b1cdbd2cSJim Jagielski 			t_long2ptr::iterator iter = pMap->begin();
686*b1cdbd2cSJim Jagielski 			t_long2ptr::iterator end = pMap->end();
687*b1cdbd2cSJim Jagielski 
688*b1cdbd2cSJim Jagielski 			t_long2ptr::size_type i = 0;
689*b1cdbd2cSJim Jagielski 			while( iter != end )
690*b1cdbd2cSJim Jagielski 			{
691*b1cdbd2cSJim Jagielski 				ppListenerContainers[i++] = (OInterfaceContainerHelper*)(*iter).second;
692*b1cdbd2cSJim Jagielski 				++iter;
693*b1cdbd2cSJim Jagielski 			}
694*b1cdbd2cSJim Jagielski 		}
695*b1cdbd2cSJim Jagielski 	}
696*b1cdbd2cSJim Jagielski 
697*b1cdbd2cSJim Jagielski 	// create a copy, because do not fire event in a guarded section
698*b1cdbd2cSJim Jagielski 	for( t_long2ptr::size_type i = 0;
699*b1cdbd2cSJim Jagielski 			i < nSize; i++ )
700*b1cdbd2cSJim Jagielski 	{
701*b1cdbd2cSJim Jagielski 		if( ppListenerContainers[i] )
702*b1cdbd2cSJim Jagielski 			ppListenerContainers[i]->disposeAndClear( rEvt );
703*b1cdbd2cSJim Jagielski 	}
704*b1cdbd2cSJim Jagielski 
705*b1cdbd2cSJim Jagielski 	delete [] ppListenerContainers;
706*b1cdbd2cSJim Jagielski }
clear()707*b1cdbd2cSJim Jagielski void OMultiTypeInterfaceContainerHelperInt32::clear()
708*b1cdbd2cSJim Jagielski     SAL_THROW( () )
709*b1cdbd2cSJim Jagielski {
710*b1cdbd2cSJim Jagielski 	::osl::MutexGuard aGuard( rMutex );
711*b1cdbd2cSJim Jagielski     if (!m_pMap)
712*b1cdbd2cSJim Jagielski         return;
713*b1cdbd2cSJim Jagielski     t_long2ptr * pMap = (t_long2ptr *)m_pMap;
714*b1cdbd2cSJim Jagielski 	t_long2ptr::iterator iter = pMap->begin();
715*b1cdbd2cSJim Jagielski 	t_long2ptr::iterator end = pMap->end();
716*b1cdbd2cSJim Jagielski 
717*b1cdbd2cSJim Jagielski 	while( iter != end )
718*b1cdbd2cSJim Jagielski 	{
719*b1cdbd2cSJim Jagielski 		((OInterfaceContainerHelper*)(*iter).second)->clear();
720*b1cdbd2cSJim Jagielski 		++iter;
721*b1cdbd2cSJim Jagielski 	}
722*b1cdbd2cSJim Jagielski }
723*b1cdbd2cSJim Jagielski 
724*b1cdbd2cSJim Jagielski }
725*b1cdbd2cSJim Jagielski 
726