Files
libreoffice/stoc/source/corereflection/crcomp.cxx

387 lines
14 KiB
C++
Raw Normal View History

/* -*- 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 .
*/
2000-09-18 14:29:57 +00:00
#include <rtl/strbuf.hxx>
#include <com/sun/star/reflection/XIdlField.hpp>
2001-05-03 12:57:45 +00:00
#include <com/sun/star/reflection/XIdlField2.hpp>
#include "com/sun/star/uno/TypeClass.hpp"
2000-09-18 14:29:57 +00:00
#include "base.hxx"
namespace stoc_corefl
{
2000-09-18 14:29:57 +00:00
class IdlCompFieldImpl
: public IdlMemberImpl
, public XIdlField
2001-05-03 12:57:45 +00:00
, public XIdlField2
2000-09-18 14:29:57 +00:00
{
sal_Int32 _nOffset;
public:
IdlCompFieldImpl( IdlReflectionServiceImpl * pReflection, const OUString & rName,
typelib_TypeDescription * pTypeDescr, typelib_TypeDescription * pDeclTypeDescr,
sal_Int32 nOffset )
: IdlMemberImpl( pReflection, rName, pTypeDescr, pDeclTypeDescr )
, _nOffset( nOffset )
{}
// XInterface
virtual Any SAL_CALL queryInterface( const Type & rType ) throw (::com::sun::star::uno::RuntimeException, std::exception);
2000-09-18 14:29:57 +00:00
virtual void SAL_CALL acquire() throw ();
virtual void SAL_CALL release() throw ();
// XTypeProvider
virtual Sequence< Type > SAL_CALL getTypes() throw (::com::sun::star::uno::RuntimeException, std::exception);
virtual Sequence< sal_Int8 > SAL_CALL getImplementationId() throw (::com::sun::star::uno::RuntimeException, std::exception);
2000-09-18 14:29:57 +00:00
// XIdlMember
virtual Reference< XIdlClass > SAL_CALL getDeclaringClass() throw(::com::sun::star::uno::RuntimeException, std::exception);
virtual OUString SAL_CALL getName() throw(::com::sun::star::uno::RuntimeException, std::exception);
2000-09-18 14:29:57 +00:00
// XIdlField
virtual Reference< XIdlClass > SAL_CALL getType() throw(::com::sun::star::uno::RuntimeException, std::exception);
virtual FieldAccessMode SAL_CALL getAccessMode() throw(::com::sun::star::uno::RuntimeException, std::exception);
virtual Any SAL_CALL get( const Any & rObj ) throw(::com::sun::star::lang::IllegalArgumentException, ::com::sun::star::uno::RuntimeException, std::exception);
virtual void SAL_CALL set( const Any & rObj, const Any & rValue ) throw(::com::sun::star::lang::IllegalArgumentException, ::com::sun::star::lang::IllegalAccessException, ::com::sun::star::uno::RuntimeException, std::exception);
2001-05-03 12:57:45 +00:00
// XIdlField2: getType, getAccessMode and get are equal to XIdlField
virtual void SAL_CALL set( Any & rObj, const Any & rValue ) throw(::com::sun::star::lang::IllegalArgumentException, ::com::sun::star::lang::IllegalAccessException, ::com::sun::star::uno::RuntimeException, std::exception);
2000-09-18 14:29:57 +00:00
};
// XInterface
2000-09-18 14:29:57 +00:00
Any IdlCompFieldImpl::queryInterface( const Type & rType )
throw(::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
2001-05-03 12:57:45 +00:00
Any aRet( ::cppu::queryInterface( rType,
static_cast< XIdlField * >( this ),
static_cast< XIdlField2 * >( this ) ) );
2000-09-18 14:29:57 +00:00
return (aRet.hasValue() ? aRet : IdlMemberImpl::queryInterface( rType ));
}
2000-09-18 14:29:57 +00:00
void IdlCompFieldImpl::acquire() throw()
{
IdlMemberImpl::acquire();
}
2000-09-18 14:29:57 +00:00
void IdlCompFieldImpl::release() throw()
{
IdlMemberImpl::release();
}
// XTypeProvider
2000-09-18 14:29:57 +00:00
Sequence< Type > IdlCompFieldImpl::getTypes()
throw (::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
static OTypeCollection * s_pTypes = 0;
if (! s_pTypes)
{
MutexGuard aGuard( getMutexAccess() );
if (! s_pTypes)
{
static OTypeCollection s_aTypes(
2001-05-03 12:57:45 +00:00
::getCppuType( (const Reference< XIdlField2 > *)0 ),
2000-09-18 14:29:57 +00:00
::getCppuType( (const Reference< XIdlField > *)0 ),
IdlMemberImpl::getTypes() );
s_pTypes = &s_aTypes;
}
}
return s_pTypes->getTypes();
}
2000-09-18 14:29:57 +00:00
Sequence< sal_Int8 > IdlCompFieldImpl::getImplementationId()
throw (::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
return css::uno::Sequence<sal_Int8>();
2000-09-18 14:29:57 +00:00
}
// XIdlMember
2000-09-18 14:29:57 +00:00
Reference< XIdlClass > IdlCompFieldImpl::getDeclaringClass()
throw(::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
if (! _xDeclClass.is())
{
MutexGuard aGuard( getMutexAccess() );
if (! _xDeclClass.is())
{
typelib_CompoundTypeDescription * pTD =
(typelib_CompoundTypeDescription *)getDeclTypeDescr();
while (pTD)
{
typelib_TypeDescriptionReference ** ppTypeRefs = pTD->ppTypeRefs;
for ( sal_Int32 nPos = pTD->nMembers; nPos--; )
{
if (td_equals( (typelib_TypeDescription *)getTypeDescr(), ppTypeRefs[nPos] ))
{
_xDeclClass = getReflection()->forType( (typelib_TypeDescription *)pTD );
return _xDeclClass;
}
}
pTD = pTD->pBaseTypeDescription;
}
}
}
return _xDeclClass;
}
2000-09-18 14:29:57 +00:00
OUString IdlCompFieldImpl::getName()
throw(::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
return IdlMemberImpl::getName();
}
// XIdlField
2000-09-18 14:29:57 +00:00
Reference< XIdlClass > IdlCompFieldImpl::getType()
throw(::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
return getReflection()->forType( getTypeDescr() );
}
2000-09-18 14:29:57 +00:00
FieldAccessMode IdlCompFieldImpl::getAccessMode()
throw(::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
return FieldAccessMode_READWRITE;
}
2000-09-18 14:29:57 +00:00
Any IdlCompFieldImpl::get( const Any & rObj )
throw(::com::sun::star::lang::IllegalArgumentException, ::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
if (rObj.getValueTypeClass() == com::sun::star::uno::TypeClass_STRUCT ||
rObj.getValueTypeClass() == com::sun::star::uno::TypeClass_EXCEPTION)
2000-09-18 14:29:57 +00:00
{
typelib_TypeDescription * pObjTD = 0;
TYPELIB_DANGER_GET( &pObjTD, rObj.getValueTypeRef() );
typelib_TypeDescription * pTD = pObjTD;
typelib_TypeDescription * pDeclTD = getDeclTypeDescr();
while (pTD && !typelib_typedescription_equals( pTD, pDeclTD ))
pTD = (typelib_TypeDescription *)((typelib_CompoundTypeDescription *)pTD)->pBaseTypeDescription;
2001-03-12 14:38:06 +00:00
OSL_ENSURE( pTD, "### illegal object type!" );
2000-09-18 14:29:57 +00:00
if (pTD)
{
TYPELIB_DANGER_RELEASE( pObjTD );
Any aRet;
uno_any_destruct(
&aRet, reinterpret_cast< uno_ReleaseFunc >(cpp_release) );
uno_any_construct(
&aRet, (char *)rObj.getValue() + _nOffset, getTypeDescr(),
reinterpret_cast< uno_AcquireFunc >(cpp_acquire) );
2000-09-18 14:29:57 +00:00
return aRet;
}
TYPELIB_DANGER_RELEASE( pObjTD );
}
throw IllegalArgumentException(
OUString("illegal object given!"),
2000-09-18 14:29:57 +00:00
(XWeak *)(OWeakObject *)this, 0 );
}
2000-09-18 14:29:57 +00:00
void IdlCompFieldImpl::set( const Any & rObj, const Any & rValue )
throw(::com::sun::star::lang::IllegalArgumentException, ::com::sun::star::lang::IllegalAccessException, ::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
if (rObj.getValueTypeClass() == com::sun::star::uno::TypeClass_STRUCT ||
rObj.getValueTypeClass() == com::sun::star::uno::TypeClass_EXCEPTION)
2000-09-18 14:29:57 +00:00
{
typelib_TypeDescription * pObjTD = 0;
TYPELIB_DANGER_GET( &pObjTD, rObj.getValueTypeRef() );
typelib_TypeDescription * pTD = pObjTD;
typelib_TypeDescription * pDeclTD = getDeclTypeDescr();
while (pTD && !typelib_typedescription_equals( pTD, pDeclTD ))
pTD = (typelib_TypeDescription *)((typelib_CompoundTypeDescription *)pTD)->pBaseTypeDescription;
2001-03-12 14:38:06 +00:00
OSL_ENSURE( pTD, "### illegal object type!" );
2000-09-18 14:29:57 +00:00
if (pTD)
{
TYPELIB_DANGER_RELEASE( pObjTD );
if (coerce_assign( (char *)rObj.getValue() + _nOffset, getTypeDescr(), rValue, getReflection() ))
{
return;
}
else
{
throw IllegalArgumentException(
OUString("illegal value given!"),
2000-09-18 14:29:57 +00:00
(XWeak *)(OWeakObject *)this, 1 );
}
}
TYPELIB_DANGER_RELEASE( pObjTD );
}
throw IllegalArgumentException(
OUString("illegal object given!"),
2000-09-18 14:29:57 +00:00
(XWeak *)(OWeakObject *)this, 0 );
}
2001-05-03 12:57:45 +00:00
void IdlCompFieldImpl::set( Any & rObj, const Any & rValue )
throw(::com::sun::star::lang::IllegalArgumentException, ::com::sun::star::lang::IllegalAccessException, ::com::sun::star::uno::RuntimeException, std::exception)
2001-05-03 12:57:45 +00:00
{
if (rObj.getValueTypeClass() == com::sun::star::uno::TypeClass_STRUCT ||
rObj.getValueTypeClass() == com::sun::star::uno::TypeClass_EXCEPTION)
2001-05-03 12:57:45 +00:00
{
typelib_TypeDescription * pObjTD = 0;
TYPELIB_DANGER_GET( &pObjTD, rObj.getValueTypeRef() );
typelib_TypeDescription * pTD = pObjTD;
typelib_TypeDescription * pDeclTD = getDeclTypeDescr();
while (pTD && !typelib_typedescription_equals( pTD, pDeclTD ))
pTD = (typelib_TypeDescription *)((typelib_CompoundTypeDescription *)pTD)->pBaseTypeDescription;
OSL_ENSURE( pTD, "### illegal object type!" );
if (pTD)
{
TYPELIB_DANGER_RELEASE( pObjTD );
if (coerce_assign( (char *)rObj.getValue() + _nOffset, getTypeDescr(), rValue, getReflection() ))
{
return;
}
else
{
throw IllegalArgumentException(
OUString("illegal value given!"),
2001-05-03 12:57:45 +00:00
(XWeak *)(OWeakObject *)this, 1 );
}
}
TYPELIB_DANGER_RELEASE( pObjTD );
}
throw IllegalArgumentException(
OUString("illegal object given!"),
2001-05-03 12:57:45 +00:00
(XWeak *)(OWeakObject *)this, 0 );
}
2000-09-18 14:29:57 +00:00
2000-09-18 14:29:57 +00:00
2000-09-18 14:29:57 +00:00
CompoundIdlClassImpl::~CompoundIdlClassImpl()
{
delete _pFields;
}
2000-09-18 14:29:57 +00:00
sal_Bool CompoundIdlClassImpl::isAssignableFrom( const Reference< XIdlClass > & xType )
throw(::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
if (xType.is())
{
TypeClass eTC = xType->getTypeClass();
if (eTC == TypeClass_STRUCT || eTC == TypeClass_EXCEPTION)
{
if (equals( xType ))
return sal_True;
else
{
const Sequence< Reference< XIdlClass > > & rSeq = xType->getSuperclasses();
if (rSeq.getLength())
{
2001-03-12 14:38:06 +00:00
OSL_ENSURE( rSeq.getLength() == 1, "### unexpected len of super classes!" );
2000-09-18 14:29:57 +00:00
return isAssignableFrom( rSeq[0] );
}
}
}
}
return sal_False;
}
2000-09-18 14:29:57 +00:00
Sequence< Reference< XIdlClass > > CompoundIdlClassImpl::getSuperclasses()
throw(::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
if (! _xSuperClass.is())
{
MutexGuard aGuard( getMutexAccess() );
if (! _xSuperClass.is())
{
typelib_CompoundTypeDescription * pCompTypeDescr = getTypeDescr()->pBaseTypeDescription;
if (pCompTypeDescr)
_xSuperClass = getReflection()->forType( (typelib_TypeDescription *)pCompTypeDescr );
}
}
if (_xSuperClass.is())
return Sequence< Reference< XIdlClass > >( &_xSuperClass, 1 );
else
return Sequence< Reference< XIdlClass > >();
}
2000-09-18 14:29:57 +00:00
Reference< XIdlField > CompoundIdlClassImpl::getField( const OUString & rName )
throw(::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
if (! _pFields)
getFields(); // init fields
const OUString2Field::const_iterator iFind( _aName2Field.find( rName ) );
if (iFind != _aName2Field.end())
return Reference< XIdlField >( (*iFind).second );
else
return Reference< XIdlField >();
}
2000-09-18 14:29:57 +00:00
Sequence< Reference< XIdlField > > CompoundIdlClassImpl::getFields()
throw(::com::sun::star::uno::RuntimeException, std::exception)
2000-09-18 14:29:57 +00:00
{
MutexGuard aGuard( getMutexAccess() );
if (! _pFields)
{
sal_Int32 nAll = 0;
typelib_CompoundTypeDescription * pCompTypeDescr = getTypeDescr();
for ( ; pCompTypeDescr; pCompTypeDescr = pCompTypeDescr->pBaseTypeDescription )
nAll += pCompTypeDescr->nMembers;
Sequence< Reference< XIdlField > > * pFields =
new Sequence< Reference< XIdlField > >( nAll );
Reference< XIdlField > * pSeq = pFields->getArray();
for ( pCompTypeDescr = getTypeDescr(); pCompTypeDescr;
pCompTypeDescr = pCompTypeDescr->pBaseTypeDescription )
{
typelib_TypeDescriptionReference ** ppTypeRefs = pCompTypeDescr->ppTypeRefs;
rtl_uString ** ppNames = pCompTypeDescr->ppMemberNames;
sal_Int32 * pMemberOffsets = pCompTypeDescr->pMemberOffsets;
for ( sal_Int32 nPos = pCompTypeDescr->nMembers; nPos--; )
{
typelib_TypeDescription * pTD = 0;
TYPELIB_DANGER_GET( &pTD, ppTypeRefs[nPos] );
2001-03-12 14:38:06 +00:00
OSL_ENSURE( pTD, "### cannot get field in struct!" );
2000-09-18 14:29:57 +00:00
if (pTD)
{
OUString aName( ppNames[nPos] );
_aName2Field[aName] = pSeq[--nAll] = new IdlCompFieldImpl(
getReflection(), aName, pTD, IdlClassImpl::getTypeDescr(), pMemberOffsets[nPos] );
TYPELIB_DANGER_RELEASE( pTD );
}
}
}
_pFields = pFields;
}
return *_pFields;
}
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */