blob: c278474be61de712ba2662f01d3674521fc4042c [file] [log] [blame]
/**************************************************************
*
* 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
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_framework.hxx"
//_________________________________________________________________________________________________________________
// my own includes
//_________________________________________________________________________________________________________________
#include <uielement/itemcontainer.hxx>
#include <uielement/constitemcontainer.hxx>
#include <threadhelp/resetableguard.hxx>
//_________________________________________________________________________________________________________________
// other includes
//_________________________________________________________________________________________________________________
using namespace cppu;
using namespace com::sun::star::uno;
using namespace com::sun::star::lang;
using namespace com::sun::star::beans;
using namespace com::sun::star::container;
const char WRONG_TYPE_EXCEPTION[] = "Type must be com::sun::star::uno::Sequence< com::sun::star::beans::PropertyValue >";
namespace framework
{
//*****************************************************************************************************************
// XInterface, XTypeProvider
//*****************************************************************************************************************
ItemContainer::ItemContainer( const ShareableMutex& rMutex ) :
m_aShareMutex( rMutex )
{
}
ItemContainer::ItemContainer( const ConstItemContainer& rConstItemContainer, const ShareableMutex& rMutex ) : m_aShareMutex( rMutex )
{
copyItemContainer( rConstItemContainer.m_aItemVector, rMutex );
}
ItemContainer::ItemContainer( const Reference< XIndexAccess >& rSourceContainer, const ShareableMutex& rMutex ) :
m_aShareMutex( rMutex )
{
if ( rSourceContainer.is() )
{
sal_Int32 nCount = rSourceContainer->getCount();
try
{
for ( sal_Int32 i = 0; i < nCount; i++ )
{
Sequence< PropertyValue > aPropSeq;
if ( rSourceContainer->getByIndex( i ) >>= aPropSeq )
{
sal_Int32 nContainerIndex = -1;
Reference< XIndexAccess > xIndexAccess;
for ( sal_Int32 j = 0; j < aPropSeq.getLength(); j++ )
{
if ( aPropSeq[j].Name.equalsAscii( "ItemDescriptorContainer" ))
{
aPropSeq[j].Value >>= xIndexAccess;
nContainerIndex = j;
break;
}
}
if ( xIndexAccess.is() && nContainerIndex >= 0 )
aPropSeq[nContainerIndex].Value <<= deepCopyContainer( xIndexAccess, rMutex );
m_aItemVector.push_back( aPropSeq );
}
}
}
catch ( IndexOutOfBoundsException& )
{
}
}
}
ItemContainer::~ItemContainer()
{
}
// private
void ItemContainer::copyItemContainer( const std::vector< Sequence< PropertyValue > >& rSourceVector, const ShareableMutex& rMutex )
{
const sal_uInt32 nCount = rSourceVector.size();
for ( sal_uInt32 i = 0; i < nCount; ++i )
{
sal_Int32 nContainerIndex = -1;
Sequence< PropertyValue > aPropSeq( rSourceVector[i] );
Reference< XIndexAccess > xIndexAccess;
for ( sal_Int32 j = 0; j < aPropSeq.getLength(); j++ )
{
if ( aPropSeq[j].Name.equalsAscii( "ItemDescriptorContainer" ))
{
aPropSeq[j].Value >>= xIndexAccess;
nContainerIndex = j;
break;
}
}
if ( xIndexAccess.is() && nContainerIndex >= 0 )
aPropSeq[nContainerIndex].Value <<= deepCopyContainer( xIndexAccess, rMutex );
m_aItemVector.push_back( aPropSeq );
}
}
Reference< XIndexAccess > ItemContainer::deepCopyContainer( const Reference< XIndexAccess >& rSubContainer, const ShareableMutex& rMutex )
{
Reference< XIndexAccess > xReturn;
if ( rSubContainer.is() )
{
ConstItemContainer* pSource = ConstItemContainer::GetImplementation( rSubContainer );
ItemContainer* pSubContainer( 0 );
if ( pSource )
pSubContainer = new ItemContainer( *pSource, rMutex );
else
pSubContainer = new ItemContainer( rSubContainer, rMutex );
xReturn = Reference< XIndexAccess >( static_cast< OWeakObject* >( pSubContainer ), UNO_QUERY );
}
return xReturn;
}
// XUnoTunnel
sal_Int64 ItemContainer::getSomething( const ::com::sun::star::uno::Sequence< sal_Int8 >& rIdentifier ) throw(::com::sun::star::uno::RuntimeException)
{
if( ( rIdentifier.getLength() == 16 ) && ( 0 == rtl_compareMemory( ItemContainer::GetUnoTunnelId().getConstArray(), rIdentifier.getConstArray(), 16 ) ) )
return sal::static_int_cast< sal_Int64 >( reinterpret_cast< sal_IntPtr >( this ));
return 0;
}
const Sequence< sal_Int8 >& ItemContainer::GetUnoTunnelId() throw()
{
static ::com::sun::star::uno::Sequence< sal_Int8 > * pSeq = NULL;
if( !pSeq )
{
::osl::Guard< ::osl::Mutex > aGuard( ::osl::Mutex::getGlobalMutex() );
if( !pSeq )
{
static ::com::sun::star::uno::Sequence< sal_Int8 > aSeq( 16 );
rtl_createUuid( (sal_uInt8*)aSeq.getArray(), 0, sal_True );
pSeq = &aSeq;
}
}
return *pSeq;
}
ItemContainer* ItemContainer::GetImplementation( const ::com::sun::star::uno::Reference< ::com::sun::star::uno::XInterface >& rxIFace ) throw()
{
::com::sun::star::uno::Reference< ::com::sun::star::lang::XUnoTunnel > xUT( rxIFace, ::com::sun::star::uno::UNO_QUERY );
return xUT.is() ? reinterpret_cast< ItemContainer* >(sal::static_int_cast< sal_IntPtr >(
xUT->getSomething( ItemContainer::GetUnoTunnelId() ))) : NULL;
}
// XElementAccess
sal_Bool SAL_CALL ItemContainer::hasElements()
throw ( RuntimeException )
{
ShareGuard aLock( m_aShareMutex );
return ( !m_aItemVector.empty() );
}
// XIndexAccess
sal_Int32 SAL_CALL ItemContainer::getCount()
throw ( RuntimeException )
{
ShareGuard aLock( m_aShareMutex );
return m_aItemVector.size();
}
Any SAL_CALL ItemContainer::getByIndex( sal_Int32 Index )
throw ( IndexOutOfBoundsException, WrappedTargetException, RuntimeException )
{
ShareGuard aLock( m_aShareMutex );
if ( sal_Int32( m_aItemVector.size()) > Index )
return makeAny( m_aItemVector[Index] );
else
throw IndexOutOfBoundsException( ::rtl::OUString(), (OWeakObject *)this );
}
// XIndexContainer
void SAL_CALL ItemContainer::insertByIndex( sal_Int32 Index, const Any& aItem )
throw ( IllegalArgumentException, IndexOutOfBoundsException, WrappedTargetException, RuntimeException )
{
Sequence< PropertyValue > aSeq;
if ( aItem >>= aSeq )
{
ShareGuard aLock( m_aShareMutex );
if ( sal_Int32( m_aItemVector.size()) == Index )
m_aItemVector.push_back( aSeq );
else if ( sal_Int32( m_aItemVector.size()) >Index )
{
std::vector< Sequence< PropertyValue > >::iterator aIter = m_aItemVector.begin();
aIter += Index;
m_aItemVector.insert( aIter, aSeq );
}
else
throw IndexOutOfBoundsException( ::rtl::OUString(), (OWeakObject *)this );
}
else
throw IllegalArgumentException( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( WRONG_TYPE_EXCEPTION )),
(OWeakObject *)this, 2 );
}
void SAL_CALL ItemContainer::removeByIndex( sal_Int32 Index )
throw ( IndexOutOfBoundsException, WrappedTargetException, RuntimeException )
{
ShareGuard aLock( m_aShareMutex );
if ( (sal_Int32)m_aItemVector.size() > Index )
{
std::vector< Sequence< PropertyValue > >::iterator aIter = m_aItemVector.begin();
aIter += Index;
m_aItemVector.erase( aIter );
}
else
throw IndexOutOfBoundsException( ::rtl::OUString(), (OWeakObject *)this );
}
void SAL_CALL ItemContainer::replaceByIndex( sal_Int32 Index, const Any& aItem )
throw ( IllegalArgumentException, IndexOutOfBoundsException, WrappedTargetException, RuntimeException )
{
Sequence< PropertyValue > aSeq;
if ( aItem >>= aSeq )
{
ShareGuard aLock( m_aShareMutex );
if ( sal_Int32( m_aItemVector.size()) > Index )
m_aItemVector[Index] = aSeq;
else
throw IndexOutOfBoundsException( ::rtl::OUString(), (OWeakObject *)this );
}
else
throw IllegalArgumentException( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( WRONG_TYPE_EXCEPTION )),
(OWeakObject *)this, 2 );
}
} // namespace framework