forked from amazingfate/loongoffice
Change-Id: Idc8ecbb65c7c4c63950d8b249e0d0d91788d6d3b Reviewed-on: https://gerrit.libreoffice.org/11012 Reviewed-by: Björn Michaelsen <bjoern.michaelsen@canonical.com> Tested-by: Björn Michaelsen <bjoern.michaelsen@canonical.com>
922 lines
32 KiB
C++
922 lines
32 KiB
C++
/* -*- 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 .
|
|
*/
|
|
|
|
#include <vcl/waitobj.hxx>
|
|
#include <com/sun/star/util/URL.hpp>
|
|
#include <vcl/msgbox.hxx>
|
|
#include <vcl/stdtext.hxx>
|
|
#include <comphelper/types.hxx>
|
|
#include <comphelper/sequence.hxx>
|
|
#include "framectr.hxx"
|
|
#include "datman.hxx"
|
|
#include "bibresid.hxx"
|
|
#include "bib.hrc"
|
|
#include <toolkit/helper/vclunohelper.hxx>
|
|
#include "bibconfig.hxx"
|
|
#include <cppuhelper/implbase1.hxx>
|
|
#include <vcl/svapp.hxx>
|
|
#include "bibliography.hrc"
|
|
#include <comphelper/processfactory.hxx>
|
|
#include <com/sun/star/form/XConfirmDeleteListener.hpp>
|
|
#include <com/sun/star/form/runtime/XFormController.hpp>
|
|
#include <com/sun/star/beans/PropertyState.hpp>
|
|
#include <com/sun/star/beans/PropertyValue.hpp>
|
|
#include <com/sun/star/ui/dialogs/XExecutableDialog.hpp>
|
|
#include <com/sun/star/sdbcx/Privilege.hpp>
|
|
#include <com/sun/star/sdbc/XResultSetUpdate.hpp>
|
|
#include <com/sun/star/sdb/FilterDialog.hpp>
|
|
#include <com/sun/star/sdb/RowChangeAction.hpp>
|
|
#include <com/sun/star/frame/CommandGroup.hpp>
|
|
#include <com/sun/star/datatransfer/clipboard/XClipboard.hpp>
|
|
#include <cppuhelper/interfacecontainer.hxx>
|
|
#include <cppuhelper/supportsservice.hxx>
|
|
#include <sot/exchange.hxx>
|
|
#include <sot/formats.hxx>
|
|
#include <vcl/edit.hxx>
|
|
#include <osl/mutex.hxx>
|
|
|
|
#include <boost/unordered_map.hpp>
|
|
|
|
using namespace osl;
|
|
using namespace cppu;
|
|
using namespace com::sun::star::sdbc;
|
|
using namespace com::sun::star::frame;
|
|
using namespace com::sun::star::uno;
|
|
using namespace com::sun::star;
|
|
|
|
|
|
struct DispatchInfo
|
|
{
|
|
const char* pCommand;
|
|
sal_Int16 nGroupId;
|
|
bool bActiveConnection;
|
|
};
|
|
|
|
struct CacheDispatchInfo
|
|
{
|
|
sal_Int16 nGroupId;
|
|
bool bActiveConnection;
|
|
};
|
|
|
|
// Attention: commands must be sorted by command groups. Implementation is dependent
|
|
// on this!!
|
|
static const DispatchInfo SupportedCommandsArray[] =
|
|
{
|
|
{ ".uno:Undo" , frame::CommandGroup::EDIT , false },
|
|
{ ".uno:Cut" , frame::CommandGroup::EDIT , false },
|
|
{ ".uno:Copy" , frame::CommandGroup::EDIT , false },
|
|
{ ".uno:Paste" , frame::CommandGroup::EDIT , false },
|
|
{ ".uno:SelectAll" , frame::CommandGroup::EDIT , false },
|
|
{ ".uno:CloseDoc" , frame::CommandGroup::DOCUMENT , false },
|
|
{ ".uno:StatusBarVisible" , frame::CommandGroup::VIEW , false },
|
|
{ ".uno:AvailableToolbars" , frame::CommandGroup::VIEW , false },
|
|
{ ".uno:Bib/standardFilter" , frame::CommandGroup::DATA , true },
|
|
{ ".uno:Bib/DeleteRecord" , frame::CommandGroup::DATA , true },
|
|
{ ".uno:Bib/InsertRecord" , frame::CommandGroup::DATA , true },
|
|
{ ".uno:Bib/query" , frame::CommandGroup::DATA , true },
|
|
{ ".uno:Bib/autoFilter" , frame::CommandGroup::DATA , true },
|
|
{ ".uno:Bib/source" , frame::CommandGroup::DATA , true },
|
|
{ ".uno:Bib/removeFilter" , frame::CommandGroup::DATA , true },
|
|
{ ".uno:Bib/sdbsource" , frame::CommandGroup::DATA , true },
|
|
{ ".uno:Bib/Mapping" , frame::CommandGroup::DATA , true },
|
|
{ 0 , 0 , false }
|
|
};
|
|
|
|
typedef ::boost::unordered_map< OUString, CacheDispatchInfo, OUStringHash, ::std::equal_to< OUString > > CmdToInfoCache;
|
|
|
|
const CmdToInfoCache& GetCommandToInfoCache()
|
|
{
|
|
static bool bCacheInitialized = false;
|
|
static CmdToInfoCache aCmdToInfoCache;
|
|
|
|
if ( !bCacheInitialized )
|
|
{
|
|
::osl::MutexGuard aGuard( ::osl::Mutex::getGlobalMutex() );
|
|
if ( !bCacheInitialized )
|
|
{
|
|
sal_Int32 i( 0 );
|
|
while ( SupportedCommandsArray[i].pCommand != 0 )
|
|
{
|
|
OUString aCommand( OUString::createFromAscii( SupportedCommandsArray[i].pCommand ));
|
|
|
|
CacheDispatchInfo aDispatchInfo;
|
|
aDispatchInfo.nGroupId = SupportedCommandsArray[i].nGroupId;
|
|
aDispatchInfo.bActiveConnection = SupportedCommandsArray[i].bActiveConnection;
|
|
aCmdToInfoCache.insert( CmdToInfoCache::value_type( aCommand, aDispatchInfo ));
|
|
++i;
|
|
}
|
|
bCacheInitialized = true;
|
|
}
|
|
}
|
|
|
|
return aCmdToInfoCache;
|
|
}
|
|
|
|
|
|
class BibFrameCtrl_Impl : public cppu::WeakImplHelper1 < XFrameActionListener >
|
|
{
|
|
public:
|
|
Mutex aMutex;
|
|
OMultiTypeInterfaceContainerHelper aLC;
|
|
|
|
BibFrameController_Impl* pController;
|
|
|
|
BibFrameCtrl_Impl()
|
|
: aLC( aMutex )
|
|
, pController(0)
|
|
{}
|
|
|
|
virtual ~BibFrameCtrl_Impl();
|
|
|
|
virtual void SAL_CALL frameAction(const FrameActionEvent& aEvent) throw( RuntimeException, std::exception ) SAL_OVERRIDE;
|
|
virtual void SAL_CALL disposing( const lang::EventObject& Source ) throw (::com::sun::star::uno::RuntimeException, std::exception) SAL_OVERRIDE;
|
|
};
|
|
|
|
|
|
BibFrameCtrl_Impl::~BibFrameCtrl_Impl()
|
|
{
|
|
}
|
|
|
|
void BibFrameCtrl_Impl::frameAction(const FrameActionEvent& aEvent) throw( uno::RuntimeException, std::exception )
|
|
{
|
|
if ( pController && aEvent.Frame == pController->getFrame())
|
|
{
|
|
if(aEvent.Action == FrameAction_FRAME_ACTIVATED)
|
|
{
|
|
pController->activate();
|
|
}
|
|
else if(aEvent.Action == FrameAction_FRAME_DEACTIVATING)
|
|
{
|
|
pController->deactivate();
|
|
}
|
|
}
|
|
}
|
|
|
|
void BibFrameCtrl_Impl::disposing( const lang::EventObject& /*Source*/ )
|
|
throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
::SolarMutexGuard aGuard;
|
|
if ( pController )
|
|
pController->getFrame()->removeFrameActionListener( this );
|
|
}
|
|
|
|
BibFrameController_Impl::BibFrameController_Impl( const uno::Reference< awt::XWindow > & xComponent,
|
|
BibDataManager* pDataManager)
|
|
:xWindow( xComponent )
|
|
,m_xDatMan( pDataManager )
|
|
,pDatMan( pDataManager )
|
|
,pBibMod(NULL)
|
|
{
|
|
Window* pParent = VCLUnoHelper::GetWindow( xWindow );
|
|
pParent->SetUniqueId(UID_BIB_FRAME_WINDOW);
|
|
bDisposing=false;
|
|
bHierarchical=true;
|
|
pImp = new BibFrameCtrl_Impl;
|
|
pImp->pController = this;
|
|
pImp->acquire();
|
|
}
|
|
|
|
BibFrameController_Impl::~BibFrameController_Impl()
|
|
{
|
|
pImp->pController = NULL;
|
|
pImp->release();
|
|
delete pDatMan;
|
|
if(pBibMod)
|
|
CloseBibModul(pBibMod);
|
|
}
|
|
|
|
OUString SAL_CALL BibFrameController_Impl::getImplementationName() throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
return OUString("com.sun.star.comp.extensions.Bibliography");
|
|
}
|
|
|
|
sal_Bool SAL_CALL BibFrameController_Impl::supportsService( const OUString& sServiceName ) throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
return cppu::supportsService( this, sServiceName );
|
|
}
|
|
|
|
::com::sun::star::uno::Sequence< OUString > SAL_CALL BibFrameController_Impl::getSupportedServiceNames() throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
// return only top level services ...
|
|
// base services are included there and should be asked by uno-rtti.
|
|
::com::sun::star::uno::Sequence< OUString > lNames(1);
|
|
lNames[0] = "com.sun.star.frame.Bibliography";
|
|
return lNames;
|
|
}
|
|
|
|
void BibFrameController_Impl::attachFrame( const uno::Reference< XFrame > & xArg ) throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
xFrame = xArg;
|
|
xFrame->addFrameActionListener( pImp );
|
|
}
|
|
|
|
sal_Bool BibFrameController_Impl::attachModel( const uno::Reference< XModel > & /*xModel*/ ) throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
return sal_False;
|
|
}
|
|
|
|
sal_Bool BibFrameController_Impl::suspend( sal_Bool bSuspend ) throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
if ( bSuspend )
|
|
getFrame()->removeFrameActionListener( pImp );
|
|
else
|
|
getFrame()->addFrameActionListener( pImp );
|
|
return sal_True;
|
|
}
|
|
|
|
uno::Any BibFrameController_Impl::getViewData() throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
return uno::Any();
|
|
}
|
|
|
|
void BibFrameController_Impl::restoreViewData( const uno::Any& /*Value*/ ) throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
}
|
|
|
|
uno::Reference< XFrame > BibFrameController_Impl::getFrame() throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
return xFrame;
|
|
}
|
|
|
|
uno::Reference< XModel > BibFrameController_Impl::getModel() throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
return uno::Reference< XModel > ();
|
|
}
|
|
|
|
void BibFrameController_Impl::dispose() throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
bDisposing = true;
|
|
lang::EventObject aObject;
|
|
aObject.Source = (XController*)this;
|
|
pImp->aLC.disposeAndClear(aObject);
|
|
m_xDatMan = 0;
|
|
pDatMan = 0;
|
|
aStatusListeners.clear();
|
|
}
|
|
|
|
void BibFrameController_Impl::addEventListener( const uno::Reference< lang::XEventListener > & aListener ) throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
pImp->aLC.addInterface( cppu::UnoType<lang::XEventListener>::get(), aListener );
|
|
}
|
|
|
|
void BibFrameController_Impl::removeEventListener( const uno::Reference< lang::XEventListener > & aListener ) throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
pImp->aLC.removeInterface( cppu::UnoType<lang::XEventListener>::get(), aListener );
|
|
}
|
|
|
|
uno::Reference< frame::XDispatch > BibFrameController_Impl::queryDispatch( const util::URL& aURL, const OUString& /*aTarget*/, sal_Int32 /*nSearchFlags*/ ) throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
if ( !bDisposing )
|
|
{
|
|
const CmdToInfoCache& rCmdCache = GetCommandToInfoCache();
|
|
CmdToInfoCache::const_iterator pIter = rCmdCache.find( aURL.Complete );
|
|
if ( pIter != rCmdCache.end() )
|
|
{
|
|
if (( pDatMan->HasActiveConnection() ) ||
|
|
( !pIter->second.bActiveConnection ))
|
|
return (frame::XDispatch*) this;
|
|
}
|
|
}
|
|
|
|
return uno::Reference< frame::XDispatch > ();
|
|
}
|
|
|
|
uno::Sequence<uno::Reference< XDispatch > > BibFrameController_Impl::queryDispatches( const uno::Sequence<DispatchDescriptor>& aDescripts ) throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
uno::Sequence< uno::Reference< XDispatch > > aDispatches( aDescripts.getLength() );
|
|
for ( sal_Int32 i=0; i<aDescripts.getLength(); ++i )
|
|
aDispatches[i] = queryDispatch( aDescripts[i].FeatureURL, aDescripts[i].FrameName, aDescripts[i].SearchFlags );
|
|
return aDispatches;
|
|
}
|
|
|
|
uno::Sequence< ::sal_Int16 > SAL_CALL BibFrameController_Impl::getSupportedCommandGroups()
|
|
throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
uno::Sequence< ::sal_Int16 > aDispatchInfo( 4 );
|
|
|
|
aDispatchInfo[0] = frame::CommandGroup::EDIT;
|
|
aDispatchInfo[1] = frame::CommandGroup::DOCUMENT;
|
|
aDispatchInfo[2] = frame::CommandGroup::DATA;
|
|
aDispatchInfo[3] = frame::CommandGroup::VIEW;
|
|
|
|
return aDispatchInfo;
|
|
}
|
|
|
|
uno::Sequence< frame::DispatchInformation > SAL_CALL BibFrameController_Impl::getConfigurableDispatchInformation( ::sal_Int16 nCommandGroup )
|
|
throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
const CmdToInfoCache& rCmdCache = GetCommandToInfoCache();
|
|
|
|
bool bGroupFound( false );
|
|
frame::DispatchInformation aDispatchInfo;
|
|
std::list< frame::DispatchInformation > aDispatchInfoList;
|
|
|
|
if (( nCommandGroup == frame::CommandGroup::EDIT ) ||
|
|
( nCommandGroup == frame::CommandGroup::DOCUMENT ) ||
|
|
( nCommandGroup == frame::CommandGroup::DATA ) ||
|
|
( nCommandGroup == frame::CommandGroup::VIEW ))
|
|
{
|
|
CmdToInfoCache::const_iterator pIter = rCmdCache.begin();
|
|
while ( pIter != rCmdCache.end() )
|
|
{
|
|
if ( pIter->second.nGroupId == nCommandGroup )
|
|
{
|
|
bGroupFound = true;
|
|
aDispatchInfo.Command = pIter->first;
|
|
aDispatchInfo.GroupId = pIter->second.nGroupId;
|
|
aDispatchInfoList.push_back( aDispatchInfo );
|
|
}
|
|
else if ( bGroupFound )
|
|
break;
|
|
|
|
++pIter;
|
|
}
|
|
}
|
|
|
|
::com::sun::star::uno::Sequence< ::com::sun::star::frame::DispatchInformation > aSeq =
|
|
comphelper::containerToSequence< ::com::sun::star::frame::DispatchInformation, std::list< ::com::sun::star::frame::DispatchInformation > >( aDispatchInfoList );
|
|
|
|
return aSeq;
|
|
}
|
|
|
|
bool canInsertRecords(const Reference< beans::XPropertySet>& _rxCursorSet)
|
|
{
|
|
sal_Int32 nPriv = 0;
|
|
_rxCursorSet->getPropertyValue("Privileges") >>= nPriv;
|
|
return ((_rxCursorSet.is() && (nPriv & sdbcx::Privilege::INSERT) != 0));
|
|
}
|
|
|
|
bool BibFrameController_Impl::SaveModified(const Reference< form::runtime::XFormController>& xController)
|
|
{
|
|
if (!xController.is())
|
|
return false;
|
|
|
|
Reference< XResultSetUpdate> _xCursor = Reference< XResultSetUpdate>(xController->getModel(), UNO_QUERY);
|
|
|
|
if (!_xCursor.is())
|
|
return false;
|
|
|
|
Reference< beans::XPropertySet> _xSet = Reference< beans::XPropertySet>(_xCursor, UNO_QUERY);
|
|
if (!_xSet.is())
|
|
return false;
|
|
|
|
// need to save?
|
|
bool bIsNew = ::comphelper::getBOOL(_xSet->getPropertyValue("IsNew"));
|
|
bool bIsModified = ::comphelper::getBOOL(_xSet->getPropertyValue("IsModified"));
|
|
bool bResult = !bIsModified;
|
|
if (bIsModified)
|
|
{
|
|
try
|
|
{
|
|
if (bIsNew)
|
|
_xCursor->insertRow();
|
|
else
|
|
_xCursor->updateRow();
|
|
bResult = true;
|
|
}
|
|
catch(const Exception&)
|
|
{
|
|
OSL_FAIL("SaveModified: Exception occurred!");
|
|
}
|
|
}
|
|
return bResult;
|
|
}
|
|
|
|
static Window* lcl_GetFocusChild( Window* pParent )
|
|
{
|
|
sal_uInt16 nChildren = pParent->GetChildCount();
|
|
for( sal_uInt16 nChild = 0; nChild < nChildren; ++nChild)
|
|
{
|
|
Window* pChild = pParent->GetChild( nChild );
|
|
if(pChild->HasFocus())
|
|
return pChild;
|
|
Window* pSubChild = lcl_GetFocusChild( pChild );
|
|
if(pSubChild)
|
|
return pSubChild;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
//class XDispatch
|
|
void BibFrameController_Impl::dispatch(const util::URL& _rURL, const uno::Sequence< beans::PropertyValue >& aArgs)
|
|
throw (::com::sun::star::uno::RuntimeException,
|
|
std::exception)
|
|
{
|
|
if ( !bDisposing )
|
|
{
|
|
::SolarMutexGuard aGuard;
|
|
Window* pParent = VCLUnoHelper::GetWindow( xWindow );
|
|
WaitObject aWaitObject( pParent );
|
|
|
|
OUString aCommand( _rURL.Path);
|
|
if(aCommand == "Bib/Mapping")
|
|
{
|
|
pDatMan->CreateMappingDialog(pParent);
|
|
}
|
|
else if(aCommand == "Bib/source")
|
|
{
|
|
ChangeDataSource(aArgs);
|
|
}
|
|
else if(aCommand == "Bib/sdbsource")
|
|
{
|
|
OUString aURL = pDatMan->CreateDBChangeDialog(pParent);
|
|
if(!aURL.isEmpty())
|
|
{
|
|
try
|
|
{
|
|
uno::Sequence< beans::PropertyValue > aNewDataSource(2);
|
|
beans::PropertyValue* pProps = aNewDataSource.getArray();
|
|
pProps[0].Value <<= OUString();
|
|
pProps[1].Value <<= aURL;
|
|
ChangeDataSource(aNewDataSource);
|
|
}
|
|
catch(const Exception&)
|
|
{
|
|
OSL_FAIL("Exception catched while changing the data source");
|
|
}
|
|
}
|
|
}
|
|
else if(aCommand == "Bib/autoFilter")
|
|
{
|
|
sal_uInt16 nCount = aStatusListeners.size();
|
|
for ( sal_uInt16 n=0; n<nCount; n++ )
|
|
{
|
|
BibStatusDispatch *pObj = &aStatusListeners[n];
|
|
if ( pObj->aURL.Path == "Bib/removeFilter" )
|
|
{
|
|
FeatureStateEvent aEvent;
|
|
aEvent.FeatureURL = pObj->aURL;
|
|
aEvent.IsEnabled = sal_True;
|
|
aEvent.Requery = sal_False;
|
|
aEvent.Source = (XDispatch *) this;
|
|
pObj->xListener->statusChanged( aEvent );
|
|
//break; because there are more than one
|
|
}
|
|
}
|
|
|
|
const beans::PropertyValue* pPropertyValue = aArgs.getConstArray();
|
|
uno::Any aValue=pPropertyValue[0].Value;
|
|
OUString aQuery;
|
|
aValue >>= aQuery;
|
|
|
|
aValue=pPropertyValue[1].Value;
|
|
OUString aQueryField;
|
|
aValue >>= aQueryField;
|
|
BibConfig* pConfig = BibModul::GetConfig();
|
|
pConfig->setQueryField(aQueryField);
|
|
pDatMan->startQueryWith(aQuery);
|
|
}
|
|
else if(aCommand == "Bib/standardFilter")
|
|
{
|
|
try
|
|
{
|
|
uno::Reference< uno::XComponentContext > xContext = ::comphelper::getProcessComponentContext();
|
|
|
|
// create the dialog object
|
|
uno::Reference< ui::dialogs::XExecutableDialog > xDialog = sdb::FilterDialog::createWithQuery(xContext, pDatMan->getParser(),
|
|
Reference<sdbc::XRowSet>(pDatMan->getForm(), uno::UNO_QUERY_THROW), xWindow);
|
|
// execute it
|
|
if ( xDialog->execute( ) )
|
|
{
|
|
// the dialog has been executed successfully, and the filter on the query composer
|
|
// has been changed
|
|
OUString sNewFilter = pDatMan->getParser()->getFilter();
|
|
pDatMan->setFilter( sNewFilter );
|
|
}
|
|
}
|
|
catch( const uno::Exception& )
|
|
{
|
|
OSL_FAIL( "BibFrameController_Impl::dispatch: caught an exception!" );
|
|
}
|
|
|
|
sal_uInt16 nCount = aStatusListeners.size();
|
|
for ( sal_uInt16 n=0; n<nCount; n++ )
|
|
{
|
|
BibStatusDispatch *pObj = &aStatusListeners[n];
|
|
if ( pObj->aURL.Path == "Bib/removeFilter" && pDatMan->getParser().is())
|
|
{
|
|
FeatureStateEvent aEvent;
|
|
aEvent.FeatureURL = pObj->aURL;
|
|
aEvent.IsEnabled = !pDatMan->getParser()->getFilter().isEmpty();
|
|
aEvent.Requery = sal_False;
|
|
aEvent.Source = (XDispatch *) this;
|
|
pObj->xListener->statusChanged( aEvent );
|
|
}
|
|
}
|
|
}
|
|
else if(aCommand == "Bib/removeFilter")
|
|
{
|
|
RemoveFilter();
|
|
}
|
|
else if( _rURL.Complete == "slot:5503" || aCommand == "CloseDoc" )
|
|
{
|
|
Application::PostUserEvent( STATIC_LINK( this, BibFrameController_Impl,
|
|
DisposeHdl ), 0 );
|
|
|
|
}
|
|
else if(aCommand == "Bib/InsertRecord")
|
|
{
|
|
Reference<form::runtime::XFormController > xFormCtrl = pDatMan->GetFormController();
|
|
if(SaveModified(xFormCtrl))
|
|
{
|
|
try
|
|
{
|
|
Reference< sdbc::XResultSet > xCursor( pDatMan->getForm(), UNO_QUERY );
|
|
xCursor->last();
|
|
|
|
Reference< XResultSetUpdate > xUpdateCursor( pDatMan->getForm(), UNO_QUERY );
|
|
xUpdateCursor->moveToInsertRow();
|
|
}
|
|
catch(const Exception&)
|
|
{
|
|
OSL_FAIL("Exception in last() or moveToInsertRow()");
|
|
}
|
|
}
|
|
}
|
|
else if(aCommand == "Bib/DeleteRecord")
|
|
{
|
|
Reference< ::com::sun::star::sdbc::XResultSet > xCursor(pDatMan->getForm(), UNO_QUERY);
|
|
Reference< XResultSetUpdate > xUpdateCursor(xCursor, UNO_QUERY);
|
|
Reference< beans::XPropertySet > xSet(pDatMan->getForm(), UNO_QUERY);
|
|
bool bIsNew = ::comphelper::getBOOL(xSet->getPropertyValue("IsNew"));
|
|
if(!bIsNew)
|
|
{
|
|
sal_uInt32 nCount = 0;
|
|
xSet->getPropertyValue("RowCount") >>= nCount;
|
|
// determine next position
|
|
bool bSuccess = false;
|
|
bool bLeft = false;
|
|
bool bRight = false;
|
|
try
|
|
{
|
|
bLeft = xCursor->isLast() && nCount > 1;
|
|
bRight= !xCursor->isLast();
|
|
// ask for confirmation
|
|
Reference< frame::XController > xCtrl = pImp->pController;
|
|
Reference< form::XConfirmDeleteListener > xConfirm(pDatMan->GetFormController(),UNO_QUERY);
|
|
if (xConfirm.is())
|
|
{
|
|
sdb::RowChangeEvent aEvent;
|
|
aEvent.Source = Reference< XInterface > (xCursor, UNO_QUERY);
|
|
aEvent.Action = sdb::RowChangeAction::DELETE;
|
|
aEvent.Rows = 1;
|
|
bSuccess = xConfirm->confirmDelete(aEvent);
|
|
}
|
|
|
|
// delete it
|
|
if (bSuccess)
|
|
xUpdateCursor->deleteRow();
|
|
}
|
|
catch(const Exception&)
|
|
{
|
|
bSuccess = false;
|
|
}
|
|
if (bSuccess)
|
|
{
|
|
if (bLeft || bRight)
|
|
xCursor->relative(bRight ? 1 : -1);
|
|
else
|
|
{
|
|
bool bCanInsert = canInsertRecords(xSet);
|
|
// can another entry be inserted?
|
|
try
|
|
{
|
|
if (bCanInsert)
|
|
xUpdateCursor->moveToInsertRow();
|
|
else
|
|
// move data entry to reset state
|
|
xCursor->first();
|
|
}
|
|
catch(const Exception&)
|
|
{
|
|
OSL_FAIL("DeleteRecord: exception caught!");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if(aCommand == "Cut")
|
|
{
|
|
Window* pChild = lcl_GetFocusChild( VCLUnoHelper::GetWindow( xWindow ) );
|
|
if(pChild)
|
|
{
|
|
KeyEvent aEvent( 0, KEYFUNC_CUT );
|
|
pChild->KeyInput( aEvent );
|
|
}
|
|
}
|
|
else if(aCommand == "Copy")
|
|
{
|
|
Window* pChild = lcl_GetFocusChild( VCLUnoHelper::GetWindow( xWindow ) );
|
|
if(pChild)
|
|
{
|
|
KeyEvent aEvent( 0, KEYFUNC_COPY );
|
|
pChild->KeyInput( aEvent );
|
|
}
|
|
}
|
|
else if(aCommand == "Paste")
|
|
{
|
|
Window* pChild = lcl_GetFocusChild( VCLUnoHelper::GetWindow( xWindow ) );
|
|
if(pChild)
|
|
{
|
|
KeyEvent aEvent( 0, KEYFUNC_PASTE );
|
|
pChild->KeyInput( aEvent );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
IMPL_STATIC_LINK( BibFrameController_Impl, DisposeHdl, void*, EMPTYARG )
|
|
{
|
|
pThis->xFrame->dispose();
|
|
return 0;
|
|
};
|
|
|
|
void BibFrameController_Impl::addStatusListener(
|
|
const uno::Reference< frame::XStatusListener > & aListener,
|
|
const util::URL& aURL)
|
|
throw (::com::sun::star::uno::RuntimeException,
|
|
std::exception)
|
|
{
|
|
BibConfig* pConfig = BibModul::GetConfig();
|
|
// create a new Reference and insert into listener array
|
|
aStatusListeners.push_back( new BibStatusDispatch( aURL, aListener ) );
|
|
|
|
// send first status synchronously
|
|
FeatureStateEvent aEvent;
|
|
aEvent.FeatureURL = aURL;
|
|
aEvent.Requery = sal_False;
|
|
aEvent.Source = (XDispatch *) this;
|
|
if ( aURL.Path == "StatusBarVisible" )
|
|
{
|
|
aEvent.IsEnabled = sal_False;
|
|
aEvent.State <<= false;
|
|
}
|
|
else if ( aURL.Path == "Bib/hierarchical" )
|
|
{
|
|
aEvent.IsEnabled = sal_True;
|
|
const char* pHier = bHierarchical? "" : "*" ;
|
|
aEvent.State <<= OUString::createFromAscii(pHier);
|
|
}
|
|
else if(aURL.Path == "Bib/MenuFilter")
|
|
{
|
|
aEvent.IsEnabled = sal_True;
|
|
aEvent.FeatureDescriptor=pDatMan->getQueryField();
|
|
|
|
uno::Sequence<OUString> aStringSeq=pDatMan->getQueryFields();
|
|
aEvent.State.setValue(&aStringSeq,::getCppuType((uno::Sequence<OUString>*)0));
|
|
|
|
}
|
|
else if ( aURL.Path == "Bib/source")
|
|
{
|
|
aEvent.IsEnabled = sal_True;
|
|
aEvent.FeatureDescriptor=pDatMan->getActiveDataTable();
|
|
|
|
uno::Sequence<OUString> aStringSeq=pDatMan->getDataSources();
|
|
aEvent.State.setValue(&aStringSeq,::getCppuType((uno::Sequence<OUString>*)0));
|
|
}
|
|
else if( aURL.Path == "Bib/sdbsource" ||
|
|
aURL.Path == "Bib/Mapping" ||
|
|
aURL.Path == "Bib/autoFilter" ||
|
|
aURL.Path == "Bib/standardFilter" )
|
|
{
|
|
aEvent.IsEnabled = sal_True;
|
|
}
|
|
else if(aURL.Path == "Bib/query")
|
|
{
|
|
aEvent.IsEnabled = sal_True;
|
|
aEvent.State <<= pConfig->getQueryText();
|
|
}
|
|
else if (aURL.Path == "Bib/removeFilter" )
|
|
{
|
|
OUString aFilterStr=pDatMan->getFilter();
|
|
aEvent.IsEnabled = !aFilterStr.isEmpty();
|
|
}
|
|
else if(aURL.Path == "Cut")
|
|
{
|
|
Window* pChild = lcl_GetFocusChild( VCLUnoHelper::GetWindow( xWindow ) );
|
|
Edit* pEdit = dynamic_cast<Edit*>( pChild );
|
|
if( pEdit )
|
|
aEvent.IsEnabled = !pEdit->IsReadOnly() && pEdit->GetSelection().Len();
|
|
}
|
|
if(aURL.Path == "Copy")
|
|
{
|
|
Window* pChild = lcl_GetFocusChild( VCLUnoHelper::GetWindow( xWindow ) );
|
|
Edit* pEdit = dynamic_cast<Edit*>( pChild );
|
|
if( pEdit )
|
|
aEvent.IsEnabled = pEdit->GetSelection().Len() > 0;
|
|
}
|
|
else if(aURL.Path == "Paste" )
|
|
{
|
|
aEvent.IsEnabled = sal_False;
|
|
Window* pChild = lcl_GetFocusChild( VCLUnoHelper::GetWindow( xWindow ) );
|
|
if(pChild)
|
|
{
|
|
uno::Reference< datatransfer::clipboard::XClipboard > xClip = pChild->GetClipboard();
|
|
if(xClip.is())
|
|
{
|
|
uno::Reference< datatransfer::XTransferable > xDataObj;
|
|
const sal_uInt32 nRef = Application::ReleaseSolarMutex();
|
|
try
|
|
{
|
|
xDataObj = xClip->getContents();
|
|
}
|
|
catch( const uno::Exception& )
|
|
{
|
|
}
|
|
Application::AcquireSolarMutex( nRef );
|
|
|
|
if ( xDataObj.is() )
|
|
{
|
|
datatransfer::DataFlavor aFlavor;
|
|
SotExchange::GetFormatDataFlavor( SOT_FORMAT_STRING, aFlavor );
|
|
try
|
|
{
|
|
uno::Any aData = xDataObj->getTransferData( aFlavor );
|
|
OUString aText;
|
|
aData >>= aText;
|
|
aEvent.IsEnabled = !aText.isEmpty();
|
|
}
|
|
catch( const uno::Exception& )
|
|
{
|
|
}
|
|
}
|
|
}
|
|
uno::Reference< datatransfer::XTransferable > xContents = xClip->getContents( );
|
|
}
|
|
}
|
|
else if(aURL.Path == "Bib/DeleteRecord")
|
|
{
|
|
Reference< ::com::sun::star::sdbc::XResultSet > xCursor(pDatMan->getForm(), UNO_QUERY);
|
|
Reference< XResultSetUpdate > xUpdateCursor(xCursor, UNO_QUERY);
|
|
Reference< beans::XPropertySet > xSet(pDatMan->getForm(), UNO_QUERY);
|
|
bool bIsNew = ::comphelper::getBOOL(xSet->getPropertyValue("IsNew"));
|
|
if(!bIsNew)
|
|
{
|
|
sal_uInt32 nCount = 0;
|
|
xSet->getPropertyValue("RowCount") >>= nCount;
|
|
aEvent.IsEnabled = nCount > 0;
|
|
}
|
|
}
|
|
else if (aURL.Path == "Bib/InsertRecord")
|
|
{
|
|
Reference< beans::XPropertySet > xSet(pDatMan->getForm(), UNO_QUERY);
|
|
aEvent.IsEnabled = canInsertRecords(xSet);
|
|
}
|
|
aListener->statusChanged( aEvent );
|
|
}
|
|
|
|
void BibFrameController_Impl::removeStatusListener(
|
|
const uno::Reference< frame::XStatusListener > & aObject, const util::URL& aURL)
|
|
throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
// search listener array for given listener
|
|
// for checking equality always "cast" to XInterface
|
|
if ( !bDisposing )
|
|
{
|
|
sal_uInt16 nCount = aStatusListeners.size();
|
|
for ( sal_uInt16 n=0; n<nCount; n++ )
|
|
{
|
|
BibStatusDispatch *pObj = &aStatusListeners[n];
|
|
bool bFlag=pObj->xListener.is();
|
|
if (!bFlag || (pObj->xListener == aObject &&
|
|
( aURL.Complete.isEmpty() || pObj->aURL.Path == aURL.Path )))
|
|
{
|
|
aStatusListeners.erase( aStatusListeners.begin() + n );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void BibFrameController_Impl::RemoveFilter()
|
|
{
|
|
OUString aQuery;
|
|
pDatMan->startQueryWith(aQuery);
|
|
|
|
sal_uInt16 nCount = aStatusListeners.size();
|
|
|
|
bool bRemoveFilter=false;
|
|
bool bQueryText=false;
|
|
|
|
for ( sal_uInt16 n=0; n<nCount; n++ )
|
|
{
|
|
BibStatusDispatch *pObj = &aStatusListeners[n];
|
|
if ( pObj->aURL.Path == "Bib/removeFilter" )
|
|
{
|
|
FeatureStateEvent aEvent;
|
|
aEvent.FeatureURL = pObj->aURL;
|
|
aEvent.IsEnabled = sal_False;
|
|
aEvent.Requery = sal_False;
|
|
aEvent.Source = (XDispatch *) this;
|
|
pObj->xListener->statusChanged( aEvent );
|
|
bRemoveFilter=true;
|
|
}
|
|
else if(pObj->aURL.Path == "Bib/query")
|
|
{
|
|
FeatureStateEvent aEvent;
|
|
aEvent.FeatureURL = pObj->aURL;
|
|
aEvent.IsEnabled = sal_True;
|
|
aEvent.Requery = sal_False;
|
|
aEvent.Source = (XDispatch *) this;
|
|
aEvent.State <<= aQuery;
|
|
pObj->xListener->statusChanged( aEvent );
|
|
bQueryText=true;
|
|
}
|
|
|
|
if(bRemoveFilter && bQueryText)
|
|
break;
|
|
|
|
}
|
|
}
|
|
|
|
void BibFrameController_Impl::ChangeDataSource(const uno::Sequence< beans::PropertyValue >& aArgs)
|
|
{
|
|
const beans::PropertyValue* pPropertyValue = aArgs.getConstArray();
|
|
uno::Any aValue=pPropertyValue[0].Value;
|
|
OUString aDBTableName;
|
|
aValue >>= aDBTableName;
|
|
|
|
|
|
if(aArgs.getLength() > 1)
|
|
{
|
|
uno::Any aDB = pPropertyValue[1].Value;
|
|
OUString aURL;
|
|
aDB >>= aURL;
|
|
pDatMan->setActiveDataSource(aURL);
|
|
aDBTableName = pDatMan->getActiveDataTable();
|
|
}
|
|
else
|
|
{
|
|
m_xDatMan->unload();
|
|
pDatMan->setActiveDataTable(aDBTableName);
|
|
pDatMan->updateGridModel();
|
|
m_xDatMan->load();
|
|
}
|
|
|
|
|
|
sal_uInt16 nCount = aStatusListeners.size();
|
|
|
|
bool bMenuFilter=false;
|
|
bool bQueryText=false;
|
|
for ( sal_uInt16 n=0; n<nCount; n++ )
|
|
{
|
|
BibStatusDispatch *pObj = &aStatusListeners[n];
|
|
if (pObj->aURL.Path == "Bib/MenuFilter")
|
|
{
|
|
FeatureStateEvent aEvent;
|
|
aEvent.FeatureURL = pObj->aURL;
|
|
aEvent.IsEnabled = sal_True;
|
|
aEvent.Requery = sal_False;
|
|
aEvent.Source = (XDispatch *) this;
|
|
aEvent.FeatureDescriptor=pDatMan->getQueryField();
|
|
|
|
uno::Sequence<OUString> aStringSeq=pDatMan->getQueryFields();
|
|
aEvent.State = makeAny( aStringSeq );
|
|
|
|
pObj->xListener->statusChanged( aEvent );
|
|
bMenuFilter=true;
|
|
}
|
|
else if (pObj->aURL.Path == "Bib/query")
|
|
{
|
|
FeatureStateEvent aEvent;
|
|
aEvent.FeatureURL = pObj->aURL;
|
|
aEvent.IsEnabled = sal_True;
|
|
aEvent.Requery = sal_False;
|
|
aEvent.Source = (XDispatch *) this;
|
|
BibConfig* pConfig = BibModul::GetConfig();
|
|
aEvent.State <<= pConfig->getQueryText();
|
|
pObj->xListener->statusChanged( aEvent );
|
|
bQueryText=true;
|
|
}
|
|
|
|
if (bMenuFilter && bQueryText)
|
|
break;
|
|
|
|
}
|
|
}
|
|
|
|
void BibFrameController_Impl::activate()
|
|
{
|
|
}
|
|
void BibFrameController_Impl::deactivate()
|
|
{
|
|
}
|
|
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|