/* -*- 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 using namespace ::svx; using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::sdb; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::sdbcx; using namespace ::com::sun::star::frame; struct DBTextStruct_Impl { SwDBData aDBData; Sequence aSelection; Reference xCursor; Reference xConnection; }; void SwTextShell::ExecDB(SfxRequest const &rReq) { const SfxItemSet *pArgs = rReq.GetArgs(); SwDBManager* pDBManager = GetShell().GetDBManager(); OUString sSourceArg, sCommandArg; sal_Int32 nCommandTypeArg = 0; const SfxUnoAnyItem* pSourceItem = nullptr; const SfxUnoAnyItem* pCommandItem = nullptr; const SfxUnoAnyItem* pCommandTypeItem = nullptr; const SfxUnoAnyItem* pConnectionItem = nullptr; // first get the selection of rows to be inserted Sequence aSelection; if(const SfxUnoAnyItem* pSelectionItem = pArgs->GetItemIfSet(FN_DB_DATA_SELECTION_ANY, false)) pSelectionItem->GetValue() >>= aSelection; // get the data source name pSourceItem = pArgs->GetItemIfSet(FN_DB_DATA_SOURCE_ANY, false); if(pSourceItem) pSourceItem->GetValue() >>= sSourceArg; // get the command pCommandItem = pArgs->GetItemIfSet(FN_DB_DATA_COMMAND_ANY, false); if(pCommandItem) pCommandItem->GetValue() >>= sCommandArg; // get the command type pCommandTypeItem = pArgs->GetItemIfSet(FN_DB_DATA_COMMAND_TYPE_ANY, false); if(pCommandTypeItem) pCommandTypeItem->GetValue() >>= nCommandTypeArg; Reference xConnection; pConnectionItem = pArgs->GetItemIfSet(FN_DB_CONNECTION_ANY, false); if(pConnectionItem) pConnectionItem->GetValue() >>= xConnection; // may be we even get no connection if ( !xConnection.is() ) { Reference xSource; SwView &rSwView = GetView(); xConnection = SwDBManager::GetConnection(sSourceArg, xSource, &rSwView); } if(!xConnection.is()) return ; // get the cursor, we use to travel, may be NULL Reference xCursor; if ( const SfxUnoAnyItem* pCursorItem = pArgs->GetItemIfSet(FN_DB_DATA_CURSOR_ANY, false) ) pCursorItem->GetValue() >>= xCursor; switch (rReq.GetSlot()) { case FN_QRY_INSERT: { if(pSourceItem && pCommandItem && pCommandTypeItem) { DBTextStruct_Impl* pNew = new DBTextStruct_Impl; pNew->aDBData.sDataSource = sSourceArg; pNew->aDBData.sCommand = sCommandArg; pNew->aDBData.nCommandType = nCommandTypeArg; pNew->aSelection = aSelection; //if the cursor is NULL, it must be created inside InsertDBTextHdl // because it called via a PostUserEvent pNew->xCursor = xCursor; pNew->xConnection = xConnection; Application::PostUserEvent( LINK( this, SwBaseShell, InsertDBTextHdl ), pNew ); // the pNew will be removed in InsertDBTextHdl !! } } break; case FN_QRY_MERGE_FIELD: { // we don't get any cursor, so we must create our own bool bDisposeResultSet = false; if ( !xCursor.is() ) { SwView &rSwView = GetView(); xCursor = SwDBManager::createCursor(sSourceArg,sCommandArg,nCommandTypeArg,xConnection,&rSwView); bDisposeResultSet = xCursor.is(); } ODataAccessDescriptor aDescriptor; aDescriptor.setDataSource(sSourceArg); aDescriptor[DataAccessDescriptorProperty::Command] <<= sCommandArg; aDescriptor[DataAccessDescriptorProperty::Cursor] <<= xCursor; aDescriptor[DataAccessDescriptorProperty::Selection] <<= aSelection; aDescriptor[DataAccessDescriptorProperty::CommandType] <<= nCommandTypeArg; SwMergeDescriptor aMergeDesc( DBMGR_MERGE, *GetShellPtr(), aDescriptor ); pDBManager->Merge(aMergeDesc); if ( bDisposeResultSet ) ::comphelper::disposeComponent(xCursor); } break; case FN_QRY_INSERT_FIELD: { const SfxUnoAnyItem* pColumnItem = pArgs->GetItemIfSet(FN_DB_COLUMN_ANY, false); const SfxUnoAnyItem* pColumnNameItem = pArgs->GetItemIfSet(FN_DB_DATA_COLUMN_NAME_ANY, false); OUString sColumnName; if(pColumnNameItem) pColumnNameItem->GetValue() >>= sColumnName; OUString sDBName = sSourceArg + OUStringChar(DB_DELIM) + sCommandArg + OUStringChar(DB_DELIM) + OUString::number(nCommandTypeArg) + OUStringChar(DB_DELIM) + sColumnName; SwFieldMgr aFieldMgr(GetShellPtr()); SwInsertField_Data aData(SwFieldTypesEnum::Database, 0, sDBName, OUString(), 0); if(pConnectionItem) aData.m_aDBConnection = pConnectionItem->GetValue(); if(pColumnItem) aData.m_aDBColumn = pColumnItem->GetValue(); aFieldMgr.InsertField(aData); SfxViewFrame& rViewFrame = GetView().GetViewFrame(); uno::Reference< XDispatchRecorder > xRecorder = rViewFrame.GetBindings().GetRecorder(); if ( xRecorder.is() ) { SfxRequest aReq(rViewFrame, FN_INSERT_DBFIELD); aReq.AppendItem( SfxUInt16Item(FN_PARAM_FIELD_TYPE, static_cast(SwFieldTypesEnum::Database))); aReq.AppendItem( SfxStringItem( FN_INSERT_DBFIELD, sDBName )); aReq.AppendItem( SfxStringItem( FN_PARAM_1, sCommandArg )); aReq.AppendItem( SfxStringItem( FN_PARAM_2, sColumnName )); aReq.AppendItem( SfxInt32Item( FN_PARAM_3, nCommandTypeArg)); aReq.Done(); } } break; default: OSL_ENSURE(false, "wrong dispatcher"); return; } } IMPL_LINK( SwBaseShell, InsertDBTextHdl, void*, p, void ) { std::shared_ptr pDBStruct(static_cast(p)); if( !pDBStruct ) return; bool bDispose = false; Reference< sdbc::XConnection> xConnection = pDBStruct->xConnection; Reference xSource = SwDBManager::getDataSourceAsParent(xConnection,pDBStruct->aDBData.sDataSource); // #111987# the connection is disposed and so no parent has been found if(xConnection.is() && !xSource.is()) return; if ( !xConnection.is() ) { SwView &rSwView = GetView(); xConnection = SwDBManager::GetConnection(pDBStruct->aDBData.sDataSource, xSource, &rSwView); bDispose = true; } Reference< XColumnsSupplier> xColSupp; if(xConnection.is()) xColSupp = SwDBManager::GetColumnSupplier(xConnection, pDBStruct->aDBData.sCommand, pDBStruct->aDBData.nCommandType == CommandType::QUERY ? SwDBSelect::QUERY : SwDBSelect::TABLE); if( !xColSupp ) { if ( bDispose ) ::comphelper::disposeComponent(xConnection); return; } SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create(); VclPtrpDlg (pFact->CreateSwInsertDBColAutoPilot(GetView(), xSource, xColSupp, pDBStruct->aDBData)); pDlg->StartExecuteAsync( [pDlg, pDBStruct2=std::move(pDBStruct), xSource, xConnection, bDispose] (sal_Int32 nResult) mutable { if (nResult == RET_OK) pDlg->DataToDoc(pDBStruct2->aSelection, xSource, xConnection, pDBStruct2->xCursor); pDlg->disposeOnce(); if ( bDispose ) ::comphelper::disposeComponent(xConnection); } ); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */