Files
libreoffice/dbaccess/source/ui/browser/genericcontroller.cxx

1148 lines
43 KiB
C++
Raw Normal View History

2001-01-09 14:43:04 +00:00
/*************************************************************************
*
* $RCSfile: genericcontroller.cxx,v $
*
* $Revision: 1.45 $
2001-01-09 14:43:04 +00:00
*
* last change: $Author: fs $ $Date: 2002-09-24 15:13:36 $
2001-01-09 14:43:04 +00:00
*
* The Contents of this file are made available subject to the terms of
* either of the following licenses
*
* - GNU Lesser General Public License Version 2.1
* - Sun Industry Standards Source License Version 1.1
*
* Sun Microsystems Inc., October, 2000
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2000 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
*
* 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.
*
* 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.
*
* 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
*
*
* Sun Industry Standards Source License Version 1.1
* =================================================
* The contents of this file are subject to the Sun Industry Standards
* Source License Version 1.1 (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.openoffice.org/license.html.
*
* Software provided under this License is provided on an AS IS basis,
* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
* WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
* MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
* See the License for the specific provisions governing your rights and
* obligations concerning the Software.
*
* The Initial Developer of the Original Code is: Sun Microsystems, Inc.
*
* Copyright: 2000 by Sun Microsystems, Inc.
*
* All Rights Reserved.
*
* Contributor(s): _______________________________________
*
*
************************************************************************/
#ifndef DBAUI_GENERICCONTROLLER_HXX
#include "genericcontroller.hxx"
#endif
#ifndef _COMPHELPER_UNO3_HXX_
#include <comphelper/uno3.hxx>
#endif
#ifndef _TOOLKIT_AWT_VCLXWINDOW_HXX_
#include <toolkit/awt/vclxwindow.hxx>
#endif
#ifndef DBACCESS_UI_BROWSER_ID_HXX
#include "browserids.hxx"
#endif
#ifndef _SV_TOOLBOX_HXX
#include <vcl/toolbox.hxx>
#endif
2001-04-23 08:30:31 +00:00
#ifndef _TOOLKIT_HELPER_VCLUNOHELPER_HXX_
#include <toolkit/helper/vclunohelper.hxx>
#endif
2001-01-09 14:43:04 +00:00
#ifndef DBAUI_DATAVIEW_HXX
#include "dataview.hxx"
#endif
#ifndef _TOOLS_DEBUG_HXX
#include <tools/debug.hxx>
#endif
#ifndef DBACCESS_SHARED_DBUSTRINGS_HRC
#include "dbustrings.hrc"
#endif
#ifndef _VCL_STDTEXT_HXX
#include <vcl/stdtext.hxx>
#endif
#ifndef _CPPUHELPER_TYPEPROVIDER_HXX_
#include <cppuhelper/typeprovider.hxx>
#endif
#ifndef _COMPHELPER_SEQUENCE_HXX_
#include <comphelper/sequence.hxx>
#endif
#ifndef _COMPHELPER_EXTRACT_HXX_
#include <comphelper/extract.hxx>
2001-01-09 14:43:04 +00:00
#endif
#ifndef _COM_SUN_STAR_SDBC_XDATASOURCE_HPP_
#include <com/sun/star/sdbc/XDataSource.hpp>
#endif
2001-02-27 14:11:05 +00:00
#ifndef _COM_SUN_STAR_SDB_SQLCONTEXT_HPP_
#include <com/sun/star/sdb/SQLContext.hpp>
#endif
2001-01-09 14:43:04 +00:00
#ifndef _COM_SUN_STAR_SDB_XCOMPLETEDCONNECTION_HPP_
#include <com/sun/star/sdb/XCompletedConnection.hpp>
#endif
#ifndef _COM_SUN_STAR_BEANS_XPROPERTYSET_HPP_
#include <com/sun/star/beans/XPropertySet.hpp>
#endif
#ifndef _COM_SUN_STAR_TASK_XINTERACTIONHANDLER_HPP_
#include <com/sun/star/task/XInteractionHandler.hpp>
#endif
2002-06-26 07:17:57 +00:00
#ifndef _COM_SUN_STAR_UTIL_XCLOSEABLE_HPP_
#include <com/sun/star/util/XCloseable.hpp>
#endif
2001-02-23 14:17:34 +00:00
#ifndef DBAUI_TOOLS_HXX
#include "UITools.hxx"
2001-01-09 14:43:04 +00:00
#endif
#ifndef _SV_WAITOBJ_HXX
#include <vcl/waitobj.hxx>
#endif
2001-04-24 13:36:44 +00:00
#ifndef _URLOBJ_HXX
#include <tools/urlobj.hxx>
#endif
#ifndef SVTOOLS_URIHELPER_HXX
#include <svtools/urihelper.hxx>
#endif
#ifndef _DBAUI_DATASOURCECONNECTOR_HXX_
#include "datasourceconnector.hxx"
#endif
2002-08-26 06:52:15 +00:00
#ifndef INCLUDED_SVTOOLS_MODULEOPTIONS_HXX
#include <svtools/moduleoptions.hxx>
#endif
#ifndef _COM_SUN_STAR_FRAME_FRAMESEARCHFLAG_HPP_
#include <com/sun/star/frame/FrameSearchFlag.hpp>
#endif
2001-10-26 12:10:16 +00:00
#include <algorithm>
2001-01-09 14:43:04 +00:00
using namespace ::com::sun::star::uno;
using namespace ::com::sun::star::beans;
using namespace ::com::sun::star::frame;
using namespace ::com::sun::star::util;
using namespace ::com::sun::star::lang;
using namespace ::com::sun::star::container;
using namespace ::com::sun::star::sdbc;
using namespace ::com::sun::star::sdb;
using namespace ::com::sun::star::task;
2001-04-23 08:30:31 +00:00
using namespace ::com::sun::star::awt;
2001-01-09 14:43:04 +00:00
using namespace ::dbtools;
using namespace ::dbaui;
using namespace ::comphelper;
// -------------------------------------------------------------------------
#define ALL_FEATURES -1
// -------------------------------------------------------------------------
const ::rtl::OUString& getConfirmDeletionURL()
{
static const ::rtl::OUString sConfirmDeletionURL( RTL_CONSTASCII_USTRINGPARAM( ".uno:FormSlots/ConfirmDeletion" ) );
return sConfirmDeletionURL;
}
2001-01-09 14:43:04 +00:00
// -------------------------------------------------------------------------
OGenericUnoController::OGenericUnoController(const Reference< XMultiServiceFactory >& _rM)
:OGenericUnoController_COMPBASE(m_aMutex)
2001-03-01 14:17:55 +00:00
,m_aAsyncInvalidateAll(LINK(this, OGenericUnoController, OnAsyncInvalidateAll))
2001-09-19 12:20:12 +00:00
,m_aAsyncCloseTask(LINK(this, OGenericUnoController, OnAsyncCloseTask))
2001-01-09 14:43:04 +00:00
,m_xMultiServiceFacatory(_rM)
,m_bCurrentlyModified(sal_False)
,m_bFrameUiActive(sal_False)
,m_pView(NULL)
{
try
{
m_xUrlTransformer = Reference< XURLTransformer > (_rM->createInstance(::rtl::OUString::createFromAscii("com.sun.star.util.URLTransformer")), UNO_QUERY);
2001-01-09 14:43:04 +00:00
}
catch(Exception&)
{
}
}
// -----------------------------------------------------------------------------
sal_Bool OGenericUnoController::Construct(Window* pParent)
{
OSL_ENSURE( getView(), "the view is NULL!" );
if ( getView() )
{
getView()->Construct();
getView()->Show();
}
2001-01-09 14:43:04 +00:00
// want to have a toolbox ?
ToolBox* pTB = CreateToolBox(getView());
getView()->setToolBox(pTB);
if (pTB) // we want to handle the select
{
pTB->SetSelectHdl( LINK( this, OGenericUnoController, OnToolBoxSelected ) );
pTB->SetClickHdl( LINK( this, OGenericUnoController, OnToolBoxClicked ) );
}
2001-01-09 14:43:04 +00:00
AddSupportedFeatures();
// create the database context
DBG_ASSERT(m_xMultiServiceFacatory.is(), "OGenericUnoController::Construct need a service factory!");
2001-01-09 14:43:04 +00:00
try
{
m_xDatabaseContext = Reference< XNameAccess >(m_xMultiServiceFacatory->createInstance(SERVICE_SDB_DATABASECONTEXT), UNO_QUERY);
}
catch(Exception&)
{
DBG_ERROR("OGenericUnoController::Construct: could not create (or start listening at) the database context!");
2001-01-09 14:43:04 +00:00
}
if (!m_xDatabaseContext.is())
{ // at least notify the user. Though the whole component does not make any sense without the database context ...
ShowServiceNotAvailableError(getView(), String(SERVICE_SDB_DATABASECONTEXT), sal_True);
}
return sal_True;
}
//------------------------------------------------------------------------------
IMPL_LINK(OGenericUnoController, OnAsyncInvalidateAll, void*, EMPTYARG)
{
if ( !OGenericUnoController_COMPBASE::rBHelper.bInDispose && !OGenericUnoController_COMPBASE::rBHelper.bDisposed )
InvalidateFeature_Impl();
2001-01-09 14:43:04 +00:00
return 0L;
}
// -----------------------------------------------------------------------------
Reference< XWindow > OGenericUnoController::getComponentWindow() const
2001-01-09 14:43:04 +00:00
{
return VCLUnoHelper::GetInterface( getView() );
2001-01-09 14:43:04 +00:00
}
2001-04-23 08:30:31 +00:00
2001-01-09 14:43:04 +00:00
// -------------------------------------------------------------------------
void SAL_CALL OGenericUnoController::initialize( const Sequence< Any >& aArguments ) throw(Exception, RuntimeException)
{
Reference< XWindow > xParent;
Reference< XFrame > xFrame;
2001-01-09 14:43:04 +00:00
PropertyValue aValue;
const Any* pBegin = aArguments.getConstArray();
const Any* pEnd = pBegin + aArguments.getLength();
for ( ; pBegin != pEnd; ++pBegin )
2001-01-09 14:43:04 +00:00
{
if ( ( *pBegin >>= aValue ) && ( 0 == aValue.Name.compareToAscii( "Frame" ) ) )
2001-01-09 14:43:04 +00:00
{
if((aValue.Value >>= xFrame) && xFrame.is())
{
xParent = xFrame->getContainerWindow();
VCLXWindow* pParentComponent = VCLXWindow::GetImplementation(xParent);
Window* pParentWin = pParentComponent ? pParentComponent->GetWindow() : NULL;
if (!pParentWin)
{
throw Exception(::rtl::OUString::createFromAscii("Parent window is null"),*this);
}
if(xFrame.is() && Construct(pParentWin))
{
2001-04-23 08:30:31 +00:00
xFrame->setComponent(getComponentWindow(), this);
2001-01-09 14:43:04 +00:00
attachFrame(xFrame);
pParentComponent->setVisible(sal_True);
2001-04-24 13:36:44 +00:00
loadMenu(xFrame);
2001-01-09 14:43:04 +00:00
}
// remove the container window
::dbaui::notifySystemWindow(pParentWin,pParentWin,::comphelper::mem_fun(&TaskPaneList::RemoveWindow));
2001-01-09 14:43:04 +00:00
break; // no more needed here
}
else
{
2001-03-23 12:33:59 +00:00
OSL_ENSURE(0,"OGenericUnoController::initialize: Frame is null!");
2001-01-09 14:43:04 +00:00
}
}
}
}
//------------------------------------------------------------------------------
Any SAL_CALL OGenericUnoController::queryInterface(const Type& _rType) throw (RuntimeException)
{
Any aReturn = OGenericUnoController_COMPBASE::queryInterface(_rType);
if (!aReturn.hasValue())
aReturn = OGenericUnoController_CTRBASE::queryInterface(_rType);
return aReturn;
}
//------------------------------------------------------------------------------
void SAL_CALL OGenericUnoController::acquire( ) throw ()
{
OGenericUnoController_COMPBASE::acquire();
}
//------------------------------------------------------------------------------
void SAL_CALL OGenericUnoController::release( ) throw ()
{
OGenericUnoController_COMPBASE::release();
}
//------------------------------------------------------------------------------
Sequence< Type > SAL_CALL OGenericUnoController::getTypes( ) throw (RuntimeException)
{
return ::comphelper::concatSequences(
OGenericUnoController_COMPBASE::getTypes(),
OGenericUnoController_CTRBASE::getTypes()
);
}
//------------------------------------------------------------------------------
Sequence< sal_Int8 > SAL_CALL OGenericUnoController::getImplementationId( ) throw (RuntimeException)
{
static ::cppu::OImplementationId * pId = 0;
if (! pId)
{
::osl::MutexGuard aGuard( ::osl::Mutex::getGlobalMutex() );
if (! pId)
{
static ::cppu::OImplementationId aId;
pId = &aId;
}
}
return pId->getImplementationId();
}
// -------------------------------------------------------------------------
sal_Bool OGenericUnoController::startFrameListening( )
{
if ( m_xCurrentFrame.is() )
m_xCurrentFrame->addFrameActionListener( static_cast< XFrameActionListener* >( this ) );
return m_xCurrentFrame.is();
}
// -------------------------------------------------------------------------
void OGenericUnoController::stopFrameListening( )
{
if ( m_xCurrentFrame.is() )
m_xCurrentFrame->removeFrameActionListener( static_cast< XFrameActionListener* >( this ) );
}
2001-01-09 14:43:04 +00:00
// -------------------------------------------------------------------------
void OGenericUnoController::disposing(const EventObject& Source) throw( RuntimeException )
{
// our frame ?
Reference< XFrame > xSourceFrame(Source.Source, UNO_QUERY);
if ( xSourceFrame == m_xCurrentFrame )
stopFrameListening( );
2001-01-09 14:43:04 +00:00
}
//------------------------------------------------------------------------
void OGenericUnoController::modified(const EventObject& aEvent) throw( RuntimeException )
{
m_bCurrentlyModified = sal_True;
InvalidateFeature(ID_BROWSER_SAVEDOC);
2001-01-09 14:43:04 +00:00
InvalidateFeature(ID_BROWSER_UNDO);
}
// -----------------------------------------------------------------------
void OGenericUnoController::attachFrame(const Reference< XFrame > & xFrame) throw( RuntimeException )
2001-01-09 14:43:04 +00:00
{
stopFrameListening( );
2001-01-09 14:43:04 +00:00
m_xCurrentFrame = xFrame;
if ( startFrameListening( ) )
2001-01-09 14:43:04 +00:00
m_bFrameUiActive = m_xCurrentFrame->isActive();
}
2001-01-09 14:43:04 +00:00
// -----------------------------------------------------------------------
sal_Bool OGenericUnoController::ImplInvalidateTBItem(sal_uInt16 nId, const FeatureState& rState)
{
ToolBox* pTB = getView() ? getView()->getToolBox() : NULL;
if (!pTB || (pTB->GetItemPos(nId) == TOOLBOX_ITEM_NOTFOUND))
return sal_False;
pTB->EnableItem(nId, rState.bEnabled);
switch (rState.aState.getValueTypeClass())
{
case TypeClass_BOOLEAN:
pTB->CheckItem(nId, ::comphelper::getBOOL(rState.aState));
break;
case TypeClass_STRING:
if (pTB->GetItemWindow(nId))
pTB->GetItemWindow(nId)->SetText(::comphelper::getString(rState.aState));
2002-08-19 06:32:53 +00:00
else
pTB->SetQuickHelpText(nId,::comphelper::getString(rState.aState));
2001-01-09 14:43:04 +00:00
break;
case TypeClass_VOID:
break;
default:
DBG_WARNING("OGenericUnoController::ImplInvalidateTBItem : don't know what to do with the item state !");
}
return sal_True;
}
// -----------------------------------------------------------------------
void OGenericUnoController::ImplBroadcastFeatureState(const ::rtl::OUString& _rFeature, const Reference< XStatusListener > & xListener, sal_Bool _bIgnoreCache)
2001-01-09 14:43:04 +00:00
{
sal_uInt16 nFeat = (sal_uInt16)m_aSupportedFeatures[_rFeature];
FeatureState aFeatState(GetState(nFeat));
2001-01-09 14:43:04 +00:00
FeatureState& rCachedState = m_aStateCache[nFeat]; // creates if neccessary
if(!_bIgnoreCache)
{
// check if we really need to notify the listeners : this method may be called much more often than needed, so check
// the cached state of the feature
sal_Bool bAlreadyCached = (m_aStateCache.find(nFeat) != m_aStateCache.end());
if (bAlreadyCached && (rCachedState.bEnabled == aFeatState.bEnabled))
{ // the enabled flag hasn't changed, maybe the state ?
if (rCachedState.aState.getValueTypeClass() == aFeatState.aState.getValueTypeClass())
{ // at least the type of the state hasn't
sal_Bool bEqualValue = sal_False;
switch (rCachedState.aState.getValueTypeClass())
{
case TypeClass_VOID:
bEqualValue = !aFeatState.aState.hasValue();
break;
case TypeClass_BOOLEAN:
bEqualValue = ::comphelper::getBOOL(rCachedState.aState) == ::comphelper::getBOOL(aFeatState.aState);
break;
case TypeClass_SHORT:
bEqualValue = ::comphelper::getINT16(rCachedState.aState) == ::comphelper::getINT16(aFeatState.aState);
break;
case TypeClass_LONG:
bEqualValue = ::comphelper::getINT32(rCachedState.aState) == ::comphelper::getINT32(aFeatState.aState);
break;
case TypeClass_STRING:
bEqualValue = ::comphelper::getString(rCachedState.aState).equals(::comphelper::getString(aFeatState.aState));
break;
default:
DBG_ERROR("OGenericUnoController::ImplBroadcastFeatureState : unknown state type (not implemented yet) !");
break;
}
if (bEqualValue)
return;
}
}
}
rCachedState = aFeatState;
FeatureStateEvent aEvent;
2001-01-09 14:43:04 +00:00
aEvent.FeatureURL.Complete = _rFeature;
if (m_xUrlTransformer.is())
m_xUrlTransformer->parseStrict(aEvent.FeatureURL);
aEvent.Source = (XDispatch*)this;
2001-01-09 14:43:04 +00:00
aEvent.IsEnabled = aFeatState.bEnabled;
aEvent.Requery = aFeatState.bRequery;
aEvent.State = aFeatState.aState;
// a special listener ?
if (xListener.is())
xListener->statusChanged(aEvent);
else
{ // no -> iterate through all listeners responsible for the URL
2001-01-09 14:43:04 +00:00
DispatchIterator iterSearch = m_arrStatusListener.begin();
DispatchIterator iterEnd = m_arrStatusListener.end();
while (iterSearch != iterEnd)
{
DispatchTarget& rCurrent = *iterSearch;
if (rCurrent.aURL.Complete.equals(_rFeature))
rCurrent.xListener->statusChanged(aEvent);
++iterSearch;
}
}
// give the TB a chance
ImplInvalidateTBItem(nFeat, aFeatState);
}
// -----------------------------------------------------------------------
void OGenericUnoController::InvalidateFeature(const ::rtl::OUString& _rURLPath, const Reference< XStatusListener > & _xListener, sal_Bool _bForceBroadcast)
2001-01-09 14:43:04 +00:00
{
ImplInvalidateFeature( m_aSupportedFeatures[_rURLPath], _xListener, _bForceBroadcast );
2001-01-09 14:43:04 +00:00
}
2001-01-09 14:43:04 +00:00
// -----------------------------------------------------------------------------
void OGenericUnoController::InvalidateFeature_Impl()
{
#ifdef DBG_UTIL
static sal_Int32 s_nRecursions = 0;
++s_nRecursions;
#endif
2001-01-09 14:43:04 +00:00
sal_Bool bEmpty = sal_True;
FeaturePair aNextFeature;
2001-01-09 14:43:04 +00:00
{
::osl::MutexGuard aGuard( m_aFeatureMutex);
bEmpty = m_aFeaturesToInvalidate.empty();
if (!bEmpty)
aNextFeature = m_aFeaturesToInvalidate.front();
2001-01-09 14:43:04 +00:00
}
while(!bEmpty)
{
if ( ALL_FEATURES == aNextFeature.nId )
{
2001-01-09 14:43:04 +00:00
InvalidateAll_Impl();
break;
}
2001-01-09 14:43:04 +00:00
else
{
SupportedFeatures::iterator aFeaturePos = ::std::find_if(
m_aSupportedFeatures.begin(),
m_aSupportedFeatures.end(),
::std::bind2nd( SupportedFeaturesFunc(), aNextFeature.nId )
);
if ( m_aSupportedFeatures.end() != aFeaturePos )
// we really know this feature
ImplBroadcastFeatureState( aFeaturePos->first, aNextFeature.xListener, aNextFeature.bForceBroadcast );
else
// we don't know this feature -> at least give the TB a chance
ImplInvalidateTBItem((sal_uInt16)aNextFeature.nId, GetState((sal_uInt16)aNextFeature.nId));
2001-01-09 14:43:04 +00:00
}
::osl::MutexGuard aGuard( m_aFeatureMutex);
m_aFeaturesToInvalidate.pop_front();
bEmpty = m_aFeaturesToInvalidate.empty();
if (!bEmpty)
aNextFeature = m_aFeaturesToInvalidate.front();
2001-01-09 14:43:04 +00:00
}
#ifdef DBG_UTIL
--s_nRecursions;
#endif
2001-01-09 14:43:04 +00:00
}
2001-01-09 14:43:04 +00:00
// -----------------------------------------------------------------------
void OGenericUnoController::ImplInvalidateFeature( sal_Int32 _nId, const ::com::sun::star::uno::Reference< ::com::sun::star::frame::XStatusListener >& _xListener, sal_Bool _bForceBroadcast )
2001-01-09 14:43:04 +00:00
{
FeaturePair aPair;
aPair.nId = _nId;
aPair.xListener = _xListener;
aPair.bForceBroadcast = _bForceBroadcast;
sal_Bool bWasEmpty;
2001-01-09 14:43:04 +00:00
{
::osl::MutexGuard aGuard( m_aFeatureMutex);
bWasEmpty = m_aFeaturesToInvalidate.empty();
m_aFeaturesToInvalidate.push_back(aPair);
}
if ( bWasEmpty )
2001-01-09 14:43:04 +00:00
m_aAsyncInvalidateAll.Call();
}
// -----------------------------------------------------------------------
void OGenericUnoController::InvalidateFeature(sal_uInt16 _nId, const Reference< XStatusListener > & _xListener, sal_Bool _bForceBroadcast)
{
ImplInvalidateFeature( _nId, _xListener, _bForceBroadcast );
}
2001-01-09 14:43:04 +00:00
// -----------------------------------------------------------------------
void OGenericUnoController::InvalidateAll()
{
ImplInvalidateFeature( ALL_FEATURES, NULL, sal_True );
2001-01-09 14:43:04 +00:00
}
2001-01-09 14:43:04 +00:00
// -----------------------------------------------------------------------------
void OGenericUnoController::InvalidateAll_Impl()
{
sal_uInt16 i;
// ---------------------------------
// invalidate all aupported features
SupportedFeatures::const_iterator aIter = m_aSupportedFeatures.begin();
for(;aIter != m_aSupportedFeatures.end();++aIter)
{
ImplBroadcastFeatureState(aIter->first, Reference< XStatusListener > (), sal_True);
2001-01-09 14:43:04 +00:00
}
// ------------------------------
// invalidate all slots in the TB (if any)
ToolBox* pTB = getView() ? getView()->getToolBox() : NULL;
if (pTB)
{
sal_uInt16 nCount = pTB->GetItemCount();
for ( i = 0; i < nCount; ++i )
{
sal_uInt16 nId = pTB->GetItemId(i);
if ( nId )
InvalidateFeature( nId );
}
2001-01-09 14:43:04 +00:00
}
{
::osl::MutexGuard aGuard( m_aFeatureMutex);
DBG_ASSERT(m_aFeaturesToInvalidate.size(), "OGenericUnoController::InvalidateAll_Impl: to be called from within InvalidateFeature_Impl only!");
2001-01-09 14:43:04 +00:00
m_aFeaturesToInvalidate.pop_front();
if(!m_aFeaturesToInvalidate.empty())
m_aAsyncInvalidateAll.Call();
}
}
// -----------------------------------------------------------------------
Reference< XDispatch > OGenericUnoController::queryDispatch(const URL& aURL, const ::rtl::OUString& aTargetFrameName, sal_Int32 nSearchFlags) throw( RuntimeException )
2001-01-09 14:43:04 +00:00
{
Reference< XDispatch > xReturn;
2001-01-09 14:43:04 +00:00
// URL's we can handle ourself?
if ( aURL.Complete.equals( getConfirmDeletionURL() )
|| ( m_aSupportedFeatures.find( aURL.Complete ) != m_aSupportedFeatures.end() )
)
{
xReturn = static_cast< XDispatch* >( this );
}
// no? -> ask the slave dispatcher
else if ( m_xSlaveDispatcher.is() )
{
xReturn = m_xSlaveDispatcher->queryDispatch(aURL, aTargetFrameName, nSearchFlags);
}
2001-01-09 14:43:04 +00:00
// outta here
return xReturn;
2001-01-09 14:43:04 +00:00
}
// -----------------------------------------------------------------------
Sequence< Reference< XDispatch > > OGenericUnoController::queryDispatches(const Sequence< DispatchDescriptor >& aDescripts) throw( RuntimeException )
2001-01-09 14:43:04 +00:00
{
Sequence< Reference< XDispatch > > aReturn;
sal_Int32 nLen = aDescripts.getLength();
if ( nLen )
{
aReturn.realloc( nLen );
Reference< XDispatch >* pReturn = aReturn.getArray();
const Reference< XDispatch >* pReturnEnd = aReturn.getArray() + nLen;
const DispatchDescriptor* pDescripts = aDescripts.getConstArray();
2001-01-09 14:43:04 +00:00
for ( ; pReturn != pReturnEnd; ++ pReturn, ++pDescripts )
{
*pReturn = queryDispatch( pDescripts->FeatureURL, pDescripts->FrameName, pDescripts->SearchFlags );
}
}
2001-01-09 14:43:04 +00:00
return aReturn;
}
// -----------------------------------------------------------------------
Reference< XDispatchProvider > OGenericUnoController::getSlaveDispatchProvider(void) throw( RuntimeException )
2001-01-09 14:43:04 +00:00
{
return m_xSlaveDispatcher;
}
// -----------------------------------------------------------------------
void OGenericUnoController::setSlaveDispatchProvider(const Reference< XDispatchProvider > & _xNewProvider) throw( RuntimeException )
2001-01-09 14:43:04 +00:00
{
m_xSlaveDispatcher = _xNewProvider;
}
// -----------------------------------------------------------------------
Reference< XDispatchProvider > OGenericUnoController::getMasterDispatchProvider(void) throw( RuntimeException )
2001-01-09 14:43:04 +00:00
{
return m_xMasterDispatcher;
}
// -----------------------------------------------------------------------
void OGenericUnoController::setMasterDispatchProvider(const Reference< XDispatchProvider > & _xNewProvider) throw( RuntimeException )
2001-01-09 14:43:04 +00:00
{
m_xMasterDispatcher = _xNewProvider;
}
// -----------------------------------------------------------------------
2002-02-19 12:54:45 +00:00
void OGenericUnoController::dispatch(const URL& _aURL, const Sequence< PropertyValue >& aArgs) throw(RuntimeException)
2001-01-09 14:43:04 +00:00
{
2002-02-11 11:43:58 +00:00
executeUnChecked(_aURL);
2001-01-09 14:43:04 +00:00
}
// -----------------------------------------------------------------------
void OGenericUnoController::addStatusListener(const Reference< XStatusListener > & aListener, const URL& _rURL) throw(RuntimeException)
2001-01-09 14:43:04 +00:00
{
// remeber the listener together with the URL
2001-04-24 13:36:44 +00:00
m_arrStatusListener.insert(m_arrStatusListener.end(), DispatchTarget(_rURL, aListener));
2001-01-09 14:43:04 +00:00
// initially broadcast the state
ImplBroadcastFeatureState(_rURL.Complete, aListener, sal_True);
2001-01-09 14:43:04 +00:00
// force the new state to be broadcasted to the new listener
}
// -----------------------------------------------------------------------
void OGenericUnoController::removeStatusListener(const Reference< XStatusListener > & aListener, const URL& _rURL) throw(RuntimeException)
2001-01-09 14:43:04 +00:00
{
DispatchIterator iterSearch = m_arrStatusListener.begin();
2001-04-24 13:36:44 +00:00
sal_Bool bRemoveForAll = (_rURL.Complete.getLength() == 0);
while ( iterSearch != m_arrStatusListener.end() )
2001-01-09 14:43:04 +00:00
{
DispatchTarget& rCurrent = *iterSearch;
if ( (rCurrent.xListener == aListener)
&& ( bRemoveForAll
2001-04-24 13:36:44 +00:00
|| (rCurrent.aURL.Complete.equals(_rURL.Complete))
2001-01-09 14:43:04 +00:00
)
)
{
m_arrStatusListener.erase( iterSearch );
2001-01-09 14:43:04 +00:00
if (!bRemoveForAll)
// remove the listener only for the given URL, so we can exit the loop after deletion
2001-01-09 14:43:04 +00:00
break;
}
else
++iterSearch;
}
2001-04-24 13:36:44 +00:00
SupportedFeatures::const_iterator aIter = m_aSupportedFeatures.find(_rURL.Complete);
2001-01-09 14:43:04 +00:00
if (aIter != m_aSupportedFeatures.end())
{ // clear the cache for that feature
StateCacheIterator aCachePos = m_aStateCache.find((sal_uInt16)aIter->second);
2001-01-09 14:43:04 +00:00
if (aCachePos != m_aStateCache.end())
m_aStateCache.erase(aCachePos);
}
// now remove the listener from the deque
::osl::MutexGuard aGuard( m_aFeatureMutex);
2001-08-27 05:57:24 +00:00
m_aFeaturesToInvalidate.erase(
::std::remove_if( m_aFeaturesToInvalidate.begin(),
m_aFeaturesToInvalidate.end(),
::std::bind2nd(FeaturePairFunctor(),aListener))
,m_aFeaturesToInvalidate.end());
2001-01-09 14:43:04 +00:00
}
// -----------------------------------------------------------------------
2001-03-01 14:17:55 +00:00
void OGenericUnoController::disposing()
2001-01-09 14:43:04 +00:00
{
// our status listeners, too
2001-08-27 05:57:24 +00:00
while (!m_arrStatusListener.empty())
2001-01-09 14:43:04 +00:00
{
DispatchIterator iterCurrent = m_arrStatusListener.begin();
DispatchTarget& rCurrent = *iterCurrent;
EventObject aDisposeEvent;
aDisposeEvent.Source = static_cast<XWeak*>(this);
2001-01-09 14:43:04 +00:00
#ifdef DBG_UTIL
sal_Int32 nSize = m_arrStatusListener.size();
#endif
rCurrent.xListener->disposing(aDisposeEvent);
2001-04-24 13:36:44 +00:00
DBG_ASSERT(nSize > (sal_Int32)m_arrStatusListener.size(), "OGenericUnoController::dispose : the listener did not call removeStatusListener !");
2001-01-09 14:43:04 +00:00
// in disposing the status listener should remove itself via removeStatusListener, therein we remove it from
// m_arrStatusListener, so the size should have decreased.
}
{
::osl::MutexGuard aGuard( m_aFeatureMutex);
m_aAsyncInvalidateAll.CancelCall();
2001-01-09 14:43:04 +00:00
m_aFeaturesToInvalidate.clear();
}
// check out from all the objects we are listening
// the frame
stopFrameListening( );
2001-01-09 14:43:04 +00:00
}
// -----------------------------------------------------------------------
void OGenericUnoController::addEventListener(const Reference< XEventListener > & aListener) throw(RuntimeException)
2001-01-09 14:43:04 +00:00
{
OGenericUnoController_COMPBASE::addEventListener(aListener);
2001-01-09 14:43:04 +00:00
}
// -----------------------------------------------------------------------
void OGenericUnoController::removeEventListener(const Reference< XEventListener > & aListener) throw(RuntimeException)
2001-01-09 14:43:04 +00:00
{
OGenericUnoController_COMPBASE::removeEventListener(aListener);
2001-01-09 14:43:04 +00:00
}
//------------------------------------------------------------------------------
void OGenericUnoController::frameAction(const FrameActionEvent& aEvent) throw( RuntimeException )
2001-01-09 14:43:04 +00:00
{
if ((XFrame*)aEvent.Frame.get() == (XFrame*)m_xCurrentFrame.get())
2001-04-24 13:36:44 +00:00
m_bFrameUiActive = (aEvent.Action == FrameAction_FRAME_UI_ACTIVATED);
2001-01-09 14:43:04 +00:00
}
//------------------------------------------------------------------------------
void OGenericUnoController::EmptyWindow()
{
2001-11-22 12:17:07 +00:00
if(m_xCurrentFrame.is())
2001-01-09 14:43:04 +00:00
{
2001-11-22 12:17:07 +00:00
m_xCurrentFrame->setComponent(NULL,NULL);
::comphelper::disposeComponent(m_xCurrentFrame);
2001-01-09 14:43:04 +00:00
}
2001-11-22 12:17:07 +00:00
2001-01-09 14:43:04 +00:00
}
//------------------------------------------------------------------------------
void OGenericUnoController::AddSupportedFeatures()
{
// add all supported features
m_aSupportedFeatures[ ::rtl::OUString::createFromAscii(".uno:Copy")] = ID_BROWSER_COPY;
m_aSupportedFeatures[ ::rtl::OUString::createFromAscii(".uno:Cut")] = ID_BROWSER_CUT;
m_aSupportedFeatures[ ::rtl::OUString::createFromAscii(".uno:Paste")] = ID_BROWSER_PASTE;
m_aSupportedFeatures[ ::rtl::OUString::createFromAscii(".uno:ClipboardFormatItems")] = ID_BROWSER_CLIPBOARD_FORMAT_ITEMS;
// since
2001-01-09 14:43:04 +00:00
m_aSupportedFeatures[ ::rtl::OUString::createFromAscii(".uno:DBSlots/EditDoc")] = ID_BROWSER_EDITDOC;
}
//------------------------------------------------------------------------------
FeatureState OGenericUnoController::GetState(sal_uInt16 nId) const
2001-01-09 14:43:04 +00:00
{
FeatureState aReturn;
// (disabled automatically)
try
{
switch (nId)
{
case ID_BROWSER_UNDO:
case ID_BROWSER_SAVEDOC:
aReturn.bEnabled = sal_True;
break;
}
}
catch(Exception& e)
{
#if DBG_UTIL
String sMessage("OGenericUnoController::GetState(", RTL_TEXTENCODING_ASCII_US);
sMessage += String::CreateFromInt32(nId);
sMessage.AppendAscii(") : catched an exception ! message : ");
sMessage += (const sal_Unicode*)e.Message;
DBG_ERROR(ByteString(sMessage, gsl_getSystemTextEncoding()).GetBuffer());
#else
e; // make compiler happy
#endif
}
return aReturn;
}
//------------------------------------------------------------------------------
void OGenericUnoController::onToolBoxSelected( sal_uInt16 _nSelectedItem )
{
Execute( _nSelectedItem );
}
//------------------------------------------------------------------------------
void OGenericUnoController::onToolBoxClicked( sal_uInt16 _nClickedItem )
{
}
//------------------------------------------------------------------------------
IMPL_LINK(OGenericUnoController, OnToolBoxClicked, ToolBox*, pToolBox)
{
onToolBoxClicked( pToolBox->GetCurItemId() );
return 0L;
}
2001-01-09 14:43:04 +00:00
//------------------------------------------------------------------------------
IMPL_LINK(OGenericUnoController, OnToolBoxSelected, ToolBox*, pToolBox)
{
onToolBoxSelected( pToolBox->GetCurItemId() );
2001-01-09 14:43:04 +00:00
return 0L;
}
// -------------------------------------------------------------------------
URL OGenericUnoController::getURLForId(sal_Int32 _nId) const
2001-01-09 14:43:04 +00:00
{
URL aReturn;
if ( m_xUrlTransformer.is() )
2001-08-27 05:57:24 +00:00
{
SupportedFeatures::const_iterator aIter = ::std::find_if(
m_aSupportedFeatures.begin(),
m_aSupportedFeatures.end(),
::std::bind2nd( SupportedFeaturesFunc(), _nId )
);
if ( m_aSupportedFeatures.end() != aIter && aIter->first.getLength() )
2001-01-09 14:43:04 +00:00
{
2001-08-27 05:57:24 +00:00
aReturn.Complete = aIter->first;
m_xUrlTransformer->parseStrict( aReturn );
2001-01-09 14:43:04 +00:00
}
2001-08-27 05:57:24 +00:00
}
2001-01-09 14:43:04 +00:00
return aReturn;
}
2001-01-09 14:43:04 +00:00
//-------------------------------------------------------------------------
sal_Bool SAL_CALL OGenericUnoController::supportsService(const ::rtl::OUString& ServiceName) throw(RuntimeException)
{
Sequence< ::rtl::OUString > aSupported(getSupportedServiceNames());
2001-01-09 14:43:04 +00:00
const ::rtl::OUString* pArray = aSupported.getConstArray();
const ::rtl::OUString* pArrayEnd = aSupported.getConstArray() + aSupported.getLength();
for ( ;( pArray != pArrayEnd ) && !pArray->equals( ServiceName ); ++pArray )
;
return pArray != pArrayEnd;
2001-01-09 14:43:04 +00:00
}
2001-01-09 14:43:04 +00:00
// -----------------------------------------------------------------------------
void OGenericUnoController::startConnectionListening(const Reference< XConnection >& _rxConnection)
2001-01-09 14:43:04 +00:00
{
// we have to remove ourself before dispoing the connection
Reference< XComponent > xComponent(_rxConnection, UNO_QUERY);
if (xComponent.is())
xComponent->addEventListener(static_cast<XFrameActionListener*>(this));
}
2001-01-09 14:43:04 +00:00
// -----------------------------------------------------------------------------
void OGenericUnoController::stopConnectionListening(const Reference< XConnection >& _rxConnection)
{
// we have to remove ourself before dispoing the connection
Reference< XComponent > xComponent(_rxConnection, UNO_QUERY);
if (xComponent.is())
xComponent->removeEventListener(static_cast<XFrameActionListener*>(this));
}
2001-01-26 13:15:24 +00:00
// -----------------------------------------------------------------------------
Reference<XConnection> OGenericUnoController::connect(const ::rtl::OUString& _rDataSourceName, sal_Bool _bStartListening)
{
const ::rtl::OUString sNoContext;
return connect( _rDataSourceName, sNoContext, sNoContext, _bStartListening );
}
// -----------------------------------------------------------------------------
Reference< XConnection > OGenericUnoController::connect(
const ::rtl::OUString& _rDataSourceName, const ::rtl::OUString& _rContextInformation,
const ::rtl::OUString& _rContextDetails, sal_Bool _bStartListening )
{
WaitObject aWaitCursor(getView());
2001-01-09 14:43:04 +00:00
ODatasourceConnector aConnector(m_xMultiServiceFacatory, getView(), _rContextInformation, _rContextDetails);
Reference<XConnection> xConnection = aConnector.connect(_rDataSourceName);
2001-01-09 14:43:04 +00:00
// be notified when connection is in disposing
if (_bStartListening)
startConnectionListening(xConnection);
return xConnection;
}
// -----------------------------------------------------------------------------
void OGenericUnoController::showError(const SQLExceptionInfo& _rInfo)
{
2001-02-23 14:17:34 +00:00
::dbaui::showError(_rInfo,getView(),getORB());
2001-01-09 14:43:04 +00:00
}
2001-02-23 14:17:34 +00:00
// -----------------------------------------------------------------------------
void OGenericUnoController::loadMenu(const Reference< XFrame >& _xFrame)
2001-04-24 13:36:44 +00:00
{
String sMenuName = getMenu();
if(sMenuName.Len())
{
INetURLObject aEntry( URIHelper::SmartRelToAbs(OModule::getResManager()->GetFileName()) );
String aMenuRes( RTL_CONSTASCII_USTRINGPARAM( "private:resource/" ));
aMenuRes += ( aEntry.GetName() += '/' );
aMenuRes += sMenuName;
URL aURL;
aURL.Complete = aMenuRes;
if ( m_xUrlTransformer.is() )
2001-04-24 13:36:44 +00:00
{
// Datei laden
m_xUrlTransformer->parseStrict( aURL );
2001-01-09 14:43:04 +00:00
2001-04-24 13:36:44 +00:00
Reference< XDispatchProvider > xProv( _xFrame, UNO_QUERY );
if ( xProv.is() )
{
Reference< XDispatch > aDisp = xProv->queryDispatch( aURL, ::rtl::OUString::createFromAscii("_menubar"), 12 );
if ( aDisp.is() )
aDisp->dispatch( aURL, Sequence<PropertyValue>() );
}
}
}
}
// -----------------------------------------------------------------------------
String OGenericUnoController::getMenu() const
{
return String();
}
// -----------------------------------------------------------------------------
void OGenericUnoController::closeTask()
{
2001-09-19 12:20:12 +00:00
m_aAsyncCloseTask.Call();
}
// -----------------------------------------------------------------------------
IMPL_LINK(OGenericUnoController, OnAsyncCloseTask, void*, EMPTYARG)
{
if(!OGenericUnoController_COMPBASE::rBHelper.bInDispose)
{
2002-06-24 09:29:26 +00:00
try
{
Reference< ::com::sun::star::util::XCloseable > xCloseable(m_xCurrentFrame,UNO_QUERY);
if(xCloseable.is())
xCloseable->close(sal_False); // false - holds the owner ship for this frame inside this object!
} catch(CloseVetoException&) {}
2001-09-19 12:20:12 +00:00
}
return 0L;
2001-04-24 13:36:44 +00:00
}
// -----------------------------------------------------------------------------
Any SAL_CALL OGenericUnoController::getViewData(void) throw( RuntimeException )
{
return Any();
}
// -----------------------------------------------------------------------------
void SAL_CALL OGenericUnoController::restoreViewData(const Any& Data) throw( RuntimeException )
{
}
// -----------------------------------------------------------------------------
sal_Bool SAL_CALL OGenericUnoController::attachModel(const Reference< XModel > & xModel) throw( RuntimeException )
{
return sal_False;
}
// -----------------------------------------------------------------------------
2002-02-11 11:43:58 +00:00
void OGenericUnoController::executeUnChecked(const ::com::sun::star::util::URL& _rCommand)
{
SupportedFeatures::const_iterator aIter = m_aSupportedFeatures.find(_rCommand.Complete);
if (aIter != m_aSupportedFeatures.end())
Execute( static_cast<sal_uInt16>(aIter->second) );
}
// -----------------------------------------------------------------------------
void OGenericUnoController::executeChecked(const ::com::sun::star::util::URL& _rCommand)
{
SupportedFeatures::const_iterator aIter = m_aSupportedFeatures.find(_rCommand.Complete);
if (aIter != m_aSupportedFeatures.end())
{
sal_uInt16 nSlotId = static_cast<sal_uInt16>(aIter->second);
if ( GetState( nSlotId ).bEnabled )
Execute( nSlotId );
}
}
// -----------------------------------------------------------------------------
2002-08-26 06:52:15 +00:00
//------------------------------------------------------------------------------
namespace
{
::rtl::OUString lcl_getModuleHelpModuleName( const Reference< XFrame >& _rxFrame )
{
const sal_Char* pReturn = NULL;
try
{
// get the model of the document in the given frame
Reference< XController > xController;
if ( _rxFrame.is() )
xController = _rxFrame->getController();
Reference< XModel > xModel;
if ( xController.is() )
xModel = xController->getModel();
Reference< XServiceInfo > xSI( xModel, UNO_QUERY );
if ( !xSI.is() )
{ // try to go up the frame hierarchy
Reference< XFrame > xParentFrame;
if ( _rxFrame.is() )
xParentFrame = xParentFrame.query( _rxFrame->getCreator() );
// did we find a parent frame? Which is no top-level frame?
if ( xParentFrame.is() && !_rxFrame->isTop() )
// TODO: to prevent framework assertions, re-insert this "isTop" once 98303 is fixed
return lcl_getModuleHelpModuleName( xParentFrame );
}
else
{
#ifdef _DEBUG
Sequence< ::rtl::OUString > sServiceNames = xSI->getSupportedServiceNames();
const ::rtl::OUString* pLoop = sServiceNames.getConstArray();
for ( sal_Int32 i=0; i<sServiceNames.getLength(); ++i, ++pLoop )
{
sal_Int32 nDummy = 0;
}
#endif
// check which service we know ....
static const sal_Char* pTransTable[] = {
"com.sun.star.text.TextDocument", "swriter",
"com.sun.star.sheet.SpreadsheetDocument", "scalc",
"com.sun.star.presentation.PresentationDocument", "simpress",
"com.sun.star.drawing.DrawingDocument", "sdraw",
"com.sun.star.formula.FormularProperties", "smath",
"com.sun.star.chart.ChartDocument", "schart"
};
OSL_ENSURE( ( sizeof( pTransTable ) / sizeof( pTransTable[0] ) ) % 2 == 0,
"lcl_getModuleHelpModuleName: odd size of translation table!" );
// loop through the table
sal_Int32 nTableEntries = ( sizeof( pTransTable ) / sizeof( pTransTable[0] ) ) / 2;
const sal_Char** pDocumentService = pTransTable;
const sal_Char** pHelpModuleName = pTransTable + 1;
for ( sal_Int32 j=0; j<nTableEntries; ++j )
{
if ( xSI->supportsService( ::rtl::OUString::createFromAscii( *pDocumentService ) ) )
{ // found a table entry which matches the model's services
pReturn = *pHelpModuleName;
break;
}
++pDocumentService; ++pDocumentService;
++pHelpModuleName; ++pHelpModuleName;
}
}
if ( !pReturn )
{
// could not determine the document type we're living in
// ->fallback
SvtModuleOptions aModOpt;
if ( aModOpt.IsModuleInstalled( SvtModuleOptions::E_SWRITER ) )
pReturn = "swriter";
else if ( aModOpt.IsModuleInstalled( SvtModuleOptions::E_SCALC ) )
pReturn = "scalc";
else if ( aModOpt.IsModuleInstalled( SvtModuleOptions::E_SIMPRESS ) )
pReturn = "simpress";
else if ( aModOpt.IsModuleInstalled( SvtModuleOptions::E_SDRAW ) )
pReturn = "sdraw";
else if ( aModOpt.IsModuleInstalled( SvtModuleOptions::E_SMATH ) )
pReturn = "smath";
else if ( aModOpt.IsModuleInstalled( SvtModuleOptions::E_SCHART ) )
pReturn = "schart";
else if ( aModOpt.IsModuleInstalled( SvtModuleOptions::E_SBASIC ) )
pReturn = "sbasic";
else
{
OSL_ENSURE( sal_False, "lcl_getModuleHelpModuleName: no installed module found" );
}
}
}
catch( const Exception& )
{
OSL_ENSURE( sal_False, "lcl_getModuleHelpModuleName: caught an exception!" );
}
if ( !pReturn )
pReturn = "swriter";
return ::rtl::OUString::createFromAscii( pReturn );
}
}
//------------------------------------------------------------------------------
void OGenericUnoController::openHelpAgent(sal_Int32 _nHelpId)
{
try
{
URL aURL = createHelpAgentURL(lcl_getModuleHelpModuleName( getFrame() ),_nHelpId);
if (m_xUrlTransformer.is())
m_xUrlTransformer->parseStrict(aURL);
Reference< XDispatchProvider > xDispProv(m_xCurrentFrame, UNO_QUERY);
Reference< XDispatch > xHelpDispatch;
if (xDispProv.is())
xHelpDispatch = xDispProv->queryDispatch(aURL, ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("_helpagent")), FrameSearchFlag::PARENT | FrameSearchFlag::SELF);
OSL_ENSURE(xHelpDispatch.is(), "SbaTableQueryBrowser::openHelpAgent: could not get a dispatcher!");
if (xHelpDispatch.is())
{
xHelpDispatch->dispatch(aURL, Sequence< PropertyValue >());
}
}
catch(const Exception&)
{
OSL_ENSURE(sal_False, "SbaTableQueryBrowser::openHelpAgent: caught an exception while executing the dispatch!");
}
}
// -----------------------------------------------------------------------------
2001-01-09 14:43:04 +00:00