/************************************************************************* * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: AppControllerDnD.cxx,v $ * * $Revision: 1.11 $ * * last change: $Author: rt $ $Date: 2005-09-08 14:19:23 $ * * The Contents of this file are made available subject to * the terms of GNU Lesser General Public License Version 2.1. * * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2005 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 * ************************************************************************/ #ifndef DBAUI_APPCONTROLLER_HXX #include "AppController.hxx" #endif #ifndef _COMPHELPER_SEQUENCE_HXX_ #include #endif #ifndef DBACCESS_SHARED_DBUSTRINGS_HRC #include "dbustrings.hrc" #endif #ifndef _COM_SUN_STAR_SDB_XSINGLESELECTQUERYCOMPOSER_HPP_ #include #endif #ifndef _COM_SUN_STAR_CONTAINER_XNAMECONTAINER_HPP_ #include #endif #ifndef _COM_SUN_STAR_UNO_XNAMINGSERVICE_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDBC_XDATASOURCE_HPP_ #include #endif #ifndef _COM_SUN_STAR_FRAME_XSTORABLE_HPP_ #include #endif #ifndef _COM_SUN_STAR_CONTAINER_XCHILD_HPP_ #include #endif #ifndef _COM_SUN_STAR_UTIL_XMODIFIABLE_HPP_ #include #endif #ifndef _COM_SUN_STAR_CONTAINER_XHIERARCHICALNAMECONTAINER_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDBC_DATATYPE_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDB_COMMANDTYPE_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDB_XBOOKMARKSSUPPLIER_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDB_SQLCONTEXT_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDBCX_XTABLESSUPPLIER_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDBCX_XVIEWSSUPPLIER_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDB_XQUERYDEFINITIONSSUPPLIER_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDBCX_XDROP_HPP_ #include #endif #ifndef _TOOLS_DEBUG_HXX #include #endif #ifndef _URLOBJ_HXX #include #endif #ifndef _UNOTOOLS_UCBHELPER_HXX #include #endif #ifndef DBAUI_DLGSAVE_HXX #include "dlgsave.hxx" #endif #ifndef _COMPHELPER_TYPES_HXX_ #include #endif #ifndef _SV_MSGBOX_HXX #include #endif #ifndef _CPPUHELPER_TYPEPROVIDER_HXX_ #include #endif #ifndef _DBHELPER_DBEXCEPTION_HXX_ #include #endif #ifndef _SV_WAITOBJ_HXX #include #endif #ifndef _RTL_USTRBUF_HXX_ #include #endif #ifndef DBAUI_APPVIEW_HXX #include "AppView.hxx" #endif #ifndef _SVX_DATACCESSDESCRIPTOR_HXX_ #include #endif #ifndef SVX_DBAOBJECTEX_HXX #include #endif #ifndef DBACCESS_UI_BROWSER_ID_HXX #include "browserids.hxx" #endif #ifndef _DBAU_REGHELPER_HXX_ #include "dbu_reghelper.hxx" #endif #ifndef _DBU_APP_HRC_ #include "dbu_app.hrc" #endif #ifndef _SV_MENU_HXX #include #endif #ifndef _COMPHELPER_UNO3_HXX_ #include #endif #ifndef _DBAUI_QUERYDESIGNACCESS_HXX_ #include "querydesignaccess.hxx" #endif #ifndef _SV_SVAPP_HXX //autogen #include #endif #ifndef _SVLBOXITM_HXX #include #endif #ifndef _DBAUI_LISTVIEWITEMS_HXX_ #include "listviewitems.hxx" #endif #ifndef DBAUI_APPDETAILVIEW_HXX #include "AppDetailView.hxx" #endif #ifndef _DBAUI_LINKEDDOCUMENTS_HXX_ #include "linkeddocuments.hxx" #endif #ifndef _SV_LSTBOX_HXX #include #endif #ifndef _DBHELPER_DBEXCEPTION_HXX_ #include #endif #ifndef _CONNECTIVITY_DBTOOLS_HXX_ #include #endif #ifndef _DBAUI_SQLMESSAGE_HXX_ #include "sqlmessage.hxx" #endif #ifndef _STRING_HXX #include #endif #ifndef DBAUI_DBEXCHANGE_HXX #include "dbexchange.hxx" #endif #ifndef DBAUI_TOOLS_HXX #include "UITools.hxx" #endif #include #ifndef _SVTREEBOX_HXX #include #endif #ifndef _COM_SUN_STAR_SDB_XREPORTDOCUMENTSSUPPLIER_HPP_ #include #endif #ifndef _COM_SUN_STAR_SDB_XFORMDOCUMENTSSUPPLIER_HPP_ #include #endif #ifndef _FILEDLGHELPER_HXX #include #endif #ifndef INCLUDED_SVTOOLS_PATHOPTIONS_HXX #include #endif #ifndef _SFX_DOCFILT_HACK_HXX #include #endif #ifndef _SVT_FILEVIEW_HXX #include #endif //........................................................................ namespace dbaui { //........................................................................ using namespace ::dbtools; using namespace ::svx; using namespace ::svtools; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::task; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::container; using namespace ::com::sun::star::sdb; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::sdbcx; using namespace ::com::sun::star::frame; using namespace ::com::sun::star::ucb; using namespace ::com::sun::star::util; // ----------------------------------------------------------------------------- void OApplicationController::deleteTables(const ::std::vector< ::rtl::OUString>& _rList) { Reference xConnection; ensureConnection(xConnection); Reference xSup(xConnection,UNO_QUERY); OSL_ENSURE(xSup.is(),"OApplicationController::deleteTable: no XTablesSuppier!"); if ( xSup.is() ) { Reference xTables = xSup->getTables(); Reference xDrop(xTables,UNO_QUERY); if ( xDrop.is() ) { bool bConfirm = true; ::std::vector< ::rtl::OUString>::const_iterator aEnd = _rList.end(); for (::std::vector< ::rtl::OUString>::const_iterator aIter = _rList.begin(); aIter != aEnd; ++aIter) { ::rtl::OUString sTableName = *aIter; sal_Int32 nResult = RET_YES; if ( bConfirm ) nResult = ::dbaui::askForUserAction(getView(),STR_TITLE_CONFIRM_DELETION ,STR_QUERY_DELETE_TABLE,_rList.size() > 1 && (aIter+1) != _rList.end(),sTableName); if ( (RET_YES == nResult) || (RET_ALL == nResult) ) { SQLExceptionInfo aErrorInfo; try { if ( xTables->hasByName(sTableName) ) xDrop->dropByName(sTableName); else {// could be a view Reference xViewsSup(xConnection,UNO_QUERY); Reference xViews; if ( xViewsSup.is() ) { xViews = xViewsSup->getViews(); if ( xViews.is() && xViews->hasByName(sTableName) ) { xDrop.set(xViews,UNO_QUERY); if ( xDrop.is() ) xDrop->dropByName(sTableName); } } } } catch(SQLContext& e) { aErrorInfo = e; } catch(SQLWarning& e) { aErrorInfo = e; } catch(SQLException& e) { aErrorInfo = e; } catch(WrappedTargetException& e) { SQLException aSql; if(e.TargetException >>= aSql) aErrorInfo = aSql; else OSL_ENSURE(sal_False, "OApplicationController::implDropTable: something strange happended!"); } catch(Exception&) { DBG_ERROR("OApplicationController::implDropTable: suspicious exception caught!"); } if ( aErrorInfo.isValid() ) showError(aErrorInfo); if ( RET_ALL == nResult ) bConfirm = false; } else break; } } else { String sMessage(ModuleRes(STR_MISSING_TABLES_XDROP)); ErrorBox aError(getView(), WB_OK, sMessage); aError.Execute(); } } } // ----------------------------------------------------------------------------- void OApplicationController::deleteObjects( ElementType _eType, const ::std::vector< ::rtl::OUString>& _rList, bool _bConfirm ) { deleteObjects( Reference< XNameContainer >( getElements( _eType ), UNO_QUERY ), _rList, _bConfirm ); } // ----------------------------------------------------------------------------- void OApplicationController::deleteObjects( const Reference< XNameContainer>& _rxNames, const ::std::vector< ::rtl::OUString>& _rList, bool _bConfirm ) { Reference< XHierarchicalNameContainer > xHierarchyName( _rxNames, UNO_QUERY ); if ( _rxNames.is() ) { bool bConfirm = true; ByteString sDialogPosition; svtools::QueryDeleteResult_Impl eResult = _bConfirm ? svtools::QUERYDELETE_YES : svtools::QUERYDELETE_ALL; // The list of elements to delete is allowed to contain related elements: A given element may // be the ancestor or child of another element from the list. // We want to ensure that ancestors get deleted first, so we normalize the list in this respect. // #i33353# - 2004-09-27 - fs@openoffice.org ::std::set< ::rtl::OUString > aDeleteNames; // Note that this implicitly uses ::std::less< ::rtl::OUString > a comparison operation, which // results in lexicographical order, which is exactly what we need, because "foo" is *before* // any "foo/bar" in this order. ::std::copy( _rList.begin(), _rList.end(), ::std::insert_iterator< ::std::set< ::rtl::OUString > >( aDeleteNames, aDeleteNames.begin() ) ); ::std::set< ::rtl::OUString >::size_type nCount = aDeleteNames.size(); for ( ::std::set< ::rtl::OUString >::size_type nObjectsLeft = nCount; !aDeleteNames.empty(); ) { ::std::set< ::rtl::OUString >::iterator aThisRound = aDeleteNames.begin(); if ( eResult != svtools::QUERYDELETE_ALL ) { svtools::QueryDeleteDlg_Impl aDlg( getView(), *aThisRound ); if ( sDialogPosition.Len() ) aDlg.SetWindowState( sDialogPosition ); if ( nObjectsLeft > 1 ) aDlg.EnableAllButton(); if ( aDlg.Execute() == RET_OK ) eResult = aDlg.GetResult(); else return; sDialogPosition = aDlg.GetWindowState( ); } bool bSuccess = false; if ( ( eResult == svtools::QUERYDELETE_ALL ) || ( eResult == svtools::QUERYDELETE_YES ) ) { try { if ( xHierarchyName.is() ) xHierarchyName->removeByHierarchicalName( *aThisRound ); else _rxNames->removeByName( *aThisRound ); bSuccess = true; // now that we removed the element, care for all it's child elements // which may also be a part of the list // #i33353# - 2004-09-27 - fs@openoffice.org sal_Int32 nLastCharPos = aThisRound->getLength() - 1; OSL_ENSURE( nLastCharPos >= 0, "OApplicationController::deleteObjects: empty name?" ); ::rtl::OUStringBuffer sSmallestSiblingName( *aThisRound ); sSmallestSiblingName.append( (sal_Unicode)( '/' + 1) ); ::std::set< ::rtl::OUString >::iterator aUpperChildrenBound = aDeleteNames.lower_bound( sSmallestSiblingName.makeStringAndClear() ); for ( ::std::set< ::rtl::OUString >::iterator aObsolete = aThisRound; aObsolete != aUpperChildrenBound; ) { #if OSL_DEBUG_LEVEL > 0 ::rtl::OUString sObsoleteName = *aObsolete; #endif ::std::set< ::rtl::OUString >::iterator aNextObsolete = aObsolete; ++aNextObsolete; aDeleteNames.erase( aObsolete ); --nObjectsLeft; aObsolete = aNextObsolete; } } catch(SQLException& e) { showError( SQLExceptionInfo(e) ); } catch(WrappedTargetException& e) { SQLException aSql; if(e.TargetException >>= aSql) showError( SQLExceptionInfo( aSql ) ); else OSL_ENSURE( sal_False, "OApplicationController::deleteObjects: something strange happended!" ); } catch(Exception&) { DBG_ERROR( "OApplicationController::deleteObjects: caught a generic exception!" ); } } if ( !bSuccess ) { // okay, this object could not be deleted (or the user did not want to delete it), // but continue with the rest aDeleteNames.erase( aThisRound ); --nObjectsLeft; } } } } // ----------------------------------------------------------------------------- void OApplicationController::deleteEntries() { ::vos::OGuard aSolarGuard(Application::GetSolarMutex()); ::osl::MutexGuard aGuard(m_aMutex); if ( getContainer() ) { ::std::vector< ::rtl::OUString> aList; getSelectionElementNames(aList); ElementType eType = getContainer()->getElementType(); switch(eType) { case E_TABLE: deleteTables(aList); break; case E_QUERY: deleteObjects( E_QUERY, aList, true ); break; case E_FORM: deleteObjects( E_FORM, aList, true ); break; case E_REPORT: deleteObjects( E_REPORT, aList, true ); break; } } } // ----------------------------------------------------------------------------- Reference OApplicationController::getActiveConnection() const { Reference xConnection; if ( getContainer() ) { ::rtl::OUString sDataSourceName = getDatabaseName(); TDataSourceConnections::const_iterator aFind = m_aDataSourceConnections.find(sDataSourceName); if ( aFind != m_aDataSourceConnections.end() ) xConnection = aFind->second; } return xConnection; } // ----------------------------------------------------------------------------- bool OApplicationController::ensureConnection(Reference& _xConnection,sal_Bool _bCreate) { ::vos::OGuard aSolarGuard(Application::GetSolarMutex()); ::osl::MutexGuard aGuard(m_aMutex); ::rtl::OUString sDataSourceName = getDatabaseName(); TDataSourceConnections::iterator aFind = m_aDataSourceConnections.find(sDataSourceName); if ( aFind == m_aDataSourceConnections.end() ) aFind = m_aDataSourceConnections.insert(TDataSourceConnections::value_type(sDataSourceName,Reference())).first; if ( !aFind->second.is() && _bCreate ) { WaitObject aWO(getView()); String sConnectingContext( ModuleRes( STR_COULDNOTCONNECT_DATASOURCE ) ); sConnectingContext.SearchAndReplaceAscii("$name$", getStrippedDatabaseName()); aFind->second = connect(sDataSourceName, sConnectingContext, rtl::OUString(), sal_True); // otherwise we got a loop when connecting to db throws an error // if ( !aFind->second.is() ) // getContainer()->clearSelection(); } _xConnection = aFind->second; return _xConnection.is(); } // ----------------------------------------------------------------------------- sal_Bool OApplicationController::isDataSourceReadOnly() const { Reference xStore(m_xModel,UNO_QUERY); return !xStore.is() || xStore->isReadonly(); } // ----------------------------------------------------------------------------- sal_Bool OApplicationController::isConnectionReadOnly() const { sal_Bool bIsConnectionReadOnly = sal_True; Reference xConnection = getActiveConnection(); if ( xConnection.is() ) { try { bIsConnectionReadOnly = xConnection->getMetaData()->isReadOnly(); } catch(SQLException&) { } } // TODO check configuration return bIsConnectionReadOnly; } // ----------------------------------------------------------------------------- sal_Bool OApplicationController::isRelationDesignAllowed() const { return sal_True; } // ----------------------------------------------------------------------------- Reference< XNameAccess > OApplicationController::getElements(ElementType _eType) { OSL_ENSURE(getContainer(),"View is NULL! -> GPF"); // TODO get a list for all object ::rtl::OUString sDataSource = getDatabaseName(); Reference< XNameAccess > xElements; try { switch ( _eType ) { case E_REPORT: // TODO: seperate handling of forms and reports { Reference< XReportDocumentsSupplier > xSupp(m_xModel,UNO_QUERY); OSL_ENSURE(xSupp.is(),"Data source doesn't return a XReportDocumentsSupplier -> GPF"); if ( xSupp.is() ) xElements = xSupp->getReportDocuments(); } break; case E_FORM: { Reference< XFormDocumentsSupplier > xSupp(m_xModel,UNO_QUERY); OSL_ENSURE(xSupp.is(),"Data source doesn't return a XFormDocumentsSupplier -> GPF"); if ( xSupp.is() ) xElements = xSupp->getFormDocuments(); } break; case E_QUERY: { xElements.set(getQueryDefintions(),UNO_QUERY); } break; case E_TABLE: { Reference xConnection; ensureConnection(xConnection,sal_False); if ( xConnection.is() ) { Reference xSup(xConnection,UNO_QUERY); OSL_ENSURE(xSup.is(),"OApplicationController::getElements: no XTablesSuppier!"); if ( xSup.is() ) xElements = xSup->getTables(); } } break; } } catch(const Exception&) { OSL_ENSURE(0,"Could not get element container!"); } return xElements; } // ----------------------------------------------------------------------------- void OApplicationController::getElements(ElementType _eType,::std::vector< ::rtl::OUString>& _rList) { OSL_ENSURE(getContainer(),"View is NULL! -> GPF"); // TODO get a list for all object Reference< XNameAccess > xElements = getElements(_eType); if ( xElements.is() ) { Sequence< ::rtl::OUString> aSeq = xElements->getElementNames(); _rList.reserve(aSeq.getLength()); const ::rtl::OUString* pBegin = aSeq.getConstArray(); const ::rtl::OUString* pEnd = pBegin + aSeq.getLength(); ::std::copy(pBegin,pEnd,::std::back_inserter(_rList)); } } // ----------------------------------------------------------------------------- void OApplicationController::impl_initialize( const Sequence< Any >& aArguments ) { sal_Bool bInteractive = sal_False; const Any* pIter = aArguments.getConstArray(); const Any* pEnd = pIter + aArguments.getLength(); PropertyValue aProp; for(;pIter != pEnd;++pIter) { if ( (*pIter >>= aProp) && aProp.Name == URL_INTERACTIVE ) { aProp.Value >>= bInteractive; break; } } Reference xModi(m_xModel,UNO_QUERY); m_bCurrentlyModified = (xModi.is() && xModi->isModified()); sal_Bool bNew = sal_False; if ( bInteractive && m_xModel.is() && !m_xModel->getURL().getLength() && getView() ) { WinBits nBits(WB_STDMODAL|WB_SAVEAS); ::sfx2::FileDialogHelper aFileDlg( ::sfx2::FILESAVE_AUTOEXTENSION,static_cast(nBits) ,getView()); aFileDlg.SetDisplayDirectory( SvtPathOptions().GetWorkPath() ); const SfxFilter* pFilter = getStandardDatabaseFilter(); if ( pFilter ) { aFileDlg.AddFilter(pFilter->GetUIName(),pFilter->GetDefaultExtension()); aFileDlg.SetCurrentFilter(pFilter->GetUIName()); } Reference< XNameAccess > xDatabaseContext(getORB()->createInstance(SERVICE_SDB_DATABASECONTEXT), UNO_QUERY); if ( xDatabaseContext.is() ) { if ( aFileDlg.Execute() == ERRCODE_NONE ) { INetURLObject aURL( aFileDlg.GetPath() ); if( aURL.GetProtocol() != INET_PROT_NOT_VALID ) { ::rtl::OUString sFileName = aURL.GetMainURL( INetURLObject::NO_DECODE ); if ( ::utl::UCBContentHelper::IsDocument(sFileName) ) ::utl::UCBContentHelper::Kill(sFileName); try { Sequence< PropertyValue > aArgs; m_xModel->attachResource(sFileName,aArgs); attachModel(m_xModel); Reference xStr(m_xModel,UNO_QUERY); if ( xStr.is() ) xStr->store(); Execute(SID_DB_APP_DSCONNECTION_TYPE,Sequence()); getContainer()->disableControls(isDataSourceReadOnly()); } catch(Exception) { } } } else throw Exception(); } } } // ----------------------------------------------------------------------------- void OApplicationController::getSelectionElementNames(::std::vector< ::rtl::OUString>& _rNames) { ::vos::OGuard aSolarGuard( Application::GetSolarMutex() ); ::osl::MutexGuard aGuard(m_aMutex); OSL_ENSURE(getContainer(),"View isn't valid! -> GPF"); Reference< XDatabaseMetaData> xMetaData; if ( getContainer()->getElementType() == E_TABLE ) { Reference xConnection; ensureConnection(xConnection,sal_False); if ( xConnection.is() ) xMetaData = xConnection->getMetaData(); } getContainer()->getSelectionElementNames(_rNames,xMetaData); } // ----------------------------------------------------------------------------- ::std::auto_ptr OApplicationController::getDocumentsAccess(ElementType _eType) { OSL_ENSURE(_eType == E_FORM || _eType == E_REPORT || _eType == E_QUERY || _eType == E_TABLE,"Illegal type for call!"); Reference< XNameAccess > xNameAccess; switch( _eType ) { case E_FORM: { Reference< XFormDocumentsSupplier > xSupp(m_xModel,UNO_QUERY); if ( xSupp.is() ) xNameAccess = xSupp->getFormDocuments(); break; } case E_REPORT: { Reference< XReportDocumentsSupplier > xSupp(m_xModel,UNO_QUERY); if ( xSupp.is() ) xNameAccess = xSupp->getReportDocuments(); break; } case E_QUERY: { Reference< XQueryDefinitionsSupplier > xSupp(m_xDataSource,UNO_QUERY); if ( xSupp.is() ) xNameAccess = xSupp->getQueryDefinitions(); break; } case E_TABLE: { Reference< XTablesSupplier > xSupp(m_xDataSource,UNO_QUERY); if ( xSupp.is() ) xNameAccess = xSupp->getTables(); break; } } Reference xConnection; try { ensureConnection(xConnection); } catch(SQLContext& e) { showError(SQLExceptionInfo(e)); } catch(SQLWarning& e) { showError(SQLExceptionInfo(e)); } catch(SQLException& e) { showError(SQLExceptionInfo(e)); } OSL_ENSURE(xNameAccess.is(),"Data source doesn't return a name access -> GPF"); return ::std::auto_ptr( new OLinkedDocumentsAccess(getView(), getORB(), xNameAccess,xConnection,getDatabaseName())); } // ----------------------------------------------------------------------------- TransferableHelper* OApplicationController::copyObject() { try { ::vos::OGuard aSolarGuard( Application::GetSolarMutex() ); ::osl::MutexGuard aGuard(m_aMutex); ElementType eType = getContainer()->getElementType(); TransferableHelper* pData = NULL; switch( eType ) { case E_TABLE: case E_QUERY: { Reference xConnection; ensureConnection(xConnection,sal_False); Reference< XDatabaseMetaData> xMetaData; if ( xConnection.is() ) xMetaData = xConnection->getMetaData(); ::rtl::OUString sName = getContainer()->getQualifiedName(NULL,xMetaData); OSL_ENSURE(sName.getLength(),"NO name given!"); ::rtl::OUString sDataSource = getDatabaseName(); if ( eType == E_TABLE ) { pData = new ODataClipboard(sDataSource, CommandType::TABLE, sName, xConnection, getNumberFormatter(xConnection,getORB()), getORB()); } else { pData = new ODataClipboard(sDataSource, CommandType::QUERY, sName, getNumberFormatter(xConnection,getORB()), getORB()); } } break; case E_FORM: case E_REPORT: { ::std::vector< ::rtl::OUString> aList; getSelectionElementNames(aList); Reference< XHierarchicalNameAccess > xElements(getElements(eType),UNO_QUERY); if ( xElements.is() ) { Reference< XContent> xContent(xElements->getByHierarchicalName(*aList.begin()),UNO_QUERY); pData = new OComponentTransferable(m_sDatabaseName,xContent); } } break; } // the owner ship goes to ODataClipboards return pData; } catch(SQLException& e) { showError(SQLExceptionInfo(e)); } catch(Exception&) { DBG_ERROR("OApplicationController::copyObject: caught a generic exception!"); } return NULL; } // ----------------------------------------------------------------------------- sal_Bool OApplicationController::paste( ElementType _eType,const ::svx::ODataAccessDescriptor& _rPasteData,const String& _sParentFolder ,sal_Bool _bMove) { try { ::rtl::OUString sDataSourceName = _rPasteData.getDataSource(); if ( _eType == E_QUERY ) { sal_Int32 nCommandType = CommandType::TABLE; if ( _rPasteData.has(daCommandType) ) _rPasteData[daCommandType] >>= nCommandType; if ( CommandType::QUERY == nCommandType || CommandType::COMMAND == nCommandType ) { // read all nescessary data ::rtl::OUString sCommand; sal_Bool bEscapeProcessing = sal_True; _rPasteData[daCommand] >>= sCommand; if ( _rPasteData.has(daEscapeProcessing) ) _rPasteData[daEscapeProcessing] >>= bEscapeProcessing; // plausibility check sal_Bool bValidDescriptor = sal_False; if (CommandType::QUERY == nCommandType) bValidDescriptor = sDataSourceName.getLength() && sCommand.getLength(); else if (CommandType::COMMAND == nCommandType) bValidDescriptor = (0 != sCommand.getLength()); if (!bValidDescriptor) { DBG_ERROR("OApplicationController::pasteQuery: invalid descriptor!"); return sal_False; } // three properties we can set only if we have a query object as source ::rtl::OUString sUpdateTableName; ::rtl::OUString sUpdateSchemaName; ::rtl::OUString sUpdateCatalogName; // the target object name (as we'll suggest it to the user) String sTargetName; Reference< XNameAccess > xQueries; try { // the query container xQueries.set(getQueryDefintions(),UNO_QUERY); String aQueryDefaultName = String(ModuleRes(STR_QRY_TITLE)); aQueryDefaultName = aQueryDefaultName.GetToken(0,' '); sTargetName = ::dbtools::createUniqueName(xQueries,aQueryDefaultName); } catch(Exception) { OSL_ENSURE(0,"could not create query default name!"); } if (CommandType::QUERY == nCommandType) { // need to extract the statement and the escape processing flag from the query object sal_Bool bSuccess = sal_False; try { // the concrete query Reference< XPropertySet > xQuery; if ( xQueries.is() && xQueries->hasByName(sCommand) ) { xQuery.set(xQueries->getByName(sCommand),UNO_QUERY); if (xQuery.is()) { // extract all the properties we're interested in xQuery->getPropertyValue(PROPERTY_COMMAND) >>= sCommand; xQuery->getPropertyValue(PROPERTY_USE_ESCAPE_PROCESSING) >>= bEscapeProcessing; xQuery->getPropertyValue(PROPERTY_UPDATE_TABLENAME) >>= sUpdateTableName; xQuery->getPropertyValue(PROPERTY_UPDATE_SCHEMANAME) >>= sUpdateSchemaName; xQuery->getPropertyValue(PROPERTY_UPDATE_CATALOGNAME) >>= sUpdateCatalogName; bSuccess = sal_True; } } } catch(SQLException&) { throw; } // caught and handled by the outer catch catch(Exception&) { } if (!bSuccess) { DBG_ERROR("OApplicationController::pasteQuery: could not extract the source query object!"); // TODO: maybe this is worth an error message to be displayed to the user .... return sal_False; } } Reference< XNameContainer > xDestQueries(getQueryDefintions(), UNO_QUERY); Reference< XSingleServiceFactory > xQueryFactory(xDestQueries, UNO_QUERY); if (!xQueryFactory.is()) { DBG_ERROR("OApplicationController::pasteQuery: invalid destination query container!"); return sal_False; } // here we have everything needed to create a new query object ... // ... ehm, except a new name OSaveAsDlg aAskForName( getView(), CommandType::QUERY, xDestQueries.get(), Reference< XDatabaseMetaData>(), Reference< XConnection>(), sTargetName, SAD_ADDITIONAL_DESCRIPTION | SAD_TITLE_PASTE_AS); if ( RET_OK != aAskForName.Execute() ) // cancelled by the user return sal_False; sTargetName = aAskForName.getName(); // create a new object Reference< XPropertySet > xNewQuery(xQueryFactory->createInstance(), UNO_QUERY); DBG_ASSERT(xNewQuery.is(), "OApplicationController::pasteQuery: invalid object created by factory!"); if (xNewQuery.is()) { // initialize xNewQuery->setPropertyValue( PROPERTY_COMMAND, makeAny(sCommand) ); xNewQuery->setPropertyValue( PROPERTY_USE_ESCAPE_PROCESSING, makeAny(bEscapeProcessing) ); xNewQuery->setPropertyValue( PROPERTY_UPDATE_TABLENAME, makeAny(sUpdateTableName) ); xNewQuery->setPropertyValue( PROPERTY_UPDATE_SCHEMANAME, makeAny(sUpdateSchemaName) ); xNewQuery->setPropertyValue( PROPERTY_UPDATE_CATALOGNAME, makeAny(sUpdateCatalogName) ); // insert xDestQueries->insertByName( sTargetName, makeAny(xNewQuery) ); } } else OSL_TRACE("There should be a sequence in it!"); return sal_True; } else // forms or reports { Reference xContent; _rPasteData[daComponent] >>= xContent; return insertHierachyElement(_eType,_sParentFolder,Reference(xContent,UNO_QUERY).is(),xContent,_bMove); } } catch(SQLContext& e) { showError(SQLExceptionInfo(e)); } catch(SQLWarning& e) { showError(SQLExceptionInfo(e)); } catch(SQLException& e) { showError(SQLExceptionInfo(e)); } catch(Exception& ) { DBG_ERROR("OApplicationController::paste: caught a strange exception!"); } return sal_False; } // ----------------------------------------------------------------------------- Reference OApplicationController::getQueryDefintions() const { Reference xSet(m_xDataSource,UNO_QUERY); Reference xNames; if ( xSet.is() ) { xNames.set(xSet->getQueryDefinitions(),UNO_QUERY); } return xNames; } // ----------------------------------------------------------------------------- void OApplicationController::getSupportedFormats(ElementType _eType,::std::vector& _rFormatIds) const { switch( _eType ) { case E_TABLE: _rFormatIds.push_back(SOT_FORMATSTR_ID_DBACCESS_TABLE); _rFormatIds.push_back(SOT_FORMAT_RTF); _rFormatIds.push_back(SOT_FORMATSTR_ID_HTML); _rFormatIds.push_back(SOT_FORMATSTR_ID_HTML_SIMPLE); // run through case E_QUERY: _rFormatIds.push_back(SOT_FORMATSTR_ID_DBACCESS_QUERY); break; } } // ----------------------------------------------------------------------------- sal_Bool OApplicationController::isTableFormat() const { return m_aTableCopyHelper.isTableFormat(getViewClipboard()); } // ----------------------------------------------------------------------------- sal_Bool OApplicationController::copyTagTable(OTableCopyHelper::DropDescriptor& _rDesc, sal_Bool _bCheck) { // first get the dest connection ::osl::MutexGuard aGuard(m_aMutex); Reference xDestConnection; // supports the service sdb::connection ensureConnection( xDestConnection); if ( !xDestConnection.is() ) return sal_False; return m_aTableCopyHelper.copyTagTable(_rDesc, _bCheck,xDestConnection); } // ----------------------------------------------------------------------------- IMPL_LINK( OApplicationController, OnAsyncDrop, void*, NOTINTERESTEDIN ) { m_nAsyncDrop = 0; ::vos::OGuard aSolarGuard( Application::GetSolarMutex() ); ::osl::MutexGuard aGuard(m_aMutex); if ( m_aAsyncDrop.nType == E_TABLE ) { Reference xDestConnection; // supports the service sdb::connection ensureConnection( xDestConnection); if ( xDestConnection.is() ) m_aTableCopyHelper.asyncCopyTagTable(m_aAsyncDrop,getDatabaseName(),xDestConnection); } else { if ( paste(m_aAsyncDrop.nType,m_aAsyncDrop.aDroppedData,m_aAsyncDrop.aUrl,m_aAsyncDrop.nAction == DND_ACTION_MOVE) && m_aAsyncDrop.nAction == DND_ACTION_MOVE ) { Reference xContent; m_aAsyncDrop.aDroppedData[daComponent] >>= xContent; ::std::vector< ::rtl::OUString> aList; sal_Int32 nIndex = 0; ::rtl::OUString sName = xContent->getIdentifier()->getContentIdentifier(); ::rtl::OUString sErase = sName.getToken(0,'/',nIndex); // we don't want to have the "private:forms" part if ( nIndex != -1 ) { aList.push_back(sName.copy(sErase.getLength() + 1)); Reference xNames(getElements(m_aAsyncDrop.nType), UNO_QUERY); deleteObjects( xNames, aList, false ); } } } m_aAsyncDrop.aDroppedData.clear(); return 0L; } //........................................................................ } // namespace dbaui //........................................................................