/* -*- 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 #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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define MAX_THREADS 10 extern "C" void createRegistryInfo_ORelationControl() { static ::dbaui::OMultiInstanceAutoRegistration< ::dbaui::ORelationController > aAutoRegistration; } using namespace ::com::sun::star::uno; using namespace ::com::sun::star::io; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::frame; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::container; using namespace ::com::sun::star::sdbcx; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::sdb; using namespace ::com::sun::star::ui::dialogs; using namespace ::com::sun::star::util; using namespace ::dbtools; using namespace ::dbaui; using namespace ::comphelper; using namespace ::osl; OUString SAL_CALL ORelationController::getImplementationName() { return getImplementationName_Static(); } OUString ORelationController::getImplementationName_Static() { return OUString("org.openoffice.comp.dbu.ORelationDesign"); } Sequence< OUString> ORelationController::getSupportedServiceNames_Static() { Sequence aSupported { "com.sun.star.sdb.RelationDesign" }; return aSupported; } Sequence< OUString> SAL_CALL ORelationController::getSupportedServiceNames() { return getSupportedServiceNames_Static(); } Reference< XInterface > ORelationController::Create(const Reference& _rxFactory) { return *(new ORelationController(comphelper::getComponentContext(_rxFactory))); } ORelationController::ORelationController(const Reference< XComponentContext >& _rM) : OJoinController(_rM) ,m_nThreadEvent(0) ,m_bRelationsPossible(true) { InvalidateAll(); } ORelationController::~ORelationController() { } FeatureState ORelationController::GetState(sal_uInt16 _nId) const { FeatureState aReturn; aReturn.bEnabled = m_bRelationsPossible; switch (_nId) { case SID_RELATION_ADD_RELATION: aReturn.bEnabled = !m_vTableData.empty() && isConnected() && isEditable(); aReturn.bChecked = false; break; case ID_BROWSER_SAVEDOC: aReturn.bEnabled = haveDataSource() && impl_isModified(); break; default: aReturn = OJoinController::GetState(_nId); break; } return aReturn; } void ORelationController::Execute(sal_uInt16 _nId, const Sequence< PropertyValue >& aArgs) { switch(_nId) { case ID_BROWSER_SAVEDOC: { OSL_ENSURE(isEditable(),"Slot ID_BROWSER_SAVEDOC should not be enabled!"); if(!::dbaui::checkDataSourceAvailable(::comphelper::getString(getDataSource()->getPropertyValue(PROPERTY_NAME)), getORB())) { OUString aMessage(DBA_RES(STR_DATASOURCE_DELETED)); ScopedVclPtrInstance(getView(), aMessage)->Execute(); } else { // now we save the layout information // create the output stream try { if ( haveDataSource() && getDataSource()->getPropertySetInfo()->hasPropertyByName(PROPERTY_LAYOUTINFORMATION) ) { ::comphelper::NamedValueCollection aWindowsData; saveTableWindows( aWindowsData ); getDataSource()->setPropertyValue( PROPERTY_LAYOUTINFORMATION, makeAny( aWindowsData.getPropertyValues() ) ); setModified(false); } } catch ( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } } break; case SID_RELATION_ADD_RELATION: static_cast(static_cast( getView() )->getTableView())->AddNewRelation(); break; default: OJoinController::Execute(_nId,aArgs); return; } InvalidateFeature(_nId); } void ORelationController::impl_initialize() { OJoinController::impl_initialize(); if( !getSdbMetaData().supportsRelations() ) {// check if this database supports relations setEditable(false); m_bRelationsPossible = false; { OUString sTitle(DBA_RES(STR_RELATIONDESIGN)); sTitle = sTitle.copy(3); ScopedVclPtrInstance< OSQLMessageBox > aDlg(nullptr,sTitle,DBA_RES(STR_RELATIONDESIGN_NOT_AVAILABLE)); aDlg->Execute(); } disconnect(); throw SQLException(); } if(!m_bRelationsPossible) InvalidateAll(); // we need a datasource OSL_ENSURE(haveDataSource(),"ORelationController::initialize: need a datasource!"); Reference xSup(getConnection(),UNO_QUERY); OSL_ENSURE(xSup.is(),"Connection isn't a XTablesSupplier!"); if(xSup.is()) m_xTables = xSup->getTables(); // load the layoutInformation loadLayoutInformation(); try { loadData(); if ( !m_nThreadEvent ) Application::PostUserEvent(LINK(this, ORelationController, OnThreadFinished)); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } OUString ORelationController::getPrivateTitle( ) const { OUString sName = getDataSourceName(); return ::dbaui::getStrippedDatabaseName(getDataSource(),sName); } bool ORelationController::Construct(vcl::Window* pParent) { setView( VclPtr::Create( pParent, *this, getORB() ) ); OJoinController::Construct(pParent); return true; } short ORelationController::saveModified() { short nSaved = RET_YES; if(haveDataSource() && isModified()) { ScopedVclPtrInstance aQry(getView(), "DesignSaveModifiedDialog", "dbaccess/ui/designsavemodifieddialog.ui"); nSaved = aQry->Execute(); if(nSaved == RET_YES) Execute(ID_BROWSER_SAVEDOC,Sequence()); } return nSaved; } void ORelationController::describeSupportedFeatures() { OJoinController::describeSupportedFeatures(); implDescribeSupportedFeature( ".uno:DBAddRelation", SID_RELATION_ADD_RELATION, CommandGroup::EDIT ); } namespace { class RelationLoader : public ::osl::Thread { typedef std::map, ::comphelper::UStringMixLess> TTableDataHelper; TTableDataHelper m_aTableData; TTableConnectionData m_vTableConnectionData; const Sequence< OUString> m_aTableList; ORelationController* m_pParent; const Reference< XDatabaseMetaData> m_xMetaData; const Reference< XNameAccess > m_xTables; const sal_Int32 m_nStartIndex; const sal_Int32 m_nEndIndex; public: RelationLoader(ORelationController* _pParent ,const Reference< XDatabaseMetaData>& _xMetaData ,const Reference< XNameAccess >& _xTables ,const Sequence< OUString>& _aTableList ,const sal_Int32 _nStartIndex ,const sal_Int32 _nEndIndex) :m_aTableData(_xMetaData.is() && _xMetaData->supportsMixedCaseQuotedIdentifiers()) ,m_aTableList(_aTableList) ,m_pParent(_pParent) ,m_xMetaData(_xMetaData) ,m_xTables(_xTables) ,m_nStartIndex(_nStartIndex) ,m_nEndIndex(_nEndIndex) { } /// Working method which should be overridden. virtual void SAL_CALL run() override; virtual void SAL_CALL onTerminated() override; protected: virtual ~RelationLoader() override {} void loadTableData(const Any& _aTable); }; void SAL_CALL RelationLoader::run() { osl_setThreadName("RelationLoader"); for(sal_Int32 i = m_nStartIndex; i < m_nEndIndex; ++i) { OUString sCatalog,sSchema,sTable; ::dbtools::qualifiedNameComponents(m_xMetaData, m_aTableList[i], sCatalog, sSchema, sTable, ::dbtools::EComposeRule::InDataManipulation); Any aCatalog; if ( !sCatalog.isEmpty() ) aCatalog <<= sCatalog; try { Reference< XResultSet > xResult = m_xMetaData->getImportedKeys(aCatalog, sSchema,sTable); if ( xResult.is() && xResult->next() ) { ::comphelper::disposeComponent(xResult); loadTableData(m_xTables->getByName(m_aTableList[i])); } } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } } void SAL_CALL RelationLoader::onTerminated() { m_pParent->mergeData(m_vTableConnectionData); delete this; } void RelationLoader::loadTableData(const Any& _aTable) { Reference xTableProp(_aTable,UNO_QUERY); const OUString sSourceName = ::dbtools::composeTableName( m_xMetaData, xTableProp, ::dbtools::EComposeRule::InTableDefinitions, false ); TTableDataHelper::const_iterator aFind = m_aTableData.find(sSourceName); if ( aFind == m_aTableData.end() ) { aFind = m_aTableData.emplace(sSourceName,std::make_shared(xTableProp,sSourceName, sSourceName, OUString())).first; aFind->second->ShowAll(false); } TTableWindowData::value_type pReferencingTable = aFind->second; Reference xKeys = pReferencingTable->getKeys(); const Reference xKeySup(xTableProp,UNO_QUERY); if ( !xKeys.is() && xKeySup.is() ) { xKeys = xKeySup->getKeys(); } if ( xKeys.is() ) { Reference xKey; const sal_Int32 nCount = xKeys->getCount(); for(sal_Int32 i = 0 ; i < nCount ; ++i) { xKeys->getByIndex(i) >>= xKey; sal_Int32 nKeyType = 0; xKey->getPropertyValue(PROPERTY_TYPE) >>= nKeyType; if ( KeyType::FOREIGN == nKeyType ) { OUString sReferencedTable; xKey->getPropertyValue(PROPERTY_REFERENCEDTABLE) >>= sReferencedTable; // insert windows TTableDataHelper::const_iterator aRefFind = m_aTableData.find(sReferencedTable); if ( aRefFind == m_aTableData.end() ) { if ( m_xTables->hasByName(sReferencedTable) ) { Reference xReferencedTable(m_xTables->getByName(sReferencedTable),UNO_QUERY); aRefFind = m_aTableData.emplace(sReferencedTable,std::make_shared(xReferencedTable,sReferencedTable, sReferencedTable, OUString())).first; aRefFind->second->ShowAll(false); } else continue; // table name could not be found so we do not show this table relation } TTableWindowData::value_type pReferencedTable = aRefFind->second; OUString sKeyName; xKey->getPropertyValue(PROPERTY_NAME) >>= sKeyName; // insert connection ORelationTableConnectionData* pTabConnData = new ORelationTableConnectionData( pReferencingTable, pReferencedTable, sKeyName ); m_vTableConnectionData.push_back(TTableConnectionData::value_type(pTabConnData)); // insert columns const Reference xColsSup(xKey,UNO_QUERY); OSL_ENSURE(xColsSup.is(),"Key is no XColumnsSupplier!"); const Reference xColumns = xColsSup->getColumns(); const Sequence< OUString> aNames = xColumns->getElementNames(); OUString sColumnName,sRelatedName; for(sal_Int32 j=0;j xPropSet(xColumns->getByName(aNames[j]),UNO_QUERY); OSL_ENSURE(xPropSet.is(),"Invalid column found in KeyColumns!"); if ( xPropSet.is() ) { xPropSet->getPropertyValue(PROPERTY_NAME) >>= sColumnName; xPropSet->getPropertyValue(PROPERTY_RELATEDCOLUMN) >>= sRelatedName; } pTabConnData->SetConnLine( j, sColumnName, sRelatedName ); } // set update/del flags sal_Int32 nUpdateRule = 0; sal_Int32 nDeleteRule = 0; xKey->getPropertyValue(PROPERTY_UPDATERULE) >>= nUpdateRule; xKey->getPropertyValue(PROPERTY_DELETERULE) >>= nDeleteRule; pTabConnData->SetUpdateRules( nUpdateRule ); pTabConnData->SetDeleteRules( nDeleteRule ); // set cardinality pTabConnData->SetCardinality(); } } } } } void ORelationController::mergeData(const TTableConnectionData& _aConnectionData) { ::osl::MutexGuard aGuard( getMutex() ); std::copy( _aConnectionData.begin(), _aConnectionData.end(), std::back_inserter( m_vTableConnectionData )); // here we are finished, so we can collect the table from connection data for (auto const& elem : m_vTableConnectionData) { if ( !existsTable(elem->getReferencingTable()->GetComposedName()) ) { m_vTableData.push_back(elem->getReferencingTable()); } if ( !existsTable(elem->getReferencedTable()->GetComposedName()) ) { m_vTableData.push_back(elem->getReferencedTable()); } } if ( m_nThreadEvent ) { --m_nThreadEvent; if ( !m_nThreadEvent ) Application::PostUserEvent(LINK(this, ORelationController, OnThreadFinished)); } } IMPL_LINK_NOARG( ORelationController, OnThreadFinished, void*, void ) { ::SolarMutexGuard aSolarGuard; ::osl::MutexGuard aGuard( getMutex() ); try { getView()->initialize(); // show the windows and fill with our information getView()->Invalidate(InvalidateFlags::NoErase); ClearUndoManager(); setModified(false); // and we are not modified yet if(m_vTableData.empty()) Execute(ID_BROWSER_ADDTABLE,Sequence()); } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } m_pWaitObject.reset(); } void ORelationController::loadData() { m_pWaitObject.reset( new WaitObject(getView()) ); try { if ( !m_xTables.is() ) return; DatabaseMetaData aMeta(getConnection()); // this may take some time const Reference< XDatabaseMetaData> xMetaData = getConnection()->getMetaData(); const Sequence< OUString> aNames = m_xTables->getElementNames(); const sal_Int32 nCount = aNames.getLength(); if ( aMeta.supportsThreads() ) { const sal_Int32 nMaxElements = (nCount / MAX_THREADS) +1; sal_Int32 nStart = 0,nEnd = std::min(nMaxElements,nCount); while(nStart != nEnd) { ++m_nThreadEvent; RelationLoader* pThread = new RelationLoader(this,xMetaData,m_xTables,aNames,nStart,nEnd); pThread->createSuspended(); pThread->setPriority(osl_Thread_PriorityBelowNormal); pThread->resume(); nStart = nEnd; nEnd += nMaxElements; nEnd = std::min(nEnd,nCount); } } else { RelationLoader* pThread = new RelationLoader(this,xMetaData,m_xTables,aNames,0,nCount); pThread->run(); pThread->onTerminated(); } } catch(SQLException& e) { showError(SQLExceptionInfo(e)); } catch(const Exception&) { DBG_UNHANDLED_EXCEPTION(); } } TTableWindowData::value_type ORelationController::existsTable(const OUString& _rComposedTableName) const { ::comphelper::UStringMixEqual bCase(true); for (auto const& elem : m_vTableData) { if(bCase(elem->GetComposedName(),_rComposedTableName)) return elem; } return TTableWindowData::value_type(); } void ORelationController::loadLayoutInformation() { try { OSL_ENSURE(haveDataSource(),"We need a datasource from our connection!"); if ( haveDataSource() ) { if ( getDataSource()->getPropertySetInfo()->hasPropertyByName(PROPERTY_LAYOUTINFORMATION) ) { Sequence aWindows; getDataSource()->getPropertyValue(PROPERTY_LAYOUTINFORMATION) >>= aWindows; loadTableWindows(aWindows); } } } catch(Exception&) { } } void ORelationController::reset() { loadLayoutInformation(); ODataView* pView = getView(); OSL_ENSURE(pView,"No current view!"); if(pView) { pView->initialize(); pView->Invalidate(InvalidateFlags::NoErase); } } bool ORelationController::allowViews() const { return false; } bool ORelationController::allowQueries() const { return false; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */