/* -*- 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 "AppController.hxx" #include "AppDetailView.hxx" #include "AppView.hxx" #include #include #include #include #include #include #include #include "subcomponentmanager.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace dbaui { using namespace ::dbtools; using namespace ::connectivity; using namespace ::svx; using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::awt; using namespace ::com::sun::star::util; using namespace ::com::sun::star::frame; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::ui::dialogs; using namespace ::com::sun::star::sdb; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::sdbcx; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::container; using namespace ::com::sun::star::ucb; using ::com::sun::star::ui::XContextMenuInterceptor; namespace DatabaseObject = ::com::sun::star::sdb::application::DatabaseObject; namespace ErrorCondition = ::com::sun::star::sdb::ErrorCondition; void OApplicationController::convertToView(const OUString& _sName) { try { SharedConnection xConnection( getConnection() ); Reference< XQueriesSupplier > xSup( xConnection, UNO_QUERY_THROW ); Reference< XNameAccess > xQueries( xSup->getQueries(), UNO_SET_THROW ); Reference< XPropertySet > xSourceObject( xQueries->getByName( _sName ), UNO_QUERY_THROW ); Reference< XTablesSupplier > xTablesSup( xConnection, UNO_QUERY_THROW ); Reference< XNameAccess > xTables( xTablesSup->getTables(), UNO_SET_THROW ); Reference< XDatabaseMetaData > xMeta = xConnection->getMetaData(); const OUString aDefaultName = ::dbaui::createDefaultName(xMeta, xTables, DBA_RES(STR_TBL_TITLE).getToken(0, ' ')); DynamicTableOrQueryNameCheck aNameChecker( xConnection, CommandType::TABLE ); OSaveAsDlg aDlg(getFrameWeld(), CommandType::TABLE, getORB(), xConnection, aDefaultName, aNameChecker, SADFlags::NONE); if (aDlg.run() == RET_OK) { OUString sName = aDlg.getName(); OUString sCatalog = aDlg.getCatalog(); OUString sSchema = aDlg.getSchema(); OUString sNewName( ::dbtools::composeTableName( xMeta, sCatalog, sSchema, sName, false, ::dbtools::EComposeRule::InTableDefinitions ) ); Reference xView = ::dbaui::createView(sNewName,xConnection,xSourceObject); if ( !xView.is() ) throw SQLException(DBA_RES(STR_NO_TABLE_FORMAT_INSIDE),*this, u"S1000"_ustr,0,Any()); getContainer()->elementAdded(E_TABLE,sNewName,Any(xView)); } } catch(const SQLException& ) { showError( SQLExceptionInfo( ::cppu::getCaughtException() ) ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION("dbaccess"); } } void OApplicationController::pasteFormat(SotClipboardFormatId _nFormatId) { if ( _nFormatId == SotClipboardFormatId::NONE ) return; try { const TransferableDataHelper& rClipboard = getViewClipboard(); ElementType eType = getContainer()->getElementType(); if ( eType == E_TABLE ) { m_aTableCopyHelper.pasteTable( _nFormatId, rClipboard, getDatabaseName(), ensureConnection() ); } else paste( eType, ODataAccessObjectTransferable::extractObjectDescriptor( rClipboard ) ); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION("dbaccess"); } } void OApplicationController::openDialog( const OUString& _sServiceName ) { try { SolarMutexGuard aSolarGuard; ::osl::MutexGuard aGuard( getMutex() ); weld::WaitObject aWO(getFrameWeld()); Sequence< Any > aArgs(3); auto pArgs = aArgs.getArray(); sal_Int32 nArgPos = 0; Reference< css::awt::XWindow> xWindow = getTopMostContainerWindow(); if ( !xWindow.is() ) { OSL_ENSURE( getContainer(), "OApplicationController::Construct: have no view!" ); if ( getContainer() ) xWindow = VCLUnoHelper::GetInterface(getView()->Window::GetParent()); } // the parent window pArgs[nArgPos++] <<= PropertyValue( u"ParentWindow"_ustr, 0, Any(xWindow), PropertyState_DIRECT_VALUE); // the initial selection OUString sInitialSelection; if ( getContainer() ) sInitialSelection = getDatabaseName(); if ( !sInitialSelection.isEmpty() ) { pArgs[ nArgPos++ ] <<= PropertyValue( u"InitialSelection"_ustr, 0, Any( sInitialSelection ), PropertyState_DIRECT_VALUE ); } SharedConnection xConnection( getConnection() ); if ( xConnection.is() ) { pArgs[ nArgPos++ ] <<= PropertyValue( PROPERTY_ACTIVE_CONNECTION, 0, makeAny( xConnection ), PropertyState_DIRECT_VALUE ); } aArgs.realloc( nArgPos ); // create the dialog Reference< XExecutableDialog > xAdminDialog; xAdminDialog.set( getORB()->getServiceManager()->createInstanceWithArgumentsAndContext(_sServiceName, aArgs, getORB()), UNO_QUERY); // execute it if (xAdminDialog.is()) xAdminDialog->execute(); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION("dbaccess"); } } void OApplicationController::refreshTables() { if ( !(getContainer() && getContainer()->getDetailView()) ) return; weld::WaitObject aWO(getFrameWeld()); OSL_ENSURE(getContainer()->getElementType() == E_TABLE,"Only allowed when the tables container is selected!"); try { Reference xRefresh(getElements(E_TABLE),UNO_QUERY); if ( xRefresh.is() ) xRefresh->refresh(); } catch(const Exception&) { OSL_FAIL("Could not refresh tables!"); } getContainer()->getDetailView()->clearPages(false); getContainer()->getDetailView()->createTablesPage( ensureConnection() ); } void SAL_CALL OApplicationController::propertyChange( const PropertyChangeEvent& evt ) { SolarMutexGuard aSolarGuard; ::osl::MutexGuard aGuard( getMutex() ); if ( evt.PropertyName == PROPERTY_USER ) { m_bNeedToReconnect = true; InvalidateFeature(SID_DB_APP_STATUS_USERNAME); } else if ( evt.PropertyName == PROPERTY_URL ) { m_bNeedToReconnect = true; InvalidateFeature(SID_DB_APP_STATUS_DBNAME); InvalidateFeature(SID_DB_APP_STATUS_TYPE); InvalidateFeature(SID_DB_APP_STATUS_HOSTNAME); } else if ( PROPERTY_NAME == evt.PropertyName ) { const ElementType eType = getContainer()->getElementType(); if ( eType == E_FORM || eType == E_REPORT ) { OUString sOldName,sNewName; evt.OldValue >>= sOldName; evt.NewValue >>= sNewName; // if the old name is empty, then this is a newly inserted content. We're notified of it via the // elementInserted method, so there's no need to handle it here. if ( !sOldName.isEmpty() ) { Reference xChild(evt.Source,UNO_QUERY); if ( xChild.is() ) { Reference xContent(xChild->getParent(),UNO_QUERY); if ( xContent.is() ) sOldName = xContent->getIdentifier()->getContentIdentifier() + "/" + sOldName; } getContainer()->elementReplaced( eType , sOldName, sNewName ); } } } EventObject aEvt; aEvt.Source = m_xModel; modified(aEvt); } Reference< XDataSource > SAL_CALL OApplicationController::getDataSource() { ::osl::MutexGuard aGuard( getMutex() ); Reference< XDataSource > xDataSource( m_xDataSource, UNO_QUERY ); return xDataSource; } Reference< XWindow > SAL_CALL OApplicationController::getApplicationMainWindow() { ::osl::MutexGuard aGuard( getMutex() ); Reference< XFrame > xFrame( getFrame(), UNO_SET_THROW ); Reference< XWindow > xWindow( xFrame->getContainerWindow(), UNO_SET_THROW ); return xWindow; } Sequence< Reference< XComponent > > SAL_CALL OApplicationController::getSubComponents() { ::osl::MutexGuard aGuard( getMutex() ); return m_pSubComponentManager->getSubComponents(); } Reference< XConnection > SAL_CALL OApplicationController::getActiveConnection() { ::osl::MutexGuard aGuard( getMutex() ); return m_xDataSourceConnection.getTyped(); } sal_Bool SAL_CALL OApplicationController::isConnected( ) { ::osl::MutexGuard aGuard( getMutex() ); return m_xDataSourceConnection.is(); } void SAL_CALL OApplicationController::connect( ) { SQLExceptionInfo aError; SharedConnection xConnection = ensureConnection( &aError ); if ( !xConnection.is() ) { if ( aError.isValid() ) aError.doThrow(); // no particular error, but nonetheless could not connect -> throw a generic exception OUString sConnectingContext( DBA_RES( STR_COULDNOTCONNECT_DATASOURCE ) ); OUString sDatabaseName; ::dbtools::throwGenericSQLException( sConnectingContext.replaceFirst( "$name$", ::dbaui::getStrippedDatabaseName(m_xDataSource, sDatabaseName) ), *this ); } } beans::Pair< ::sal_Int32, OUString > SAL_CALL OApplicationController::identifySubComponent( const Reference< XComponent >& i_rSubComponent ) { ::osl::MutexGuard aGuard( getMutex() ); sal_Int32 nType = -1; OUString sName; if ( !m_pSubComponentManager->lookupSubComponent( i_rSubComponent, sName, nType ) ) throw IllegalArgumentException( OUString(), *this, 1 ); if ( nType == SID_DB_APP_DSRELDESIGN ) // this is somewhat hacky ... we're expected to return a DatabaseObject value. However, there is no such // value for the relation design. /me thinks we should change the API definition here ... nType = -1; return beans::Pair< ::sal_Int32, OUString >( nType, sName ); } sal_Bool SAL_CALL OApplicationController::closeSubComponents( ) { SolarMutexGuard aSolarGuard; ::osl::MutexGuard aGuard( getMutex() ); return m_pSubComponentManager->closeSubComponents(); } namespace { ElementType lcl_objectType2ElementType( const sal_Int32 _nObjectType ) { ElementType eType( E_NONE ); switch ( _nObjectType ) { case DatabaseObject::TABLE: eType = E_TABLE; break; case DatabaseObject::QUERY: eType = E_QUERY; break; case DatabaseObject::FORM: eType = E_FORM; break; case DatabaseObject::REPORT: eType = E_REPORT; break; default: OSL_FAIL( "lcl_objectType2ElementType: unsupported object type!" ); // this should have been caught earlier } return eType; } } void OApplicationController::impl_validateObjectTypeAndName_throw( const sal_Int32 _nObjectType, const ::std::optional< OUString >& i_rObjectName ) { // ensure we're connected if ( !isConnected() ) { SQLError aError; aError.raiseException( ErrorCondition::DB_NOT_CONNECTED, *this ); } // ensure a proper object type if ( ( _nObjectType != DatabaseObject::TABLE ) && ( _nObjectType != DatabaseObject::QUERY ) && ( _nObjectType != DatabaseObject::FORM ) && ( _nObjectType != DatabaseObject::REPORT ) ) throw IllegalArgumentException( OUString(), *this, 1 ); if ( !i_rObjectName ) return; // ensure an existing object Reference< XNameAccess > xContainer( getElements( lcl_objectType2ElementType( _nObjectType ) ) ); if ( !xContainer.is() ) // all possible reasons for this (e.g. not being connected currently) should // have been handled before throw RuntimeException( OUString(), *this ); bool bExistentObject = false; switch ( _nObjectType ) { case DatabaseObject::TABLE: case DatabaseObject::QUERY: bExistentObject = xContainer->hasByName( *i_rObjectName ); break; case DatabaseObject::FORM: case DatabaseObject::REPORT: { Reference< XHierarchicalNameAccess > xHierarchy( xContainer, UNO_QUERY_THROW ); bExistentObject = xHierarchy->hasByHierarchicalName( *i_rObjectName ); } break; } if ( !bExistentObject ) throw NoSuchElementException( *i_rObjectName, *this ); } Reference< XComponent > SAL_CALL OApplicationController::loadComponent( ::sal_Int32 ObjectType, const OUString& ObjectName, sal_Bool ForEditing ) { return loadComponentWithArguments( ObjectType, ObjectName, ForEditing, Sequence< PropertyValue >() ); } Reference< XComponent > SAL_CALL OApplicationController::loadComponentWithArguments( ::sal_Int32 ObjectType, const OUString& ObjectName, sal_Bool ForEditing, const Sequence< PropertyValue >& Arguments ) { SolarMutexGuard aSolarGuard; ::osl::MutexGuard aGuard( getMutex() ); impl_validateObjectTypeAndName_throw( ObjectType, ObjectName ); Reference< XComponent > xComponent( openElementWithArguments( ObjectName, lcl_objectType2ElementType( ObjectType ), ForEditing ? ElementOpenMode::Design : ElementOpenMode::Normal, ForEditing ? SID_DB_APP_EDIT : SID_DB_APP_OPEN, ::comphelper::NamedValueCollection( Arguments ) ) ); return xComponent; } Reference< XComponent > SAL_CALL OApplicationController::createComponent( ::sal_Int32 i_nObjectType, Reference< XComponent >& o_DocumentDefinition ) { return createComponentWithArguments( i_nObjectType, Sequence< PropertyValue >(), o_DocumentDefinition ); } Reference< XComponent > SAL_CALL OApplicationController::createComponentWithArguments( ::sal_Int32 i_nObjectType, const Sequence< PropertyValue >& i_rArguments, Reference< XComponent >& o_DocumentDefinition ) { SolarMutexGuard aSolarGuard; ::osl::MutexGuard aGuard( getMutex() ); impl_validateObjectTypeAndName_throw( i_nObjectType, ::std::optional< OUString >() ); Reference< XComponent > xComponent( newElement( lcl_objectType2ElementType( i_nObjectType ), ::comphelper::NamedValueCollection( i_rArguments ), o_DocumentDefinition ) ); return xComponent; } void SAL_CALL OApplicationController::registerContextMenuInterceptor( const Reference< XContextMenuInterceptor >& Interceptor ) { if ( Interceptor.is() ) m_aContextMenuInterceptors.addInterface( Interceptor ); } void SAL_CALL OApplicationController::releaseContextMenuInterceptor( const Reference< XContextMenuInterceptor >& Interceptor ) { m_aContextMenuInterceptors.removeInterface( Interceptor ); } void OApplicationController::previewChanged( sal_Int32 _nMode ) { SolarMutexGuard aSolarGuard; ::osl::MutexGuard aGuard( getMutex() ); if ( m_xDataSource.is() && !isDataSourceReadOnly() ) { try { ::comphelper::NamedValueCollection aLayoutInfo( m_xDataSource->getPropertyValue( PROPERTY_LAYOUTINFORMATION ) ); sal_Int32 nOldMode = aLayoutInfo.getOrDefault( u"Preview"_ustr, _nMode ); if ( nOldMode != _nMode ) { aLayoutInfo.put( u"Preview"_ustr, _nMode ); m_xDataSource->setPropertyValue( PROPERTY_LAYOUTINFORMATION, Any( aLayoutInfo.getPropertyValues() ) ); } } catch ( const Exception& ) { DBG_UNHANDLED_EXCEPTION("dbaccess"); } } InvalidateFeature(SID_DB_APP_DISABLE_PREVIEW); InvalidateFeature(SID_DB_APP_VIEW_DOCINFO_PREVIEW); InvalidateFeature(SID_DB_APP_VIEW_DOC_PREVIEW); } void OApplicationController::askToReconnect() { if ( !m_bNeedToReconnect ) return; m_bNeedToReconnect = false; bool bClear = true; if ( !m_pSubComponentManager->empty() ) { std::unique_ptr xQry(Application::CreateMessageDialog(getFrameWeld(), VclMessageType::Question, VclButtonsType::YesNo, DBA_RES(STR_QUERY_CLOSEDOCUMENTS))); switch (xQry->run()) { case RET_YES: closeSubComponents(); break; default: bClear = false; break; } } if ( bClear ) { ElementType eType = getContainer()->getElementType(); disconnect(); getContainer()->getDetailView()->clearPages(false); getContainer()->selectContainer(E_NONE); // invalidate the old selection m_eCurrentType = E_NONE; getContainer()->selectContainer(eType); // reselect the current one again } } OUString OApplicationController::getDatabaseName() const { OUString sDatabaseName; try { if ( m_xDataSource.is() ) { OSL_VERIFY( m_xDataSource->getPropertyValue( PROPERTY_NAME ) >>= sDatabaseName ); } } catch ( const Exception& ) { DBG_UNHANDLED_EXCEPTION("dbaccess"); } return sDatabaseName; } void OApplicationController::onDocumentOpened( const OUString& _rName, const sal_Int32 _nType, const ElementOpenMode _eMode, const Reference< XComponent >& _xDocument, const Reference< XComponent >& _rxDefinition ) { if ( !_xDocument.is() ) return; try { OSL_ENSURE( _xDocument.is(), "OApplicationController::onDocumentOpened: is there any *valid* scenario where this fails?" ); m_pSubComponentManager->onSubComponentOpened( _rName, _nType, _eMode, _xDocument.is() ? _xDocument : _rxDefinition ); if ( _rxDefinition.is() ) { Reference< XPropertySet > xProp( _rxDefinition, UNO_QUERY_THROW ); Reference< XPropertySetInfo > xPSI( xProp->getPropertySetInfo(), UNO_SET_THROW ); xProp->addPropertyChangeListener( PROPERTY_NAME, static_cast< XPropertyChangeListener* >( this ) ); } } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION("dbaccess"); } } bool OApplicationController::insertHierarchyElement(ElementType _eType, const OUString& _sParentFolder, bool _bCollection, const Reference& _xContent, bool _bMove) { Reference xNames(getElements(_eType), UNO_QUERY); return dbaui::insertHierarchyElement(getFrameWeld() ,getORB() ,xNames ,_sParentFolder ,_eType == E_FORM ,_bCollection ,_xContent ,_bMove); } bool OApplicationController::isRenameDeleteAllowed(ElementType _eType, bool _bDelete) const { ElementType eType = getContainer()->getElementType(); bool bEnabled = !isDataSourceReadOnly() && eType == _eType; if ( bEnabled ) { if ( E_TABLE == eType ) bEnabled = !isConnectionReadOnly() && getContainer()->isALeafSelected(); bool bCompareRes = false; if ( _bDelete ) bCompareRes = getContainer()->getSelectionCount() > 0; else { bCompareRes = getContainer()->getSelectionCount() == 1; if ( bEnabled && bCompareRes && E_TABLE == eType ) { std::vector< OUString> aList; getSelectionElementNames(aList); try { Reference< XNameAccess > xContainer = const_cast(this)->getElements(eType); bEnabled = (xContainer.is() && xContainer->hasByName(*aList.begin())); if ( bEnabled ) bEnabled = Reference(xContainer->getByName(*aList.begin()),UNO_QUERY).is(); } catch(Exception&) { bEnabled = false; } } } bEnabled = bEnabled && bCompareRes; } return bEnabled; } void OApplicationController::onLoadedMenu(const Reference< css::frame::XLayoutManager >& _xLayoutManager) { if ( !_xLayoutManager.is() ) return; static constexpr OUString s_sStatusbar = u"private:resource/statusbar/statusbar"_ustr; _xLayoutManager->createElement( s_sStatusbar ); _xLayoutManager->requestElement( s_sStatusbar ); if ( getContainer() ) { // we need to share the "mnemonic space": MnemonicGenerator aMnemonicGenerator; // - the menu already has mnemonics if (SystemWindow* pSystemWindow = getContainer()->GetSystemWindow()) pSystemWindow->CollectMenuBarMnemonics(aMnemonicGenerator); // - the icons should use automatic ones getContainer()->createIconAutoMnemonics( aMnemonicGenerator ); // - as well as the entries in the task pane getContainer()->setTaskExternalMnemonics( aMnemonicGenerator ); } Execute( SID_DB_APP_VIEW_FORMS, Sequence< PropertyValue >() ); InvalidateAll(); } void OApplicationController::doAction(sal_uInt16 _nId, const ElementOpenMode _eOpenMode) { std::vector< OUString> aList; getSelectionElementNames(aList); ElementType eType = getContainer()->getElementType(); ::comphelper::NamedValueCollection aArguments; ElementOpenMode eOpenMode = _eOpenMode; if ( eType == E_REPORT && ElementOpenMode::Mail == _eOpenMode ) { aArguments.put(u"Hidden"_ustr,true); eOpenMode = ElementOpenMode::Normal; } std::vector< std::pair< OUString ,Reference< XModel > > > aComponents; for (auto const& elem : aList) { if ( SID_DB_APP_CONVERTTOVIEW == _nId ) convertToView(elem); else { Reference< XModel > xModel( openElementWithArguments( elem, eType, eOpenMode, _nId,aArguments ), UNO_QUERY ); aComponents.emplace_back( elem, xModel ); } } // special handling for mail, if more than one document is selected attach them all if ( _eOpenMode != ElementOpenMode::Mail ) return; SfxMailModel aSendMail; SfxMailModel::SendMailResult eResult = SfxMailModel::SEND_MAIL_OK; for (auto const& component : aComponents) { try { Reference< XModel > xModel = component.second; // Send document as e-Mail using stored/default type eResult = aSendMail.AttachDocument(xModel,component.first); ::comphelper::disposeComponent(xModel); if (eResult != SfxMailModel::SEND_MAIL_OK) break; } catch(const Exception&) { DBG_UNHANDLED_EXCEPTION("dbaccess"); } } if ( !aSendMail.IsEmpty() ) aSendMail.Send( getFrame() ); } ElementType OApplicationController::getElementType(const Reference< XContainer >& _xContainer) { ElementType eRet = E_NONE; Reference xServiceInfo(_xContainer,UNO_QUERY); if ( xServiceInfo.is() ) { if ( xServiceInfo->supportsService(SERVICE_SDBCX_TABLES) ) eRet = E_TABLE; else if ( xServiceInfo->supportsService(SERVICE_NAME_FORM_COLLECTION) ) eRet = E_FORM; else if ( xServiceInfo->supportsService(SERVICE_NAME_REPORT_COLLECTION) ) eRet = E_REPORT; else eRet = E_QUERY; } return eRet; } } // namespace dbaui /* vim:set shiftwidth=4 softtabstop=4 expandtab: */