Files
libreoffice/ucb/source/core/ucb.cxx

580 lines
19 KiB
C++
Raw Normal View History

2000-10-16 13:56:13 +00:00
/*************************************************************************
*
* OpenOffice.org - a multi-platform office productivity suite
2000-10-16 13:56:13 +00:00
*
* $RCSfile: ucb.cxx,v $
2000-10-16 13:56:13 +00:00
*
* $Revision: 1.10 $
2000-10-16 13:56:13 +00:00
*
* last change: $Author: obo $ $Date: 2006-09-17 13:43:48 $
2000-10-16 13:56:13 +00:00
*
* The Contents of this file are made available subject to
* the terms of GNU Lesser General Public License Version 2.1.
2000-10-16 13:56:13 +00:00
*
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2005 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
2000-10-16 13:56:13 +00:00
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1, as published by the Free Software Foundation.
2000-10-16 13:56:13 +00:00
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
2000-10-16 13:56:13 +00:00
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
2000-10-16 13:56:13 +00:00
*
************************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_ucb.hxx"
2000-10-16 13:56:13 +00:00
/**************************************************************************
TODO
**************************************************************************
*************************************************************************/
2001-06-25 07:51:54 +00:00
#ifndef _OSL_DIAGNOSE_H_
#include <osl/diagnose.h>
#endif
2000-10-16 13:56:13 +00:00
#ifndef _CPPUHELPER_INTERFACECONTAINER_HXX_
#include <cppuhelper/interfacecontainer.hxx>
#endif
#ifndef _COM_SUN_STAR_LANG_ILLEGALARGUMENTEXCEPTION_HPP_
#include <com/sun/star/lang/IllegalArgumentException.hpp>
#endif
#ifndef _COM_SUN_STAR_UCB_GLOBALTRANSFERCOMMANDARGUMENT_HPP_
#include <com/sun/star/ucb/GlobalTransferCommandArgument.hpp>
#endif
#ifndef _COM_SUN_STAR_UCB_XCOMMANDINFO_HPP_
#include <com/sun/star/ucb/XCommandInfo.hpp>
#endif
2000-10-16 13:56:13 +00:00
#ifndef _COM_SUN_STAR_UCB_XCONTENTPROVIDER_HPP_
#include <com/sun/star/ucb/XContentProvider.hpp>
#endif
#ifndef _COM_SUN_STAR_UCB_XCONTENTPROVIDERSUPPLIER_HPP_
#include <com/sun/star/ucb/XContentProviderSupplier.hpp>
#endif
#ifndef _UCBHELPER_CONFIGUREUCB_HXX_
#include <ucbhelper/configureucb.hxx>
#endif
2001-06-25 07:51:54 +00:00
#ifndef _UCBHELPER_CANCELCOMMANDEXECUTION_HXX_
#include <ucbhelper/cancelcommandexecution.hxx>
#endif
2000-10-16 13:56:13 +00:00
#ifndef _IDENTIFY_HXX
#include "identify.hxx"
#endif
#ifndef _UCBCMDS_HXX
#include "ucbcmds.hxx"
#endif
2000-10-16 13:56:13 +00:00
#include "ucb.hxx"
// Definitions for ProviderMap_Impl (Solaris wouldn't find explicit template
// instantiations for these in another compilation unit...):
#ifndef _UCB_REGEXPMAP_TPT_
#include <regexpmap.tpt>
#endif
using namespace rtl;
using namespace cppu;
using namespace com::sun::star::uno;
using namespace com::sun::star::lang;
using namespace com::sun::star::ucb;
//=========================================================================
//
// UniversalContentBroker Implementation.
//
//=========================================================================
UniversalContentBroker::UniversalContentBroker(
const Reference< com::sun::star::lang::XMultiServiceFactory >& rXSMgr )
: m_xSMgr( rXSMgr ),
m_pDisposeEventListeners( NULL ),
m_nInitCount( 0 ), //@@@ see initialize() method
m_nCommandId( 0 )
2000-10-16 13:56:13 +00:00
{
2001-06-25 07:51:54 +00:00
OSL_ENSURE( m_xSMgr.is(),
"UniversalContentBroker ctor: No service manager" );
2000-10-16 13:56:13 +00:00
}
//=========================================================================
// virtual
UniversalContentBroker::~UniversalContentBroker()
{
delete m_pDisposeEventListeners;
}
//=========================================================================
//
// XInterface methods.
//
//=========================================================================
XINTERFACE_IMPL_8( UniversalContentBroker,
XTypeProvider,
XComponent,
XServiceInfo,
XInitialization,
XContentProviderManager,
XContentProvider,
XContentIdentifierFactory,
XCommandProcessor );
2000-10-16 13:56:13 +00:00
//=========================================================================
//
// XTypeProvider methods.
//
//=========================================================================
XTYPEPROVIDER_IMPL_8( UniversalContentBroker,
XTypeProvider,
XComponent,
XServiceInfo,
XInitialization,
XContentProviderManager,
XContentProvider,
XContentIdentifierFactory,
XCommandProcessor );
2000-10-16 13:56:13 +00:00
//=========================================================================
//
// XComponent methods.
//
//=========================================================================
// virtual
void SAL_CALL UniversalContentBroker::dispose()
throw( com::sun::star::uno::RuntimeException )
{
if ( m_pDisposeEventListeners && m_pDisposeEventListeners->getLength() )
{
EventObject aEvt;
aEvt.Source = SAL_STATIC_CAST( XComponent*, this );
m_pDisposeEventListeners->disposeAndClear( aEvt );
}
}
//=========================================================================
// virtual
void SAL_CALL UniversalContentBroker::addEventListener(
const Reference< XEventListener >& Listener )
throw( com::sun::star::uno::RuntimeException )
{
if ( !m_pDisposeEventListeners )
m_pDisposeEventListeners = new OInterfaceContainerHelper( m_aMutex );
m_pDisposeEventListeners->addInterface( Listener );
}
//=========================================================================
// virtual
void SAL_CALL UniversalContentBroker::removeEventListener(
const Reference< XEventListener >& Listener )
throw( com::sun::star::uno::RuntimeException )
{
if ( m_pDisposeEventListeners )
m_pDisposeEventListeners->removeInterface( Listener );
// Note: Don't want to delete empty container here -> performance.
}
//=========================================================================
//
// XServiceInfo methods.
//
//=========================================================================
XSERVICEINFO_IMPL_1( UniversalContentBroker,
OUString::createFromAscii(
"com.sun.star.comp.ucb.UniversalContentBroker" ),
OUString::createFromAscii(
UCB_SERVICE_NAME ) );
2000-10-16 13:56:13 +00:00
//=========================================================================
//
// Service factory implementation.
//
//=========================================================================
ONE_INSTANCE_SERVICE_FACTORY_IMPL( UniversalContentBroker );
//=========================================================================
//
// XInitialization methods.
//
//=========================================================================
// virtual
void SAL_CALL UniversalContentBroker::initialize(
const com::sun::star::uno::Sequence< Any >& aArguments )
throw( com::sun::star::uno::Exception,
com::sun::star::uno::RuntimeException )
{
//@@@ At the moment, there's a problem when one (non-one-instance) factory
// 'wraps' another (one-instance) factory, causing this method to be
// called several times:
oslInterlockedCount nCount = osl_incrementInterlockedCount(&m_nInitCount);
if (nCount == 1)
::ucb::configureUcb(this, m_xSMgr, aArguments, 0);
else
osl_decrementInterlockedCount(&m_nInitCount);
2000-10-16 13:56:13 +00:00
// make the possibility of overflow less likely...
}
//=========================================================================
//
// XContentProviderManager methods.
//
//=========================================================================
// virtual
Reference< XContentProvider > SAL_CALL
UniversalContentBroker::registerContentProvider(
const Reference< XContentProvider >& Provider,
const OUString& Scheme,
sal_Bool ReplaceExisting )
throw( DuplicateProviderException, com::sun::star::uno::RuntimeException )
{
osl::MutexGuard aGuard(m_aMutex);
ProviderMap_Impl::iterator aIt;
try
{
aIt = m_aProviders.find(Scheme);
}
catch (IllegalArgumentException const &)
{
return 0; //@@@
}
Reference< XContentProvider > xPrevious;
if (aIt == m_aProviders.end())
{
ProviderList_Impl aList;
aList.push_front(Provider);
try
{
m_aProviders.add(Scheme, aList, false);
}
catch (IllegalArgumentException const &)
{
return 0; //@@@
}
}
else
{
if (!ReplaceExisting)
throw DuplicateProviderException();
ProviderList_Impl & rList = aIt->getValue();
xPrevious = rList.front().getProvider();
rList.push_front(Provider);
}
return xPrevious;
}
//=========================================================================
// virtual
void SAL_CALL UniversalContentBroker::deregisterContentProvider(
const Reference< XContentProvider >& Provider,
const OUString& Scheme )
throw( com::sun::star::uno::RuntimeException )
{
osl::MutexGuard aGuard(m_aMutex);
ProviderMap_Impl::iterator aMapIt;
try
{
aMapIt = m_aProviders.find(Scheme);
}
catch (IllegalArgumentException const &)
{
return; //@@@
}
2000-10-16 13:56:13 +00:00
if (aMapIt != m_aProviders.end())
{
ProviderList_Impl & rList = aMapIt->getValue();
ProviderList_Impl::iterator aListEnd(rList.end());
for (ProviderList_Impl::iterator aListIt(rList.begin());
aListIt != aListEnd; ++aListIt)
{
if ((*aListIt).getProvider() == Provider)
{
rList.erase(aListIt);
break;
}
}
if (rList.empty())
m_aProviders.erase(aMapIt);
}
}
//=========================================================================
// virtual
com::sun::star::uno::Sequence< ContentProviderInfo > SAL_CALL
UniversalContentBroker::queryContentProviders()
throw( com::sun::star::uno::RuntimeException )
{
// Return a list with information about active(!) content providers.
osl::MutexGuard aGuard(m_aMutex);
com::sun::star::uno::Sequence< ContentProviderInfo > aSeq(
m_aProviders.size() );
ContentProviderInfo* pInfo = aSeq.getArray();
ProviderMap_Impl::const_iterator end = m_aProviders.end();
for (ProviderMap_Impl::const_iterator it(m_aProviders.begin()); it != end;
++it)
{
// Note: Active provider is always the first list element.
pInfo->ContentProvider = it->getValue().front().getProvider();
pInfo->Scheme = it->getRegexp();
++pInfo;
}
return aSeq;
}
//=========================================================================
// virtual
Reference< XContentProvider > SAL_CALL
UniversalContentBroker::queryContentProvider( const OUString&
Identifier )
2000-10-16 13:56:13 +00:00
throw( com::sun::star::uno::RuntimeException )
{
return queryContentProvider( Identifier, sal_False );
2000-10-16 13:56:13 +00:00
}
//=========================================================================
//
// XContentProvider methods.
//
//=========================================================================
// virtual
Reference< XContent > SAL_CALL UniversalContentBroker::queryContent(
const Reference< XContentIdentifier >& Identifier )
throw( IllegalIdentifierException, com::sun::star::uno::RuntimeException )
{
//////////////////////////////////////////////////////////////////////
// Let the content provider for the scheme given with the content
// identifier create the XContent instance.
//////////////////////////////////////////////////////////////////////
if ( !Identifier.is() )
return Reference< XContent >();
Reference< XContentProvider > xProv =
queryContentProvider( Identifier->getContentIdentifier(), sal_True );
if ( xProv.is() )
return xProv->queryContent( Identifier );
return Reference< XContent >();
}
//=========================================================================
// virtual
sal_Int32 SAL_CALL UniversalContentBroker::compareContentIds(
const Reference< XContentIdentifier >& Id1,
const Reference< XContentIdentifier >& Id2 )
throw( com::sun::star::uno::RuntimeException )
{
OUString aURI1( Id1->getContentIdentifier() );
OUString aURI2( Id2->getContentIdentifier() );
Reference< XContentProvider > xProv1
= queryContentProvider( aURI1, sal_True );
Reference< XContentProvider > xProv2
= queryContentProvider( aURI2, sal_True );
// When both identifiers belong to the same provider, let that provider
// compare them; otherwise, simply compare the URI strings (which must
// be different):
if ( xProv1.is() && ( xProv1 == xProv2 ) )
return xProv1->compareContentIds( Id1, Id2 );
else
return aURI1.compareTo( aURI2 );
}
//=========================================================================
//
// XContentIdentifierFactory methods.
//
//=========================================================================
// virtual
Reference< XContentIdentifier > SAL_CALL
UniversalContentBroker::createContentIdentifier(
const OUString& ContentId )
throw( com::sun::star::uno::RuntimeException )
{
//////////////////////////////////////////////////////////////////////
// Let the content provider for the scheme given with content
// identifier create the XContentIdentifier instance, if he supports
// the XContentIdentifierFactory interface. Otherwise create standard
// implementation object for XContentIdentifier.
//////////////////////////////////////////////////////////////////////
Reference< XContentIdentifier > xIdentifier;
Reference< XContentProvider > xProv
= queryContentProvider( ContentId, sal_True );
if ( xProv.is() )
{
Reference< XContentIdentifierFactory > xFac( xProv, UNO_QUERY );
if ( xFac.is() )
xIdentifier = xFac->createContentIdentifier( ContentId );
}
if ( !xIdentifier.is() )
xIdentifier = new ContentIdentifier( m_xSMgr, ContentId );
return xIdentifier;
}
//=========================================================================
//
// XCommandProcessor methods.
2000-10-16 13:56:13 +00:00
//
//=========================================================================
// virtual
sal_Int32 SAL_CALL UniversalContentBroker::createCommandIdentifier()
throw( RuntimeException )
2000-10-16 13:56:13 +00:00
{
osl::MutexGuard aGuard( m_aMutex );
2000-10-16 13:56:13 +00:00
// Just increase counter on every call to generate an identifier.
return ++m_nCommandId;
2000-10-16 13:56:13 +00:00
}
//=========================================================================
// virtual
Any SAL_CALL UniversalContentBroker::execute(
const Command& aCommand,
sal_Int32,
const Reference< XCommandEnvironment >& Environment )
throw( Exception, CommandAbortedException, RuntimeException )
2000-10-16 13:56:13 +00:00
{
Any aRet;
2000-10-16 13:56:13 +00:00
//////////////////////////////////////////////////////////////////////
// Note: Don't forget to adapt ucb_commands::CommandProcessorInfo
// ctor in ucbcmds.cxx when adding new commands!
2000-10-16 13:56:13 +00:00
//////////////////////////////////////////////////////////////////////
if ( ( aCommand.Handle == GETCOMMANDINFO_HANDLE ) ||
2001-06-25 07:51:54 +00:00
aCommand.Name.equalsAsciiL(
RTL_CONSTASCII_STRINGPARAM( GETCOMMANDINFO_NAME ) ) )
{
//////////////////////////////////////////////////////////////////
// getCommandInfo
//////////////////////////////////////////////////////////////////
2000-10-16 13:56:13 +00:00
aRet <<= getCommandInfo();
}
else if ( ( aCommand.Handle == GLOBALTRANSFER_HANDLE ) ||
2001-06-25 07:51:54 +00:00
aCommand.Name.equalsAsciiL(
RTL_CONSTASCII_STRINGPARAM(GLOBALTRANSFER_NAME ) ) )
2000-10-16 13:56:13 +00:00
{
//////////////////////////////////////////////////////////////////
// globalTransfer
//////////////////////////////////////////////////////////////////
GlobalTransferCommandArgument aTransferArg;
2001-06-25 07:51:54 +00:00
if ( !( aCommand.Argument >>= aTransferArg ) )
2000-10-16 13:56:13 +00:00
{
2001-06-25 07:51:54 +00:00
ucbhelper::cancelCommandExecution(
makeAny( IllegalArgumentException(
rtl::OUString::createFromAscii(
"Wrong argument type!" ),
static_cast< cppu::OWeakObject * >( this ),
-1 ) ),
Environment );
// Unreachable
2000-10-16 13:56:13 +00:00
}
2001-06-25 07:51:54 +00:00
globalTransfer( aTransferArg, Environment );
2000-10-16 13:56:13 +00:00
}
else
{
//////////////////////////////////////////////////////////////////
// Unknown command
//////////////////////////////////////////////////////////////////
2000-10-16 13:56:13 +00:00
2001-06-25 07:51:54 +00:00
ucbhelper::cancelCommandExecution(
makeAny( UnsupportedCommandException(
rtl::OUString(),
static_cast< cppu::OWeakObject * >( this ) ) ),
Environment );
// Unreachable
}
return aRet;
}
//=========================================================================
// virtual
void SAL_CALL UniversalContentBroker::abort( sal_Int32 )
throw( RuntimeException )
{
// @@@ Not implemeted ( yet).
2000-10-16 13:56:13 +00:00
}
//=========================================================================
//
// Non-interface methods
//
//=========================================================================
Reference< XContentProvider > UniversalContentBroker::queryContentProvider(
const OUString& Identifier,
sal_Bool bResolved )
2000-10-16 13:56:13 +00:00
{
osl::MutexGuard aGuard( m_aMutex );
ProviderList_Impl const * pList = m_aProviders.map( Identifier );
2000-10-16 13:56:13 +00:00
return pList ? bResolved ? pList->front().getResolvedProvider()
: pList->front().getProvider()
: Reference< XContentProvider >();
}
//=========================================================================
//
// ProviderListEntry_Impl implementation.
//
//=========================================================================
Reference< XContentProvider > ProviderListEntry_Impl::resolveProvider() const
{
if ( !m_xResolvedProvider.is() )
{
Reference< XContentProviderSupplier > xSupplier(
m_xProvider, UNO_QUERY );
if ( xSupplier.is() )
m_xResolvedProvider = xSupplier->getContentProvider();
if ( !m_xResolvedProvider.is() )
m_xResolvedProvider = m_xProvider;
}
return m_xResolvedProvider;
}