Files
libreoffice/sw/source/ui/dbui/dbmgr.cxx
Kurt Zenker 00bf1cf13a INTEGRATION: CWS OS18 (1.70.68); FILE MERGED
2003/08/29 09:05:24 os 1.70.68.1: #111987# catch exception in getDataSourceAsParent()
2003-09-11 08:41:01 +00:00

2507 lines
88 KiB
C++

/*************************************************************************
*
* $RCSfile: dbmgr.cxx,v $
*
* $Revision: 1.71 $
*
* last change: $Author: kz $ $Date: 2003-09-11 09:41:01 $
*
* The Contents of this file are made available subject to the terms of
* either of the following licenses
*
* - GNU Lesser General Public License Version 2.1
* - Sun Industry Standards Source License Version 1.1
*
* Sun Microsystems Inc., October, 2000
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2000 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
*
*
* Sun Industry Standards Source License Version 1.1
* =================================================
* The contents of this file are subject to the Sun Industry Standards
* Source License Version 1.1 (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.openoffice.org/license.html.
*
* Software provided under this License is provided on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
* WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
* MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
* See the License for the specific provisions governing your rights and
* obligations concerning the Software.
*
* The Initial Developer of the Original Code is: Sun Microsystems, Inc.
*
* Copyright: 2000 by Sun Microsystems, Inc.
*
* All Rights Reserved.
*
* Contributor(s): _______________________________________
*
*
************************************************************************/
#pragma hdrstop
#if STLPORT_VERSION>=321
#include <cstdarg>
#endif
#include <stdio.h>
#include <com/sun/star/text/NotePrintMode.hpp>
#ifndef _COM_SUN_STAR_SDB_COMMANDTYPE_HPP_
#include <com/sun/star/sdb/CommandType.hpp>
#endif
#ifndef _COM_SUN_STAR_FRAME_XCOMPONENTLOADER_HPP_
#include <com/sun/star/frame/XComponentLoader.hpp>
#endif
#ifndef _COM_SUN_STAR_UTIL_iXNUMBERFORMATTER_HPP_
#include <com/sun/star/util/XNumberFormatter.hpp>
#endif
#ifndef _COM_SUN_STAR_SDB_XCOMPLETEDCONNECTION_HPP_
#include <com/sun/star/sdb/XCompletedConnection.hpp>
#endif
#ifndef _COM_SUN_STAR_SDB_XCOMPLETEDEXECUTION_HPP_
#include <com/sun/star/sdb/XCompletedExecution.hpp>
#endif
#ifndef _COM_SUN_STAR_CONTAINER_XCHILD_HPP_
#include <com/sun/star/container/XChild.hpp>
#endif
#ifndef _COM_SUN_STAR_TEXT_MAILMERGEEVENT_
#include <com/sun/star/text/MailMergeEvent.hpp>
#endif
#ifndef _SFXVIEWFRM_HXX
#include <sfx2/viewfrm.hxx>
#endif
#ifndef _DBCONFIG_HXX
#include <dbconfig.hxx>
#endif
#ifndef _SWDBTOOLSCLIENT_HXX
#include <swdbtoolsclient.hxx>
#endif
#ifndef _LSTBOX_HXX //autogen
#include <vcl/lstbox.hxx>
#endif
#ifndef _UNOTOOLS_TEMPFILE_HXX
#include <unotools/tempfile.hxx>
#endif
#ifndef SVTOOLS_URIHELPER_HXX
#include <svtools/urihelper.hxx>
#endif
#ifndef _SVSTDARR_HXX
#define _SVSTDARR_STRINGSDTOR
#include <svtools/svstdarr.hxx>
#endif
#ifndef _ZFORLIST_HXX //autogen
#include <svtools/zforlist.hxx>
#endif
#ifndef _ZFORMAT_HXX //autogen
#include <svtools/zformat.hxx>
#endif
#ifndef _SFXSTRITEM_HXX //autogen
#include <svtools/stritem.hxx>
#endif
#ifndef _SFXENUMITEM_HXX //autogen
#include <svtools/eitem.hxx>
#endif
#ifndef _SFX_PRINTER_HXX //autogen
#include <sfx2/printer.hxx>
#endif
#ifndef _SFXDOCFILE_HXX //autogen
#include <sfx2/docfile.hxx>
#endif
#ifndef _SFX_PROGRESS_HXX //autogen
#include <sfx2/progress.hxx>
#endif
#ifndef _SFX_DOCFILT_HACK_HXX //autogen
#include <sfx2/docfilt.hxx>
#endif
#ifndef _SFXDISPATCH_HXX //autogen
#include <sfx2/dispatch.hxx>
#endif
#ifndef _OFF_APP_HXX //autogen
#include <offmgr/app.hxx>
#endif
#ifndef _MAILENUM_HXX //autogen
#include <goodies/mailenum.hxx>
#endif
#include <cmdid.h>
#ifndef _SWTYPES_HXX
#include <swtypes.hxx>
#endif
#ifndef _SWMODULE_HXX
#include <swmodule.hxx>
#endif
#ifndef _VIEW_HXX
#include <view.hxx>
#endif
#ifndef _DOCSH_HXX
#include <docsh.hxx>
#endif
#ifndef _EDTWIN_HXX
#include <edtwin.hxx>
#endif
#ifndef _DBINSDLG_HXX
#include <dbinsdlg.hxx>
#endif
#ifndef _WRTSH_HXX
#include <wrtsh.hxx>
#endif
#ifndef _FLDBAS_HXX
#include <fldbas.hxx>
#endif
#ifndef _INITUI_HXX
#include <initui.hxx>
#endif
#ifndef _SWUNDO_HXX
#include <swundo.hxx>
#endif
#ifndef _FLDDAT_HXX
#include <flddat.hxx>
#endif
#ifndef _SWMODULE_HXX
#include <swmodule.hxx>
#endif
#ifndef _MODCFG_HXX
#include <modcfg.hxx>
#endif
#ifndef _SWPRTOPT_HXX
#include <swprtopt.hxx>
#endif
#ifndef _SHELLIO_HXX
#include <shellio.hxx>
#endif
#ifndef _DBUI_HXX
#include <dbui.hxx>
#endif
#ifndef _DBMGR_HXX
#include <dbmgr.hxx>
#endif
#ifndef _DOC_HXX
#include <doc.hxx>
#endif
#ifndef _SWWAIT_HXX
#include <swwait.hxx>
#endif
#ifndef _SWUNOHELPER_HXX
#include <swunohelper.hxx>
#endif
#ifndef _DBUI_HRC
#include <dbui.hrc>
#endif
#ifndef _GLOBALS_HRC
#include <globals.hrc>
#endif
#ifndef _STATSTR_HRC
#include <statstr.hrc>
#endif
#ifndef _SFXREQUEST_HXX
#include <sfx2/request.hxx>
#endif
#ifndef _HINTIDS_HXX
#include <hintids.hxx>
#endif
#ifndef _COM_SUN_STAR_LANG_XMULTISERVICEFACTORY_HPP_
#include <com/sun/star/lang/XMultiServiceFactory.hpp>
#endif
#ifndef _COM_SUN_STAR_CONTAINER_XNAMEACCESS_HPP_
#include <com/sun/star/container/XNameAccess.hpp>
#endif
#ifndef _COM_SUN_STAR_SDBC_XDATASOURCE_HPP_
#include <com/sun/star/sdbc/XDataSource.hpp>
#endif
#ifndef _COM_SUN_STAR_SDBC_XROWSET_HPP_
#include <com/sun/star/sdbc/XRowSet.hpp>
#endif
#ifndef _COM_SUN_STAR_SDBCX_XTABLESSUPPLIER_HPP_
#include <com/sun/star/sdbcx/XTablesSupplier.hpp>
#endif
#ifndef _COM_SUN_STAR_SDBCX_XCOLUMNSSUPPLIER_HPP_
#include <com/sun/star/sdbcx/XColumnsSupplier.hpp>
#endif
#ifndef _COM_SUN_STAR_SDB_XQUERIESSUPPLIER_HPP_
#include <com/sun/star/sdb/XQueriesSupplier.hpp>
#endif
#ifndef _COM_SUN_STAR_SDB_XCOLUMN_HPP_
#include <com/sun/star/sdb/XColumn.hpp>
#endif
#ifndef _COM_SUN_STAR_SDBC_DATATYPE_HPP_
#include <com/sun/star/sdbc/DataType.hpp>
#endif
#ifndef _COM_SUN_STAR_SDBC_XSTATEMENT_HPP_
#include <com/sun/star/sdbc/XStatement.hpp>
#endif
#ifndef _COM_SUN_STAR_SDBC_RESULTSETTYPE_HPP_
#include <com/sun/star/sdbc/ResultSetType.hpp>
#endif
#ifndef _COM_SUN_STAR_BEANS_XPROPERTYSET_HPP_
#include <com/sun/star/beans/XPropertySet.hpp>
#endif
#ifndef _COM_SUN_STAR_BEANS_PROPERTYVALUE_HPP_
#include <com/sun/star/beans/PropertyValue.hpp>
#endif
#ifndef _COMPHELPER_PROCESSFACTORY_HXX_
#include <comphelper/processfactory.hxx>
#endif
#ifndef _ISOLANG_HXX
#include <tools/isolang.hxx>
#endif
#ifndef _COM_SUN_STAR_UTIL_XNUMBERFORMATTYPES_HPP_
#include <com/sun/star/util/XNumberFormatTypes.hpp>
#endif
#ifndef _SVX_LANGITEM_HXX
#include <svx/langitem.hxx>
#endif
#ifndef _SVX_UNOMID_HXX
#include <svx/unomid.hxx>
#endif
#ifndef _NUMUNO_HXX
#include <svtools/numuno.hxx>
#endif
#include "mailmrge.hxx"
#include <unomailmerge.hxx>
#ifndef _SFXEVENT_HXX
#include <sfx2/event.hxx>
#endif
#ifndef _SV_MSGBOX_HXX
#include <vcl/msgbox.hxx>
#endif
#ifndef _SVX_DATACCESSDESCRIPTOR_HXX_
#include <svx/dataaccessdescriptor.hxx>
#endif
#ifndef _CPPUHELPER_IMPLBASE1_HXX_
#include <cppuhelper/implbase1.hxx>
#endif
#ifndef _VOS_MUTEX_HXX_
#include <vos/mutex.hxx>
#endif
using namespace svx;
using namespace ::com::sun::star;
using namespace com::sun::star::text;
using namespace com::sun::star::uno;
using namespace com::sun::star::container;
using namespace com::sun::star::frame;
using namespace com::sun::star::lang;
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::util;
using namespace com::sun::star::task;
#define C2S(cChar) String::CreateFromAscii(cChar)
#define C2U(cChar) ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(cChar))
#define DB_SEP_SPACE 0
#define DB_SEP_TAB 1
#define DB_SEP_RETURN 2
#define DB_SEP_NEWLINE 3
SV_IMPL_PTRARR(SwDSParamArr, SwDSParamPtr);
const sal_Char cCursor[] = "Cursor";
const sal_Char cCommand[] = "Command";
const sal_Char cCommandType[] = "CommandType";
const sal_Char cDataSourceName[] = "DataSourceName";
const sal_Char cSelection[] = "Selection";
const sal_Char cActiveConnection[] = "ActiveConnection";
// -----------------------------------------------------------------------------
// Use nameless namespace to avoid to rubbish the global namespace
// -----------------------------------------------------------------------------
namespace
{
}
/* -----------------09.12.2002 12:35-----------------
*
* --------------------------------------------------*/
class SwConnectionDisposedListener_Impl : public cppu::WeakImplHelper1
< com::sun::star::lang::XEventListener >
{
SwNewDBMgr& rDBMgr;
virtual void SAL_CALL disposing( const EventObject& Source ) throw (RuntimeException);
public:
SwConnectionDisposedListener_Impl(SwNewDBMgr& rMgr);
~SwConnectionDisposedListener_Impl();
};
// -----------------------------------------------------------------------------
struct SwNewDBMgr_Impl
{
SwDSParam* pMergeData;
SwMailMergeDlg* pMergeDialog;
Reference<XEventListener> xDisposeListener;
SwNewDBMgr_Impl(SwNewDBMgr& rDBMgr)
:pMergeData(0)
,pMergeDialog(0)
,xDisposeListener(new SwConnectionDisposedListener_Impl(rDBMgr))
{}
};
/* -----------------------------17.07.00 17:04--------------------------------
---------------------------------------------------------------------------*/
BOOL lcl_MoveAbsolute(SwDSParam* pParam, long nAbsPos)
{
BOOL bRet = FALSE;
try
{
if(pParam->bScrollable)
{
bRet = pParam->xResultSet->absolute( nAbsPos );
}
else
{
DBG_ERROR("no absolute positioning available")
}
}
catch(Exception aExcept)
{
}
return bRet;
}
/* -----------------------------17.07.00 17:23--------------------------------
---------------------------------------------------------------------------*/
BOOL lcl_GetColumnCnt(SwDSParam* pParam,
const String& rColumnName, long nLanguage, String& rResult, double* pNumber)
{
Reference< XColumnsSupplier > xColsSupp( pParam->xResultSet, UNO_QUERY );
Reference <XNameAccess> xCols = xColsSupp->getColumns();
if(!xCols->hasByName(rColumnName))
return FALSE;
Any aCol = xCols->getByName(rColumnName);
Reference< XPropertySet > xColumnProps;
aCol >>= xColumnProps;
SwDBFormatData aFormatData;
aFormatData.aNullDate = pParam->aNullDate;
aFormatData.xFormatter = pParam->xFormatter;
String sLanguage, sCountry;
::ConvertLanguageToIsoNames( (USHORT)nLanguage, sLanguage, sCountry );
aFormatData.aLocale.Language = sLanguage;
aFormatData.aLocale.Country = sCountry;
rResult = SwNewDBMgr::GetDBField( xColumnProps, aFormatData, pNumber);
return TRUE;
};
/*--------------------------------------------------------------------
Beschreibung: Daten importieren
--------------------------------------------------------------------*/
BOOL SwNewDBMgr::MergeNew(USHORT nOpt, SwWrtShell& rSh,
const ODataAccessDescriptor& _rDescriptor)
{
SetMergeType( nOpt );
DBG_ASSERT(!bInMerge && !pImpl->pMergeData, "merge already activated!")
SwDBData aData;
aData.nCommandType = CommandType::TABLE;
Reference<XResultSet> xResSet;
Sequence<Any> aSelection;
Reference< XConnection> xConnection;
_rDescriptor[daDataSource] >>= aData.sDataSource;
_rDescriptor[daCommand] >>= aData.sCommand;
_rDescriptor[daCommandType] >>= aData.nCommandType;
if ( _rDescriptor.has(daCursor) )
_rDescriptor[daCursor] >>= xResSet;
if ( _rDescriptor.has(daSelection) )
_rDescriptor[daSelection] >>= aSelection;
if ( _rDescriptor.has(daConnection) )
_rDescriptor[daConnection] >>= xConnection;
if(!aData.sDataSource.getLength() || !aData.sCommand.getLength() || !xResSet.is())
{
return FALSE;
}
pImpl->pMergeData = new SwDSParam(aData, xResSet, aSelection);
SwDSParam* pTemp = FindDSData(aData, FALSE);
if(pTemp)
*pTemp = *pImpl->pMergeData;
else
{
//#94779# calls from the calculator may have added a connection with an invalid commandtype
//"real" data base connections added here have to re-use the already available
//DSData and set the correct CommandType
SwDBData aTempData(aData);
aData.nCommandType = -1;
pTemp = FindDSData(aData, FALSE);
if(pTemp)
*pTemp = *pImpl->pMergeData;
else
{
SwDSParam* pInsert = new SwDSParam(*pImpl->pMergeData);
aDataSourceParams.Insert(pInsert, aDataSourceParams.Count());
try
{
Reference<XComponent> xComponent(pInsert->xConnection, UNO_QUERY);
if(xComponent.is())
xComponent->addEventListener(pImpl->xDisposeListener);
}
catch(Exception&)
{
}
}
}
if(!pImpl->pMergeData->xConnection.is())
pImpl->pMergeData->xConnection = xConnection;
// add an XEventListener
try{
//set to start position
if(pImpl->pMergeData->aSelection.getLength())
{
sal_Int32 nPos = 0;
pImpl->pMergeData->aSelection.getConstArray()[ pImpl->pMergeData->nSelectionIndex++ ] >>= nPos;
pImpl->pMergeData->bEndOfDB = !pImpl->pMergeData->xResultSet->absolute( nPos );
pImpl->pMergeData->CheckEndOfDB();
if(pImpl->pMergeData->nSelectionIndex >= pImpl->pMergeData->aSelection.getLength())
pImpl->pMergeData->bEndOfDB = TRUE;
}
else
{
pImpl->pMergeData->bEndOfDB = !pImpl->pMergeData->xResultSet->first();
pImpl->pMergeData->CheckEndOfDB();
}
}
catch(Exception&)
{
pImpl->pMergeData->bEndOfDB = TRUE;
pImpl->pMergeData->CheckEndOfDB();
DBG_ERROR("exception in MergeNew()")
}
Reference<XDataSource> xSource = SwNewDBMgr::getDataSourceAsParent(xConnection,aData.sDataSource);
Reference<XMultiServiceFactory> xMgr = ::comphelper::getProcessServiceFactory();
if( xMgr.is() )
{
Reference<XInterface> xInstance = xMgr->createInstance( C2U( "com.sun.star.util.NumberFormatter" ));
pImpl->pMergeData->xFormatter = Reference<util::XNumberFormatter>(xInstance, UNO_QUERY) ;
}
Reference<XPropertySet> xSourceProps(xSource, UNO_QUERY);
if(xSourceProps.is())
{
Any aFormats = xSourceProps->getPropertyValue(C2U("NumberFormatsSupplier"));
if(aFormats.hasValue())
{
Reference<XNumberFormatsSupplier> xSuppl;
aFormats >>= xSuppl;
if(xSuppl.is())
{
Reference< XPropertySet > xSettings = xSuppl->getNumberFormatSettings();
Any aNull = xSettings->getPropertyValue(C2U("NullDate"));
aNull >>= pImpl->pMergeData->aNullDate;
}
}
}
rSh.ChgDBData(aData);
bInMerge = TRUE;
if (IsInitDBFields())
{
// Bei Datenbankfeldern ohne DB-Name DB-Name von Dok einsetzen
SvStringsDtor aDBNames(1, 1);
aDBNames.Insert( new String(), 0);
SwDBData aData = rSh.GetDBData();
String sDBName = aData.sDataSource;
sDBName += DB_DELIM;
sDBName += (String)aData.sCommand;
sDBName += DB_DELIM;
sDBName += String::CreateFromInt32(aData.nCommandType);
rSh.ChangeDBFields( aDBNames, sDBName);
SetInitDBFields(FALSE);
}
BOOL bRet = TRUE;
switch(nOpt)
{
case DBMGR_MERGE:
bRet = Merge(&rSh); // Mischen
break;
case DBMGR_MERGE_MAILMERGE: // Serienbrief
{
SfxDispatcher *pDis = rSh.GetView().GetViewFrame()->GetDispatcher();
SfxBoolItem aMerge(FN_QRY_MERGE, TRUE);
// !! Currently (Jan-2003) silent is defined by supplying *any*
// !! item!! (Thus according to OS it would be silent even when
// !! other items then SID_SILENT would be supplied!)
// !! Therefore it has to be the 0 pointer when not silent.
SfxBoolItem aMergeSilent(SID_SILENT, TRUE);
SfxBoolItem *pMergeSilent = IsMergeSilent() ? &aMergeSilent : 0;
pDis->Execute(SID_PRINTDOC,
SFX_CALLMODE_SYNCHRON|SFX_CALLMODE_RECORD, &aMerge, pMergeSilent, 0L);
}
break;
case DBMGR_MERGE_MAILING:
bRet = MergeMailing(&rSh); // Mailing
break;
case DBMGR_MERGE_MAILFILES:
bRet = MergeMailFiles(&rSh); // Serienbriefe als Dateien abspeichern
break;
default: // Einfuegen der selektierten Eintraege
// (war: InsertRecord)
ImportFromConnection(&rSh);
break;
}
EndMerge();
return bRet;
}
/*--------------------------------------------------------------------
Beschreibung: Daten importieren
--------------------------------------------------------------------*/
BOOL SwNewDBMgr::Merge(SwWrtShell* pSh)
{
pSh->StartAllAction();
pSh->ViewShell::UpdateFlds(TRUE);
pSh->SetModified();
pSh->EndAllAction();
return TRUE;
}
/*--------------------------------------------------------------------
Beschreibung:
--------------------------------------------------------------------*/
void SwNewDBMgr::ImportFromConnection( SwWrtShell* pSh )
{
if(pImpl->pMergeData && !pImpl->pMergeData->bEndOfDB)
{
{
pSh->StartAllAction();
pSh->StartUndo(0);
BOOL bGroupUndo(pSh->DoesGroupUndo());
pSh->DoGroupUndo(FALSE);
if( pSh->HasSelection() )
pSh->DelRight();
SwWait *pWait = 0;
{
ULONG i = 0;
do {
ImportDBEntry(pSh);
if( 10 == ++i )
pWait = new SwWait( *pSh->GetView().GetDocShell(), TRUE);
} while(ToNextMergeRecord());
}
pSh->DoGroupUndo(bGroupUndo);
pSh->EndUndo(0);
pSh->EndAllAction();
delete pWait;
}
}
}
/*-----------------24.02.97 10.30-------------------
--------------------------------------------------*/
String lcl_FindColumn(const String& sFormatStr,USHORT &nUsedPos, BYTE &nSeparator)
{
String sReturn;
USHORT nLen = sFormatStr.Len();
nSeparator = 0xff;
while(nUsedPos < nLen && nSeparator == 0xff)
{
sal_Unicode cAkt = sFormatStr.GetChar(nUsedPos);
switch(cAkt)
{
case ',':
nSeparator = DB_SEP_SPACE;
break;
case ';':
nSeparator = DB_SEP_RETURN;
break;
case ':':
nSeparator = DB_SEP_TAB;
break;
case '#':
nSeparator = DB_SEP_NEWLINE;
break;
default:
sReturn += cAkt;
}
nUsedPos++;
}
return sReturn;
}
/*--------------------------------------------------------------------
Beschreibung:
--------------------------------------------------------------------*/
inline String lcl_GetDBInsertMode( const SwDBData& rData )
{
return aEmptyStr;
#if 0
//JP 13.11.00: must be change to the new configuration
return SFX_APP()->GetIniManager()->Get( String::CreateFromAscii(
RTL_CONSTASCII_STRINGPARAM( "DataBaseFormatInfo" )),
FALSE, FALSE, sDBName );
#endif
}
void SwNewDBMgr::ImportDBEntry(SwWrtShell* pSh)
{
if(pImpl->pMergeData && !pImpl->pMergeData->bEndOfDB)
{
Reference< XColumnsSupplier > xColsSupp( pImpl->pMergeData->xResultSet, UNO_QUERY );
Reference <XNameAccess> xCols = xColsSupp->getColumns();
String sFormatStr( lcl_GetDBInsertMode( *pImpl->pMergeData ));
USHORT nFmtLen = sFormatStr.Len();
if( nFmtLen )
{
const char cSpace = ' ';
const char cTab = '\t';
USHORT nUsedPos = 0;
BYTE nSeparator;
String sColumn = lcl_FindColumn(sFormatStr, nUsedPos, nSeparator);
while( sColumn.Len() )
{
if(!xCols->hasByName(sColumn))
return;
Any aCol = xCols->getByName(sColumn);
Reference< XPropertySet > xColumnProp;
aCol >>= xColumnProp;
if(xColumnProp.is())
{
SwDBFormatData aDBFormat;
String sInsert = GetDBField( xColumnProp, aDBFormat);
if( DB_SEP_SPACE == nSeparator )
sInsert += cSpace;
else if( DB_SEP_TAB == nSeparator)
sInsert += cTab;
pSh->Insert(sInsert);
if( DB_SEP_RETURN == nSeparator)
pSh->SplitNode();
else if(DB_SEP_NEWLINE == nSeparator)
pSh->InsertLineBreak();
}
else
{
// Spalte nicht gefunden -> Fehler anzeigen
String sInsert = '?';
sInsert += sColumn;
sInsert += '?';
pSh->Insert(sInsert);
}
sColumn = lcl_FindColumn(sFormatStr, nUsedPos, nSeparator);
}
pSh->SplitNode();
}
else
{
String sStr;
Sequence<rtl::OUString> aColNames = xCols->getElementNames();
const rtl::OUString* pColNames = aColNames.getConstArray();
long nLength = aColNames.getLength();
for(long i = 0; i < nLength; i++)
{
Any aCol = xCols->getByName(pColNames[i]);
Reference< XPropertySet > xColumnProp;
aCol >>= xColumnProp;
SwDBFormatData aDBFormat;
sStr += GetDBField( xColumnProp, aDBFormat);
if (i < nLength - 1)
sStr += '\t';
}
pSh->SwEditShell::Insert(sStr);
pSh->SwFEShell::SplitNode(); // Zeilenvorschub
}
}
}
/*--------------------------------------------------------------------
Beschreibung: Listbox mit Tabellenliste fuellen
--------------------------------------------------------------------*/
BOOL SwNewDBMgr::GetTableNames(ListBox* pListBox, const String& rDBName)
{
BOOL bRet = FALSE;
String sOldTableName(pListBox->GetSelectEntry());
pListBox->Clear();
SwDSParam* pParam = FindDSConnection(rDBName, FALSE);
Reference< XConnection> xConnection;
if(pParam && pParam->xConnection.is())
xConnection = pParam->xConnection;
else
{
rtl::OUString sDBName(rDBName);
xConnection = RegisterConnection( sDBName );
}
if(xConnection.is())
{
Reference<XTablesSupplier> xTSupplier = Reference<XTablesSupplier>(xConnection, UNO_QUERY);
if(xTSupplier.is())
{
Reference<XNameAccess> xTbls = xTSupplier->getTables();
Sequence<rtl::OUString> aTbls = xTbls->getElementNames();
const rtl::OUString* pTbls = aTbls.getConstArray();
for(long i = 0; i < aTbls.getLength(); i++)
{
USHORT nEntry = pListBox->InsertEntry(pTbls[i]);
pListBox->SetEntryData(nEntry, (void*)0);
}
}
Reference<XQueriesSupplier> xQSupplier = Reference<XQueriesSupplier>(xConnection, UNO_QUERY);
if(xQSupplier.is())
{
Reference<XNameAccess> xQueries = xQSupplier->getQueries();
Sequence<rtl::OUString> aQueries = xQueries->getElementNames();
const rtl::OUString* pQueries = aQueries.getConstArray();
for(long i = 0; i < aQueries.getLength(); i++)
{
USHORT nEntry = pListBox->InsertEntry(pQueries[i]);
pListBox->SetEntryData(nEntry, (void*)1);
}
}
if (sOldTableName.Len())
pListBox->SelectEntry(sOldTableName);
bRet = TRUE;
}
return bRet;
}
/*--------------------------------------------------------------------
Beschreibung: Listbox mit Spaltennamen einer Datenbank fuellen
--------------------------------------------------------------------*/
BOOL SwNewDBMgr::GetColumnNames(ListBox* pListBox,
const String& rDBName, const String& rTableName, BOOL bAppend)
{
if (!bAppend)
pListBox->Clear();
SwDSParam* pParam = FindDSConnection(rDBName, FALSE);
Reference< XConnection> xConnection;
if(pParam && pParam->xConnection.is())
xConnection = pParam->xConnection;
else
{
rtl::OUString sDBName(rDBName);
xConnection = RegisterConnection( sDBName );
}
Reference< XColumnsSupplier> xColsSupp = SwNewDBMgr::GetColumnSupplier(xConnection, rTableName);
if(xColsSupp.is())
{
Reference <XNameAccess> xCols = xColsSupp->getColumns();
const Sequence<rtl::OUString> aColNames = xCols->getElementNames();
const rtl::OUString* pColNames = aColNames.getConstArray();
for(int nCol = 0; nCol < aColNames.getLength(); nCol++)
{
pListBox->InsertEntry(pColNames[nCol]);
}
}
return(TRUE);
}
/* -----------------------------08.06.01 15:11--------------------------------
---------------------------------------------------------------------------*/
BOOL SwNewDBMgr::GetColumnNames(ListBox* pListBox,
Reference< XConnection> xConnection,
const String& rTableName, BOOL bAppend)
{
if (!bAppend)
pListBox->Clear();
Reference< XColumnsSupplier> xColsSupp = SwNewDBMgr::GetColumnSupplier(xConnection, rTableName);
if(xColsSupp.is())
{
Reference <XNameAccess> xCols = xColsSupp->getColumns();
const Sequence<rtl::OUString> aColNames = xCols->getElementNames();
const rtl::OUString* pColNames = aColNames.getConstArray();
for(int nCol = 0; nCol < aColNames.getLength(); nCol++)
{
pListBox->InsertEntry(pColNames[nCol]);
}
}
return(TRUE);
}
/*--------------------------------------------------------------------
Beschreibung: CTOR
--------------------------------------------------------------------*/
SwNewDBMgr::SwNewDBMgr() :
pImpl(new SwNewDBMgr_Impl(*this)),
pMergeEvtSrc(NULL),
bInMerge(FALSE),
bMergeLock(FALSE),
bMergeSilent(FALSE),
nMergeType(DBMGR_INSERT),
bInitDBFields(FALSE)
{
}
/* -----------------------------18.07.00 08:56--------------------------------
---------------------------------------------------------------------------*/
SwNewDBMgr::~SwNewDBMgr()
{
for(USHORT nPos = 0; nPos < aDataSourceParams.Count(); nPos++)
{
SwDSParam* pParam = aDataSourceParams[nPos];
if(pParam->xConnection.is())
{
try
{
Reference<XComponent> xComp(pParam->xConnection, UNO_QUERY);
if(xComp.is())
xComp->dispose();
}
catch(const RuntimeException& )
{
//may be disposed already since multiple entries may have used the same connection
}
}
}
delete pImpl;
}
/*--------------------------------------------------------------------
Beschreibung: Serienbrief drucken
--------------------------------------------------------------------*/
BOOL SwNewDBMgr::MergePrint( SwView& rView,
SwPrtOptions& rOpt, SfxProgress& rProgress )
{
SwWrtShell* pSh = &rView.GetWrtShell();
//check if the doc is synchronized and contains at least one linked section
BOOL bSynchronizedDoc = pSh->IsLabelDoc() && pSh->GetSectionFmtCount() > 1;
//merge source is already open
rOpt.nMergeCnt = 0;
if(pImpl->pMergeData)
{
if(pImpl->pMergeData->aSelection.getLength())
rOpt.nMergeCnt = pImpl->pMergeData->aSelection.getLength();
else if(pImpl->pMergeData->xResultSet.is())
{
Reference<XPropertySet> xPrSet(pImpl->pMergeData->xResultSet, UNO_QUERY);
if(xPrSet.is())
{
try
{
sal_Bool bFinal;
Any aFinal = xPrSet->getPropertyValue(C2U("IsRowCountFinal"));
aFinal >>= bFinal;
if(!bFinal)
{
pImpl->pMergeData->xResultSet->last();
pImpl->pMergeData->xResultSet->first();
}
long nCount;
Any aCount = xPrSet->getPropertyValue(C2U("RowCount"));
aCount >>= nCount;
rOpt.nMergeCnt = (ULONG)nCount;
}
catch(Exception&)
{
}
}
}
}
SwModuleOptions* pModOpt = SW_MOD()->GetModuleConfig();
pModOpt->SetSinglePrintJob(rOpt.IsPrintSingleJobs());
SfxPrinter *pPrt = pSh->GetPrt();
Link aSfxSaveLnk = pPrt->GetEndPrintHdl();
if( rOpt.IsPrintSingleJobs() )
pPrt->SetEndPrintHdl( Link() );
BOOL bNewJob = FALSE,
bUserBreak = FALSE,
bRet = FALSE;
long nStartRow, nEndRow;
//calculate number of data sets to be printed
Sequence<PropertyValue> aViewProperties(15);
PropertyValue* pViewProperties = aViewProperties.getArray();
pViewProperties[0].Name = C2U("MailMergeCount");
pViewProperties[0].Value <<= (sal_Int32)rOpt.nMergeCnt;
pViewProperties[1].Name = C2U("PrintGraphics");
pViewProperties[1].Value <<= (sal_Bool)rOpt.IsPrintGraphic();
pViewProperties[2].Name = C2U("PrintTables");
pViewProperties[2].Value <<= (sal_Bool)rOpt.IsPrintTable();
pViewProperties[3].Name = C2U("PrintDrawings");
pViewProperties[3].Value <<= (sal_Bool)rOpt.IsPrintDraw();
pViewProperties[4].Name = C2U("PrintLeftPages");
pViewProperties[4].Value <<= (sal_Bool)rOpt.IsPrintLeftPage();
pViewProperties[5].Name = C2U("PrintRightPages");
pViewProperties[5].Value <<= (sal_Bool)rOpt.IsPrintRightPage();
pViewProperties[6].Name = C2U("PrintControls");
pViewProperties[6].Value <<= (sal_Bool)rOpt.IsPrintControl();
pViewProperties[7].Name = C2U("PrintReversed");
pViewProperties[7].Value <<= (sal_Bool)rOpt.IsPrintReverse();
pViewProperties[8].Name = C2U("PrintPaperFromSetup");
pViewProperties[8].Value <<= (sal_Bool)rOpt.IsPaperFromSetup();
pViewProperties[9].Name = C2U("PrintFaxName");
pViewProperties[9].Value <<= rOpt.GetFaxName();
pViewProperties[10].Name = C2U("PrintAnnotationMode");
pViewProperties[10].Value <<= (com::sun::star::text::NotePrintMode) rOpt.GetPrintPostIts();
pViewProperties[11].Name = C2U("PrintProspect");
pViewProperties[11].Value <<= (sal_Bool)rOpt.IsPrintProspect();
pViewProperties[12].Name = C2U("PrintPageBackground");
pViewProperties[12].Value <<= (sal_Bool)rOpt.IsPrintPageBackground();
pViewProperties[13].Name = C2U("PrintBlackFonts");
pViewProperties[13].Value <<= (sal_Bool)rOpt.IsPrintBlackFont();
pViewProperties[14].Name = C2U("IsSinglePrintJob");
pViewProperties[14].Value <<= (sal_Bool)rOpt.IsPrintSingleJobs();
rView.SetAdditionalPrintOptions(aViewProperties);
do {
nStartRow = pImpl->pMergeData ? pImpl->pMergeData->xResultSet->getRow() : 0;
{
pSh->ViewShell::UpdateFlds();
++rOpt.nMergeAct;
// launch MailMergeEvent if required
const SwXMailMerge *pEvtSrc = GetMailMergeEvtSrc();
if (pEvtSrc)
{
Reference< XInterface > xRef( (XMailMergeBroadcaster *) pEvtSrc );
text::MailMergeEvent aEvt( xRef, rView.GetDocShell()->GetModel() );
pEvtSrc->LaunchMailMergeEvent( aEvt );
}
rView.SfxViewShell::Print( rProgress ); // ggf Basic-Macro ausfuehren
if( rOpt.IsPrintSingleJobs() && bRet )
{
//rOpt.bJobStartet = FALSE;
bRet = FALSE;
}
bMergeLock = TRUE;
if(rOpt.IsPrintProspect())
{
if( pPrt->IsJobActive() || pPrt->StartJob( rOpt.GetJobName() ))
{
pSh->PrintProspect( rOpt, rProgress );
bRet = TRUE;
}
}
else if( pSh->Prt( rOpt, rProgress ) )
bRet = TRUE;
bMergeLock = FALSE;
if( !pPrt->IsJobActive() )
{
bUserBreak = TRUE;
bRet = FALSE;
break;
}
if( !rOpt.IsPrintSingleJobs() )
{
String& rJNm = (String&)rOpt.GetJobName();
rJNm.Erase();
}
}
nEndRow = pImpl->pMergeData ? pImpl->pMergeData->xResultSet->getRow() : 0;
} while( bSynchronizedDoc && (nStartRow != nEndRow)? ExistsNextRecord() : ToNextMergeRecord());
if( rOpt.IsPrintSingleJobs() )
{
pSh->GetPrt()->SetEndPrintHdl( aSfxSaveLnk );
if ( !bUserBreak && !pSh->GetPrt()->IsJobActive() ) //Schon zu spaet?
aSfxSaveLnk.Call( pSh->GetPrt() );
}
rOpt.nMergeCnt = 0;
rOpt.nMergeAct = 0;
nMergeType = DBMGR_INSERT;
SwDocShell* pDocSh = rView.GetDocShell();
SfxViewFrame *pTmpFrm = SfxViewFrame::GetFirst(pDocSh);
while (pTmpFrm) // Alle Views Invalidieren
{
SwView *pVw = PTR_CAST(SwView, pTmpFrm->GetViewShell());
if (pVw)
pVw->GetEditWin().Invalidate();
pTmpFrm = pTmpFrm->GetNext(*pTmpFrm, pDocSh);
}
return bRet;
}
/*--------------------------------------------------------------------
Beschreibung: Serienbrief als Mail versenden
--------------------------------------------------------------------*/
BOOL SwNewDBMgr::MergeMailing(SwWrtShell* pSh)
{
//check if the doc is synchronized and contains at least one linked section
BOOL bSynchronizedDoc = pSh->IsLabelDoc() && pSh->GetSectionFmtCount() > 1;
BOOL bLoop = TRUE;
{
Reference< XColumnsSupplier > xColsSupp( pImpl->pMergeData->xResultSet, UNO_QUERY );
Reference <XNameAccess> xCols = xColsSupp->getColumns();
if(!xCols->hasByName(sEMailAddrFld))
return FALSE;
Any aCol = xCols->getByName(sEMailAddrFld);
Reference< XPropertySet > xColumnProp;
aCol >>= xColumnProp;
SfxDispatcher* pSfxDispatcher = pSh->GetView().GetViewFrame()->GetDispatcher();
if (!sSubject.Len()) // Kein leeres Subject wegen Automail (PB)
sSubject = ' ';
SfxStringItem aSubject(SID_MAIL_SUBJECT, sSubject);
SfxStringItem aText(SID_MAIL_TEXT, ' '); // Leerer Text ist nicht moeglich
SfxStringItem aAttached(SID_MAIL_ATTACH_FILE, sAttached);
SfxBoolItem aAttach(SID_MAIL_ATTACH, TRUE);
SwModuleOptions* pModOpt = SW_MOD()->GetModuleConfig();
BYTE nMailFmts = pModOpt->GetMailingFormats() | TXTFORMAT_ASCII; // Immer Ascii
SfxByteItem aTextFormats(SID_MAIL_TXTFORMAT, nMailFmts);
pSfxDispatcher->Execute( SID_SAVEDOC, SFX_CALLMODE_SYNCHRON|SFX_CALLMODE_RECORD);
if( !pSh->IsModified() )
{
// Beim Speichern wurde kein Abbruch gedrueckt
// neue DocShell erzeugen, alle gelinkten Bereiche embedden
// und unter temporaerem Namen wieder speichern.
String sTmpName;
const SfxFilter* pSfxFlt;
{
SfxMedium* pOrig = pSh->GetView().GetDocShell()->GetMedium();
String sMainURL( pOrig->GetURLObject().GetMainURL(
INetURLObject::NO_DECODE ));
pSfxFlt = SwIoSystem::GetFileFilter( sMainURL, ::aEmptyStr );
sTmpName = utl::TempFile::CreateTempName(0);
sTmpName = URIHelper::SmartRelToAbs(sTmpName);
SWUnoHelper::UCB_CopyFile( sMainURL, sTmpName );
}
String sAddress;
ULONG nDocNo = 1;
bCancel = FALSE;
PrintMonitor aPrtMonDlg(&pSh->GetView().GetEditWin(), TRUE);
aPrtMonDlg.aDocName.SetText(pSh->GetView().GetDocShell()->GetTitle(22));
aPrtMonDlg.aCancel.SetClickHdl(LINK(this, SwNewDBMgr, PrtCancelHdl));
if (!IsMergeSilent())
aPrtMonDlg.Show();
OfficeApplication* pOffApp = OFF_APP();
SfxRequest aReq( SID_OPENDOC, SFX_CALLMODE_SYNCHRON, pOffApp->GetPool() );
aReq.AppendItem( SfxStringItem( SID_FILE_NAME, sTmpName ));
if(pSfxFlt)
aReq.AppendItem( SfxStringItem( SID_FILTER_NAME, pSfxFlt->GetName() ));
aReq.AppendItem( SfxBoolItem( SID_HIDDEN, TRUE ) );
aReq.AppendItem( SfxStringItem( SID_REFERER, String::CreateFromAscii(URL_PREFIX_PRIV_SOFFICE )));
pOffApp->ExecuteSlot( aReq, pOffApp->SfxApplication::GetInterface());
const SfxPoolItem* pReturnValue = aReq.GetReturnValue();
if(pReturnValue)
{
// DocShell besorgen
SfxViewFrameItem* pVItem = (SfxViewFrameItem*)pReturnValue;
SwView* pView = (SwView*) pVItem->GetFrame()->GetViewShell();
SwWrtShell& rSh = pView->GetWrtShell();
pView->AttrChangedNotify( &rSh );//Damit SelectShell gerufen wird.
SwDoc* pDoc = rSh.GetDoc();
SwNewDBMgr* pOldDBMgr = pDoc->GetNewDBMgr();
pDoc->SetNewDBMgr( this );
pDoc->EmbedAllLinks();
String sTempStat(SW_RES(STR_DB_EMAIL));
long nStartRow, nEndRow;
do
{
nStartRow = pImpl->pMergeData ? pImpl->pMergeData->xResultSet->getRow() : 0;
if(UIUNDO_DELETE_INVISIBLECNTNT == rSh.GetUndoIds())
rSh.Undo();
rSh.ViewShell::UpdateFlds();
// alle versteckten Felder/Bereiche entfernen
rSh.RemoveInvisibleContent();
SfxFrameItem aFrame( SID_DOCFRAME, pVItem->GetFrame() );
SwDBFormatData aDBFormat;
sAddress = GetDBField( xColumnProp, aDBFormat);
if (!sAddress.Len())
sAddress = '_';
String sStat(sTempStat);
sStat += ' ';
sStat += String::CreateFromInt32( nDocNo++ );
aPrtMonDlg.aPrintInfo.SetText(sStat);
aPrtMonDlg.aPrinter.SetText( sAddress );
// Rechenzeit fuer EMail-Monitor:
for (USHORT i = 0; i < 25; i++)
Application::Reschedule();
sAddress.Insert(String::CreateFromAscii("mailto:"), 0);
SfxStringItem aRecipient( SID_MAIL_RECIPIENT, sAddress );
const SfxPoolItem* pRet = pSfxDispatcher->Execute(
SID_MAIL_SENDDOC,
SFX_CALLMODE_SYNCHRON|SFX_CALLMODE_RECORD,
&aRecipient, &aSubject, &aAttach, &aAttached,
&aText, &aTextFormats, &aFrame,
0L );
//this must be done here because pRet may be destroyed in Reschedule (DeleteOnIdle)
BOOL bBreak = pRet && !( (SfxBoolItem*)pRet )->GetValue();
// Rechenzeit fuer EMail-Monitor:
for (i = 0; i < 25; i++)
Application::Reschedule();
if ( bBreak )
break; // das Verschicken wurde unterbrochen
nEndRow = pImpl->pMergeData ? pImpl->pMergeData->xResultSet->getRow() : 0;
} while( !bCancel &&
(bSynchronizedDoc && (nStartRow != nEndRow) ? ExistsNextRecord() : ToNextMergeRecord()));
pDoc->SetNewDBMgr( pOldDBMgr );
pView->GetDocShell()->OwnerLock( FALSE );
}
// remove the temporary file
SWUnoHelper::UCB_DeleteFile( sTmpName );
SW_MOD()->SetView(&pSh->GetView());
}
nMergeType = DBMGR_INSERT;
}
return bLoop;
}
/*--------------------------------------------------------------------
Beschreibung: Serienbriefe als einzelne Dokumente speichern
--------------------------------------------------------------------*/
BOOL SwNewDBMgr::MergeMailFiles(SwWrtShell* pSh)
{
//check if the doc is synchronized and contains at least one linked section
BOOL bSynchronizedDoc = pSh->IsLabelDoc() && pSh->GetSectionFmtCount() > 1;
BOOL bLoop = TRUE;
Reference< XPropertySet > xColumnProp;
{
USHORT nColPos = 0;
BOOL bColumnName = sEMailAddrFld.Len() > 0;
if (bColumnName)
{
Reference< XColumnsSupplier > xColsSupp( pImpl->pMergeData->xResultSet, UNO_QUERY );
Reference <XNameAccess> xCols = xColsSupp->getColumns();
if(!xCols->hasByName(sEMailAddrFld))
return FALSE;
Any aCol = xCols->getByName(sEMailAddrFld);
aCol >>= xColumnProp;
}
SfxDispatcher* pSfxDispatcher = pSh->GetView().GetViewFrame()->GetDispatcher();
SwDocShell* pDocSh = pSh->GetView().GetDocShell();
pSfxDispatcher->Execute( pDocSh->HasName() ? SID_SAVEDOC : SID_SAVEASDOC, SFX_CALLMODE_SYNCHRON|SFX_CALLMODE_RECORD);
if( !pDocSh->IsModified() )
{
// Beim Speichern wurde kein Abbruch gedrueckt
SfxMedium* pOrig = pSh->GetView().GetDocShell()->GetMedium();
String sOldName(pOrig->GetURLObject().GetMainURL( INetURLObject::NO_DECODE ));
const SfxFilter* pSfxFlt = SwIoSystem::GetFileFilter(
sOldName, ::aEmptyStr );
String sAddress;
bCancel = FALSE;
PrintMonitor aPrtMonDlg(&pSh->GetView().GetEditWin());
aPrtMonDlg.aDocName.SetText(pSh->GetView().GetDocShell()->GetTitle(22));
aPrtMonDlg.aCancel.SetClickHdl(LINK(this, SwNewDBMgr, PrtCancelHdl));
if (!IsMergeSilent())
aPrtMonDlg.Show();
SwDocShell *pDocSh = pSh->GetView().GetDocShell();
// Progress, um KeyInputs zu unterbinden
SfxProgress aProgress(pDocSh, ::aEmptyStr, 1);
// Alle Dispatcher sperren
SfxViewFrame* pViewFrm = SfxViewFrame::GetFirst(pDocSh);
while (pViewFrm)
{
pViewFrm->GetDispatcher()->Lock(TRUE);
pViewFrm = SfxViewFrame::GetNext(*pViewFrm, pDocSh);
}
ULONG nDocNo = 1;
ULONG nCounter = 0;
String sExt( INetURLObject( sOldName ).GetExtension() );
sExt.InsertAscii(".", 0);
long nStartRow, nEndRow;
do
{
nStartRow = pImpl->pMergeData ? pImpl->pMergeData->xResultSet->getRow() : 0;
{
String sPath(sSubject);
if( bColumnName )
{
SwDBFormatData aDBFormat;
sAddress = GetDBField( xColumnProp, aDBFormat);
if (!sAddress.Len())
sAddress = '_';
sPath += sAddress;
nCounter = 0;
}
INetURLObject aEntry(sPath);
String sLeading(aEntry.GetBase());
aEntry.removeSegment();
sPath = aEntry.GetMainURL( INetURLObject::NO_DECODE );
utl::TempFile aTemp(sLeading,&sExt,&sPath );
if( !aTemp.IsValid() )
{
ErrorHandler::HandleError( ERRCODE_IO_NOTSUPPORTED );
bLoop = FALSE;
bCancel = TRUE;
}
else
{
INetURLObject aTempFile(aTemp.GetURL());
aPrtMonDlg.aPrinter.SetText( aTempFile.GetBase() );
String sStat(SW_RES(STR_STATSTR_LETTER)); // Brief
sStat += ' ';
sStat += String::CreateFromInt32( nDocNo++ );
aPrtMonDlg.aPrintInfo.SetText(sStat);
// Rechenzeit fuer Save-Monitor:
for (USHORT i = 0; i < 10; i++)
Application::Reschedule();
// Neues Dokument erzeugen und speichern
SfxObjectShellRef xDocSh( new SwDocShell( SFX_CREATE_MODE_INTERNAL ));
SfxMedium* pMed = new SfxMedium( sOldName, STREAM_STD_READ, TRUE );
pMed->SetFilter( pSfxFlt );
if (xDocSh->DoLoad(pMed))
{
//create a view frame for the document
SfxViewFrame* pFrame = SFX_APP()->CreateViewFrame( *xDocSh, 0, TRUE );
//request the layout calculation
((SwView*) pFrame->GetViewShell())->GetWrtShellPtr()->CalcLayout();
SwDoc* pDoc = ((SwDocShell*)(&xDocSh))->GetDoc();
SwNewDBMgr* pOldDBMgr = pDoc->GetNewDBMgr();
pDoc->SetNewDBMgr( this );
pDoc->UpdateFlds(0);
// alle versteckten Felder/Bereiche entfernen
pDoc->RemoveInvisibleContent();
// launch MailMergeEvent if required
const SwXMailMerge *pEvtSrc = GetMailMergeEvtSrc();
if (pEvtSrc)
{
Reference< XInterface > xRef( (XMailMergeBroadcaster *) pEvtSrc );
text::MailMergeEvent aEvt( xRef, xDocSh->GetModel() );
pEvtSrc->LaunchMailMergeEvent( aEvt );
}
SfxMedium* pDstMed = new SfxMedium(
aTempFile.GetMainURL( INetURLObject::NO_DECODE ),
STREAM_STD_READWRITE, TRUE );
pDstMed->SetFilter( pSfxFlt );
xDocSh->DoSaveAs(*pDstMed);
xDocSh->DoSaveCompleted(pDstMed);
if( xDocSh->GetError() )
{
// error message ??
ErrorHandler::HandleError( xDocSh->GetError() );
bCancel = TRUE;
bLoop = FALSE;
}
pDoc->SetNewDBMgr( pOldDBMgr );
}
xDocSh->DoClose();
}
}
nEndRow = pImpl->pMergeData ? pImpl->pMergeData->xResultSet->getRow() : 0;
} while( !bCancel &&
(bSynchronizedDoc && (nStartRow != nEndRow)? ExistsNextRecord() : ToNextMergeRecord()));
// Alle Dispatcher freigeben
pViewFrm = SfxViewFrame::GetFirst(pDocSh);
while (pViewFrm)
{
pViewFrm->GetDispatcher()->Lock(FALSE);
pViewFrm = SfxViewFrame::GetNext(*pViewFrm, pDocSh);
}
SW_MOD()->SetView(&pSh->GetView());
}
nMergeType = DBMGR_INSERT;
}
return bLoop;
}
/*--------------------------------------------------------------------
Beschreibung:
--------------------------------------------------------------------*/
IMPL_LINK_INLINE_START( SwNewDBMgr, PrtCancelHdl, Button *, pButton )
{
pButton->GetParent()->Hide();
bCancel = TRUE;
return 0;
}
IMPL_LINK_INLINE_END( SwNewDBMgr, PrtCancelHdl, Button *, pButton )
/*--------------------------------------------------------------------
Beschreibung: Numberformat der Spalte ermitteln und ggfs. in
den uebergebenen Formatter uebertragen
--------------------------------------------------------------------*/
ULONG SwNewDBMgr::GetColumnFmt( const String& rDBName,
const String& rTableName,
const String& rColNm,
SvNumberFormatter* pNFmtr,
long nLanguage )
{
ULONG nRet = 0;
if(pNFmtr)
{
Reference< XDataSource> xSource;
Reference< XConnection> xConnection;
sal_Bool bUseMergeData = sal_False;
if(pImpl->pMergeData &&
pImpl->pMergeData->sDataSource.equals(rDBName) && pImpl->pMergeData->sCommand.equals(rTableName))
{
xConnection = pImpl->pMergeData->xConnection;
Reference<XDataSource> xSource = SwNewDBMgr::getDataSourceAsParent(xConnection,rDBName);
bUseMergeData = sal_True;
}
if(!xConnection.is() || !xSource.is())
{
SwDSParam* pParam = FindDSConnection(rDBName, FALSE);
if(pParam && pParam->xConnection.is())
xConnection = pParam->xConnection;
else
{
rtl::OUString sDBName(rDBName);
xConnection = RegisterConnection( sDBName );
}
if(bUseMergeData)
pImpl->pMergeData->xConnection = xConnection;
}
Reference< XColumnsSupplier> xColsSupp = SwNewDBMgr::GetColumnSupplier(xConnection, rTableName);
if(xColsSupp.is())
{
Reference <XNameAccess> xCols;
try
{
xCols = xColsSupp->getColumns();
}
catch(Exception&)
{
DBG_ERROR("Exception in getColumns()")
}
if(!xCols.is() || !xCols->hasByName(rColNm))
return nRet;
Any aCol = xCols->getByName(rColNm);
Reference< XPropertySet > xColumn;
aCol >>= xColumn;
nRet = GetColumnFmt(xSource, xConnection, xColumn, pNFmtr, nLanguage);
}
else
nRet = pNFmtr->GetFormatIndex( NF_NUMBER_STANDARD, LANGUAGE_SYSTEM );
}
return nRet;
}
/* -----------------------------07.06.01 15:43--------------------------------
---------------------------------------------------------------------------*/
ULONG SwNewDBMgr::GetColumnFmt( Reference< XDataSource> xSource,
Reference< XConnection> xConnection,
Reference< XPropertySet> xColumn,
SvNumberFormatter* pNFmtr,
long nLanguage )
{
//JP 12.01.99: ggfs. das NumberFormat im Doc setzen
ULONG nRet = 0;
if(!xSource.is())
{
Reference<XChild> xChild(xConnection, UNO_QUERY);
xSource = Reference<XDataSource>(xChild->getParent(), UNO_QUERY);
}
if(xSource.is() && xConnection.is() && xColumn.is() && pNFmtr)
{
SvNumberFormatsSupplierObj* pNumFmt = new SvNumberFormatsSupplierObj( pNFmtr );
Reference< util::XNumberFormatsSupplier > xDocNumFmtsSupplier = pNumFmt;
Reference< XNumberFormats > xDocNumberFormats = xDocNumFmtsSupplier->getNumberFormats();
Reference< XNumberFormatTypes > xDocNumberFormatTypes(xDocNumberFormats, UNO_QUERY);
String sLanguage, sCountry;
::ConvertLanguageToIsoNames( (USHORT)nLanguage, sLanguage, sCountry );
Locale aLocale;
aLocale.Language = sLanguage;
aLocale.Country = sCountry;
//get the number formatter of the data source
Reference<XPropertySet> xSourceProps(xSource, UNO_QUERY);
Reference< XNumberFormats > xNumberFormats;
if(xSourceProps.is())
{
Any aFormats = xSourceProps->getPropertyValue(C2U("NumberFormatsSupplier"));
if(aFormats.hasValue())
{
Reference<XNumberFormatsSupplier> xSuppl;
aFormats >>= xSuppl;
if(xSuppl.is())
{
xNumberFormats = xSuppl->getNumberFormats();
}
}
}
Any aFormat = xColumn->getPropertyValue(C2U("FormatKey"));
if(aFormat.hasValue())
{
sal_Int32 nFmt;
aFormat >>= nFmt;
if(xNumberFormats.is())
{
try
{
Reference<XPropertySet> xNumProps = xNumberFormats->getByKey( nFmt );
Any aFormat = xNumProps->getPropertyValue(C2U("FormatString"));
Any aLocale = xNumProps->getPropertyValue(C2U("Locale"));
rtl::OUString sFormat;
aFormat >>= sFormat;
com::sun::star::lang::Locale aLoc;
aLocale >>= aLoc;
nFmt = xDocNumberFormats->queryKey( sFormat, aLoc, sal_False );
if(NUMBERFORMAT_ENTRY_NOT_FOUND == nFmt)
nFmt = xDocNumberFormats->addNew( sFormat, aLoc );
nRet = nFmt;
}
catch(Exception&)
{
DBG_ERROR("illegal number format key")
}
}
}
else
nRet = SwNewDBMgr::GetDbtoolsClient().getDefaultNumberFormat(xColumn, xDocNumberFormatTypes, aLocale);
}
return nRet;
}
/* -----------------------------17.07.00 09:47--------------------------------
---------------------------------------------------------------------------*/
sal_Int32 SwNewDBMgr::GetColumnType( const String& rDBName,
const String& rTableName,
const String& rColNm )
{
sal_Int32 nRet = DataType::SQLNULL;
SwDSParam* pParam = FindDSConnection(rDBName, FALSE);
Reference< XConnection> xConnection;
if(pParam && pParam->xConnection.is())
xConnection = pParam->xConnection;
else
{
rtl::OUString sDBName(rDBName);
xConnection = RegisterConnection( sDBName );
}
Reference< XColumnsSupplier> xColsSupp = SwNewDBMgr::GetColumnSupplier(xConnection, rTableName);
if(xColsSupp.is())
{
Reference <XNameAccess> xCols = xColsSupp->getColumns();
if(xCols->hasByName(rColNm))
{
Any aCol = xCols->getByName(rColNm);
Reference <XPropertySet> xCol;
aCol >>= xCol;
Any aType = xCol->getPropertyValue(C2S("Type"));
aType >>= nRet;
}
}
return nRet;
}
/* -----------------------------03.07.00 17:12--------------------------------
---------------------------------------------------------------------------*/
Reference< sdbc::XConnection> SwNewDBMgr::GetConnection(const String& rDataSource,
Reference<XDataSource>& rxSource)
{
Reference< sdbc::XConnection> xConnection;
Reference<XNameAccess> xDBContext;
Reference< XMultiServiceFactory > xMgr( ::comphelper::getProcessServiceFactory() );
if( xMgr.is() )
{
Reference<XInterface> xInstance = xMgr->createInstance( C2U( "com.sun.star.sdb.DatabaseContext" ));
xDBContext = Reference<XNameAccess>(xInstance, UNO_QUERY) ;
}
DBG_ASSERT(xDBContext.is(), "com.sun.star.sdb.DataBaseContext: service not available")
if(xDBContext.is())
{
try
{
if(xDBContext->hasByName(rDataSource))
{
Reference<XCompletedConnection> xComplConnection;
xDBContext->getByName(rDataSource) >>= xComplConnection;
rxSource = Reference<XDataSource>(xComplConnection, UNO_QUERY);
Reference< XInteractionHandler > xHandler(
xMgr->createInstance( C2U( "com.sun.star.sdb.InteractionHandler" )), UNO_QUERY);
xConnection = xComplConnection->connectWithCompletion( xHandler );
}
}
catch(Exception&) {}
}
return xConnection;
}
/* -----------------------------03.07.00 17:12--------------------------------
---------------------------------------------------------------------------*/
Reference< sdbcx::XColumnsSupplier> SwNewDBMgr::GetColumnSupplier(Reference<sdbc::XConnection> xConnection,
const String& rTableOrQuery,
BYTE eTableOrQuery)
{
Reference< sdbcx::XColumnsSupplier> xRet;
if(SW_DB_SELECT_QUERY != eTableOrQuery)
{
Reference<XTablesSupplier> xTSupplier = Reference<XTablesSupplier>(xConnection, UNO_QUERY);
if(xTSupplier.is())
{
Reference<XNameAccess> xTbls = xTSupplier->getTables();
if(xTbls->hasByName(rTableOrQuery))
try
{
Any aTable = xTbls->getByName(rTableOrQuery);
Reference<XPropertySet> xPropSet;
aTable >>= xPropSet;
xRet = Reference<XColumnsSupplier>(xPropSet, UNO_QUERY);
}
catch(Exception&){}
}
}
if(!xRet.is() && SW_DB_SELECT_QUERY != SW_DB_SELECT_TABLE)
{
Reference<XQueriesSupplier> xQSupplier = Reference<XQueriesSupplier>(xConnection, UNO_QUERY);
if(xQSupplier.is())
{
Reference<XNameAccess> xQueries = xQSupplier->getQueries();
if ( xQueries->hasByName(rTableOrQuery) )
try
{
xQueries->getByName(rTableOrQuery) >>= xRet;
}
catch(Exception&)
{
}
}
}
return xRet;
}
/* -----------------------------05.07.00 13:44--------------------------------
---------------------------------------------------------------------------*/
String SwNewDBMgr::GetDBField(Reference<XPropertySet> xColumnProps,
const SwDBFormatData& rDBFormatData,
double* pNumber)
{
Reference< XColumn > xColumn(xColumnProps, UNO_QUERY);
String sRet;
DBG_ASSERT(xColumn.is(), "SwNewDBMgr::::ImportDBField: illegal arguments")
if(!xColumn.is())
return sRet;
Any aType = xColumnProps->getPropertyValue(C2U("Type"));
sal_Int32 eDataType;
aType >>= eDataType;
switch(eDataType)
{
case DataType::CHAR:
case DataType::VARCHAR:
case DataType::LONGVARCHAR:
sRet = xColumn->getString();
break;
case DataType::BIT:
case DataType::TINYINT:
case DataType::SMALLINT:
case DataType::INTEGER:
case DataType::BIGINT:
case DataType::FLOAT:
case DataType::REAL:
case DataType::DOUBLE:
case DataType::NUMERIC:
case DataType::DECIMAL:
case DataType::DATE:
case DataType::TIME:
case DataType::TIMESTAMP:
{
// ::Date aTempDate(rDBFormatData.aNullDate.Day,
// rDBFormatData.aNullDate.Month, rDBFormatData.aNullDate.Year);
try
{
SwDbtoolsClient& aClient = SwNewDBMgr::GetDbtoolsClient();
sRet = aClient.getValue(
xColumnProps,
rDBFormatData.xFormatter,
rDBFormatData.aLocale,
rDBFormatData.aNullDate);
double fVal = xColumn->getDouble();
if (pNumber)
*pNumber = fVal;
}
catch(Exception& )
{
DBG_ERROR("exception caught")
}
}
break;
// case DataType::BINARY:
// case DataType::VARBINARY:
// case DataType::LONGVARBINARY:
// case DataType::SQLNULL:
// case DataType::OTHER:
// case DataType::OBJECT:
// case DataType::DISTINCT:
// case DataType::STRUCT:
// case DataType::ARRAY:
// case DataType::BLOB:
// case DataType::CLOB:
// case DataType::REF:
// default:
}
// if (pFormat)
// {
// SFX_ITEMSET_GET(*pCol, pFormatItem, SfxUInt32Item, SBA_DEF_FMTVALUE, sal_True);
// *pFormat = pFormatItem->GetValue();
// }
return sRet;
}
/* -----------------------------06.07.00 14:28--------------------------------
releases the merge data source table or query after merge is completed
---------------------------------------------------------------------------*/
void SwNewDBMgr::EndMerge()
{
DBG_ASSERT(bInMerge, "merge is not active")
bInMerge = FALSE;
delete pImpl->pMergeData;
pImpl->pMergeData = 0;
}
/* -----------------------------06.07.00 14:28--------------------------------
checks if a desired data source table or query is open
---------------------------------------------------------------------------*/
BOOL SwNewDBMgr::IsDataSourceOpen(const String& rDataSource,
const String& rTableOrQuery, sal_Bool bMergeOnly)
{
if(pImpl->pMergeData)
{
return !bMergeLock &&
rDataSource == (String)pImpl->pMergeData->sDataSource &&
rTableOrQuery == (String)pImpl->pMergeData->sCommand &&
pImpl->pMergeData->xResultSet.is();
}
else if(!bMergeOnly)
{
SwDBData aData;
aData.sDataSource = rDataSource;
aData.sCommand = rTableOrQuery;
aData.nCommandType = -1;
SwDSParam* pFound = FindDSData(aData, FALSE);
return (pFound && pFound->xResultSet.is());
}
return sal_False;
}
/* -----------------------------17.07.00 16:44--------------------------------
read column data a a specified position
---------------------------------------------------------------------------*/
BOOL SwNewDBMgr::GetColumnCnt(const String& rSourceName, const String& rTableName,
const String& rColumnName, sal_uInt32 nAbsRecordId,
long nLanguage,
String& rResult, double* pNumber)
{
BOOL bRet = FALSE;
SwDSParam* pFound = 0;
//check if it's the merge data source
if(pImpl->pMergeData &&
rSourceName == (String)pImpl->pMergeData->sDataSource &&
rTableName == (String)pImpl->pMergeData->sCommand)
{
pFound = pImpl->pMergeData;
}
else
{
SwDBData aData;
aData.sDataSource = rSourceName;
aData.sCommand = rTableName;
aData.nCommandType = -1;
pFound = FindDSData(aData, FALSE);
}
//check validity of supplied record Id
if(pFound->aSelection.getLength())
{
//the destination has to be an element of the selection
const Any* pSelection = pFound->aSelection.getConstArray();
sal_Bool bFound = sal_False;
for(sal_Int32 nPos = 0; !bFound && nPos < pFound->aSelection.getLength(); nPos++)
{
sal_Int32 nSelection;
pSelection[nPos] >>= nSelection;
if(nSelection == static_cast<sal_Int32>(nAbsRecordId))
bFound = sal_True;
}
if(!bFound)
return FALSE;
}
if(pFound && pFound->xResultSet.is() && !pFound->bAfterSelection)
{
sal_Int32 nOldRow = 0;
try
{
nOldRow = pFound->xResultSet->getRow();
}
catch(const Exception& )
{
return FALSE;
}
//position to the desired index
BOOL bMove = TRUE;
if ( nOldRow != static_cast<sal_Int32>(nAbsRecordId) )
bMove = lcl_MoveAbsolute(pFound, nAbsRecordId);
if(bMove)
{
bRet = lcl_GetColumnCnt(pFound, rColumnName, nLanguage, rResult, pNumber);
}
if ( nOldRow != static_cast<sal_Int32>(nAbsRecordId) )
bMove = lcl_MoveAbsolute(pFound, nOldRow);
}
return bRet;
}
/* -----------------------------06.07.00 16:47--------------------------------
reads the column data at the current position
---------------------------------------------------------------------------*/
BOOL SwNewDBMgr::GetMergeColumnCnt(const String& rColumnName, USHORT nLanguage,
String &rResult, double *pNumber, sal_uInt32 *pFormat)
{
if(!pImpl->pMergeData || !pImpl->pMergeData->xResultSet.is() || pImpl->pMergeData->bAfterSelection )
{
rResult.Erase();
return FALSE;
}
BOOL bRet = lcl_GetColumnCnt(pImpl->pMergeData, rColumnName, nLanguage, rResult, pNumber);
return bRet;
}
/* -----------------------------07.07.00 14:28--------------------------------
---------------------------------------------------------------------------*/
BOOL SwNewDBMgr::ToNextMergeRecord()
{
DBG_ASSERT(pImpl->pMergeData && pImpl->pMergeData->xResultSet.is(), "no data source in merge")
return ToNextRecord(pImpl->pMergeData);
}
/* -----------------------------10.07.01 14:28--------------------------------
---------------------------------------------------------------------------*/
BOOL SwNewDBMgr::ToNextRecord(
const String& rDataSource, const String& rCommand, sal_Int32 nCommandType)
{
SwDSParam* pFound = 0;
BOOL bRet = TRUE;
if(pImpl->pMergeData &&
rDataSource == (String)pImpl->pMergeData->sDataSource &&
rCommand == (String)pImpl->pMergeData->sCommand)
pFound = pImpl->pMergeData;
else
{
SwDBData aData;
aData.sDataSource = rDataSource;
aData.sCommand = rCommand;
aData.nCommandType = -1;
pFound = FindDSData(aData, FALSE);
}
return ToNextRecord(pFound);
}
/* -----------------------------10.07.01 14:38--------------------------------
---------------------------------------------------------------------------*/
BOOL SwNewDBMgr::ToNextRecord(SwDSParam* pParam)
{
BOOL bRet = TRUE;
if(!pParam || !pParam->xResultSet.is() || pParam->bEndOfDB ||
(pParam->aSelection.getLength() && pParam->aSelection.getLength() <= pParam->nSelectionIndex))
{
if(pParam)
pParam->CheckEndOfDB();
return FALSE;
}
try
{
if(pParam->aSelection.getLength())
{
sal_Int32 nPos = 0;
pParam->aSelection.getConstArray()[ pParam->nSelectionIndex++ ] >>= nPos;
pParam->bEndOfDB = !pParam->xResultSet->absolute( nPos );
pParam->CheckEndOfDB();
bRet = !pParam->bEndOfDB;
if(pParam->nSelectionIndex >= pParam->aSelection.getLength())
pParam->bEndOfDB = TRUE;
}
else
{
pParam->bEndOfDB = !pParam->xResultSet->next();
pParam->CheckEndOfDB();
bRet = !pParam->bEndOfDB;
++pParam->nSelectionIndex;
}
}
catch(Exception&)
{
}
return bRet;
}
/* -----------------------------13.07.00 17:23--------------------------------
synchronized labels contain a next record field at their end
to assure that the next page can be created in mail merge
the cursor position must be validated
---------------------------------------------------------------------------*/
BOOL SwNewDBMgr::ExistsNextRecord() const
{
return pImpl->pMergeData && !pImpl->pMergeData->bEndOfDB;
}
/* -----------------------------13.07.00 10:41--------------------------------
---------------------------------------------------------------------------*/
sal_uInt32 SwNewDBMgr::GetSelectedRecordId()
{
sal_uInt32 nRet = 0;
DBG_ASSERT(pImpl->pMergeData && pImpl->pMergeData->xResultSet.is(), "no data source in merge")
if(!pImpl->pMergeData || !pImpl->pMergeData->xResultSet.is())
return FALSE;
try
{
nRet = pImpl->pMergeData->xResultSet->getRow();
}
catch(Exception& )
{
}
return nRet;
}
/* -----------------------------13.07.00 10:58--------------------------------
---------------------------------------------------------------------------*/
sal_Bool SwNewDBMgr::ToRecordId(sal_Int32 nSet)
{
DBG_ASSERT(pImpl->pMergeData && pImpl->pMergeData->xResultSet.is(), "no data source in merge")
if(!pImpl->pMergeData || !pImpl->pMergeData->xResultSet.is()|| nSet < 0)
return FALSE;
sal_Bool bRet = FALSE;
sal_Int32 nAbsPos = nSet;
if(nAbsPos >= 0)
{
bRet = lcl_MoveAbsolute(pImpl->pMergeData, nAbsPos);
pImpl->pMergeData->bEndOfDB = !bRet;
pImpl->pMergeData->CheckEndOfDB();
}
return bRet;
}
/* -----------------------------17.07.00 14:50--------------------------------
---------------------------------------------------------------------------*/
void lcl_ExtractMembers(const String& rDBName, String& sSource, String& sTable, String& sStatement)
{
sSource = rDBName.GetToken(0, DB_DELIM);
sTable = rDBName.GetToken(0).GetToken(1, DB_DELIM);
sal_uInt16 nPos;
if ((nPos = rDBName.Search(';')) != STRING_NOTFOUND)
sStatement = rDBName.Copy(nPos + 1);
}
/* -----------------------------17.07.00 14:17--------------------------------
---------------------------------------------------------------------------*/
BOOL SwNewDBMgr::OpenDataSource(const String& rDataSource, const String& rTableOrQuery, sal_Int32 nCommandType)
{
SwDBData aData;
aData.sDataSource = rDataSource;
aData.sCommand = rTableOrQuery;
aData.nCommandType = nCommandType;
SwDSParam* pFound = FindDSData(aData, TRUE);
Reference< XDataSource> xSource;
if(pFound->xResultSet.is())
return TRUE;
SwDSParam* pParam = FindDSConnection(rDataSource, FALSE);
Reference< XConnection> xConnection;
if(pParam && pParam->xConnection.is())
pFound->xConnection = pParam->xConnection;
else
{
rtl::OUString sDataSource(rDataSource);
pFound->xConnection = RegisterConnection( sDataSource );
}
if(pFound->xConnection.is())
{
try
{
Reference< sdbc::XDatabaseMetaData > xMetaData = pFound->xConnection->getMetaData();
try
{
pFound->bScrollable = xMetaData
->supportsResultSetType((sal_Int32)ResultSetType::SCROLL_INSENSITIVE);
}
catch(Exception&)
{
//#98373# DB driver may not be ODBC 3.0 compliant
pFound->bScrollable = TRUE;
}
pFound->xStatement = pFound->xConnection->createStatement();
rtl::OUString aQuoteChar = xMetaData->getIdentifierQuoteString();
rtl::OUString sStatement(C2U("SELECT * FROM "));
sStatement = C2U("SELECT * FROM ");
sStatement += aQuoteChar;
sStatement += rTableOrQuery;
sStatement += aQuoteChar;
pFound->xResultSet = pFound->xStatement->executeQuery( sStatement );
//after executeQuery the cursor must be positioned
pFound->bEndOfDB = !pFound->xResultSet->next();
pFound->bAfterSelection = sal_False;
pFound->CheckEndOfDB();
++pFound->nSelectionIndex;
}
catch (Exception&)
{
pFound->xResultSet = 0;
pFound->xStatement = 0;
pFound->xConnection = 0;
}
}
return pFound->xResultSet.is();
}
/* -----------------------------14.08.2001 10:26------------------------------
---------------------------------------------------------------------------*/
Reference< XConnection> SwNewDBMgr::RegisterConnection(rtl::OUString& rDataSource)
{
SwDSParam* pFound = SwNewDBMgr::FindDSConnection(rDataSource, TRUE);
Reference< XDataSource> xSource;
if(!pFound->xConnection.is())
{
pFound->xConnection = SwNewDBMgr::GetConnection(rDataSource, xSource );
try
{
Reference<XComponent> xComponent(pFound->xConnection, UNO_QUERY);
if(xComponent.is())
xComponent->addEventListener(pImpl->xDisposeListener);
}
catch(Exception&)
{
}
}
return pFound->xConnection;
}
/* -----------------------------17.07.00 15:55--------------------------------
---------------------------------------------------------------------------*/
sal_uInt32 SwNewDBMgr::GetSelectedRecordId(
const String& rDataSource, const String& rTableOrQuery, sal_Int32 nCommandType)
{
sal_uInt32 nRet = -1;
//check for merge data source first
if(pImpl->pMergeData && rDataSource == (String)pImpl->pMergeData->sDataSource &&
rTableOrQuery == (String)pImpl->pMergeData->sCommand &&
(nCommandType == -1 || nCommandType == pImpl->pMergeData->nCommandType) &&
pImpl->pMergeData->xResultSet.is())
nRet = GetSelectedRecordId();
else
{
SwDBData aData;
aData.sDataSource = rDataSource;
aData.sCommand = rTableOrQuery;
aData.nCommandType = nCommandType;
SwDSParam* pFound = FindDSData(aData, FALSE);
if(pFound && pFound->xResultSet.is())
{
try
{ //if a selection array is set the current row at the result set may not be set yet
if(pFound->aSelection.getLength())
{
sal_Int32 nSelIndex = pFound->nSelectionIndex;
if(nSelIndex >= pFound->aSelection.getLength())
nSelIndex = pFound->aSelection.getLength() -1;
pFound->aSelection.getConstArray()[nSelIndex] >>= nRet;
}
else
nRet = pFound->xResultSet->getRow();
}
catch(Exception&){}
}
}
return nRet;
}
/* -----------------------------17.07.00 14:18--------------------------------
close all data sources - after fields were updated
---------------------------------------------------------------------------*/
void SwNewDBMgr::CloseAll(BOOL bIncludingMerge)
{
//the only thing done here is to reset the selection index
//all connections stay open
for(USHORT nPos = 0; nPos < aDataSourceParams.Count(); nPos++)
{
SwDSParam* pParam = aDataSourceParams[nPos];
if(bIncludingMerge || pParam != pImpl->pMergeData)
{
pParam->nSelectionIndex = 0;
pParam->bAfterSelection = sal_False;
try
{
if(!bInMerge && pParam->xResultSet.is())
pParam->xResultSet->first();
}
catch(Exception& )
{}
}
}
}
/* -----------------------------17.07.00 14:54--------------------------------
---------------------------------------------------------------------------*/
SwDSParam* SwNewDBMgr::FindDSData(const SwDBData& rData, BOOL bCreate)
{
SwDSParam* pFound = 0;
for(USHORT nPos = aDataSourceParams.Count(); nPos; nPos--)
{
SwDSParam* pParam = aDataSourceParams[nPos - 1];
if(rData.sDataSource == pParam->sDataSource &&
rData.sCommand == pParam->sCommand &&
(rData.nCommandType == -1 || rData.nCommandType == pParam->nCommandType ||
(bCreate && pParam->nCommandType == -1)))
{
//#94779# calls from the calculator may add a connection with an invalid commandtype
//later added "real" data base connections have to re-use the already available
//DSData and set the correct CommandType
if(bCreate && pParam->nCommandType == -1)
pParam->nCommandType = rData.nCommandType;
pFound = pParam;
break;
}
}
if(bCreate)
{
if(!pFound)
{
pFound = new SwDSParam(rData);
aDataSourceParams.Insert(pFound, aDataSourceParams.Count());
try
{
Reference<XComponent> xComponent(pFound->xConnection, UNO_QUERY);
if(xComponent.is())
xComponent->addEventListener(pImpl->xDisposeListener);
}
catch(Exception&)
{
}
}
}
return pFound;
}
/* -----------------------------14.08.2001 10:27------------------------------
---------------------------------------------------------------------------*/
SwDSParam* SwNewDBMgr::FindDSConnection(const rtl::OUString& rDataSource, BOOL bCreate)
{
SwDSParam* pFound = 0;
for(USHORT nPos = 0; nPos < aDataSourceParams.Count(); nPos++)
{
SwDSParam* pParam = aDataSourceParams[nPos];
if(rDataSource == pParam->sDataSource)
{
pFound = pParam;
break;
}
}
if(bCreate && !pFound)
{
SwDBData aData;
aData.sDataSource = rDataSource;
pFound = new SwDSParam(aData);
aDataSourceParams.Insert(pFound, aDataSourceParams.Count());
try
{
Reference<XComponent> xComponent(pFound->xConnection, UNO_QUERY);
if(xComponent.is())
xComponent->addEventListener(pImpl->xDisposeListener);
}
catch(Exception&)
{
}
}
return pFound;
}
/* -----------------------------17.07.00 14:31--------------------------------
rDBName: <Source> + DB_DELIM + <Table>; + <Statement>
---------------------------------------------------------------------------*/
void SwNewDBMgr::AddDSData(const SwDBData& rData, long nSelStart, long nSelEnd)
{
SwDSParam* pFound = FindDSData(rData, TRUE);
if(nSelStart > 0)
{
if(nSelEnd < nSelStart)
{
sal_uInt32 nZw = nSelEnd;
nSelEnd = nSelStart;
nSelStart = nZw;
}
pFound->aSelection.realloc(nSelEnd - nSelStart + 1);
Any* pSelection = pFound->aSelection.getArray();
for (long i = nSelStart; i <= nSelEnd; i++, ++pSelection)
*pSelection <<= i;
}
else
pFound->aSelection.realloc(0);
}
/* -----------------------------17.07.00 14:31--------------------------------
---------------------------------------------------------------------------*/
void SwNewDBMgr::GetDSSelection(const SwDBData& rData, long& rSelStart, long& rSelEnd)
{
SwDSParam* pFound = FindDSData(rData, FALSE);
if(!pFound || !pFound->aSelection.getLength())
rSelStart = -1L;
else
{
pFound->aSelection.getConstArray()[0] >>= rSelStart;
pFound->aSelection.getConstArray()[pFound->aSelection.getLength() - 1] >>= rSelEnd;
}
}
/* -----------------------------17.07.00 14:34--------------------------------
---------------------------------------------------------------------------*/
const SwDBData& SwNewDBMgr::GetAddressDBName()
{
return SW_MOD()->GetDBConfig()->GetAddressSource();
}
/* -----------------------------18.07.00 13:13--------------------------------
---------------------------------------------------------------------------*/
Sequence<rtl::OUString> SwNewDBMgr::GetExistingDatabaseNames()
{
Reference<XNameAccess> xDBContext;
Reference< XMultiServiceFactory > xMgr( ::comphelper::getProcessServiceFactory() );
if( xMgr.is() )
{
Reference<XInterface> xInstance = xMgr->createInstance( C2U( "com.sun.star.sdb.DatabaseContext" ));
xDBContext = Reference<XNameAccess>(xInstance, UNO_QUERY) ;
}
if(xDBContext.is())
{
return xDBContext->getElementNames();
}
return Sequence<rtl::OUString>();
}
/* -----------------------------10.11.00 17:10--------------------------------
---------------------------------------------------------------------------*/
void SwNewDBMgr::ExecuteFormLetter( SwWrtShell& rSh,
const Sequence<PropertyValue>& rProperties,
BOOL bWithDataSourceBrowser)
{
//prevent second call
if(pImpl->pMergeDialog)
return ;
rtl::OUString sDataSource, sDataTableOrQuery;
Sequence<Any> aSelection;
sal_Int32 nSelectionPos = 0;
sal_Int32 nResultSetIdx = -1;
sal_Int16 nCmdType = CommandType::TABLE;
Reference< XConnection> xConnection;
ODataAccessDescriptor aDescriptor(rProperties);
aDescriptor[daDataSource] >>= sDataSource;
aDescriptor[daCommand] >>= sDataTableOrQuery;
aDescriptor[daCommandType] >>= nCmdType;
if ( aDescriptor.has(daSelection) )
aDescriptor[daSelection] >>= aSelection;
if ( aDescriptor.has(daConnection) )
aDescriptor[daConnection] >>= xConnection;
if(!sDataSource.getLength() || !sDataTableOrQuery.getLength())
{
DBG_ERROR("PropertyValues missing or unset")
return;
}
pImpl->pMergeDialog = new SwMailMergeDlg(
&rSh.GetView().GetViewFrame()->GetWindow(), rSh,
sDataSource,
sDataTableOrQuery,
nCmdType,
xConnection,
bWithDataSourceBrowser ? 0 : &aSelection );
if(pImpl->pMergeDialog->Execute() == RET_OK)
{
aDescriptor[daSelection] <<= pImpl->pMergeDialog->GetSelection();
Reference<XResultSet> xResSet = pImpl->pMergeDialog->GetResultSet();
if(xResSet.is())
aDescriptor[daCursor] <<= xResSet;
OFF_APP()->NotifyEvent(SfxEventHint(SW_EVENT_MAIL_MERGE, rSh.GetView().GetViewFrame()->GetObjectShell()));
MergeNew(pImpl->pMergeDialog->GetMergeType(), rSh, aDescriptor);
OFF_APP()->NotifyEvent(SfxEventHint(SW_EVENT_MAIL_MERGE_END, rSh.GetView().GetViewFrame()->GetObjectShell()));
// reset the cursor inside
xResSet = NULL;
aDescriptor[daCursor] <<= xResSet;
}
DELETEZ(pImpl->pMergeDialog);
}
/* -----------------------------13.11.00 08:20--------------------------------
---------------------------------------------------------------------------*/
void SwNewDBMgr::InsertText(SwWrtShell& rSh,
const Sequence< PropertyValue>& rProperties)
{
rtl::OUString sDataSource, sDataTableOrQuery;
Reference<XResultSet> xResSet;
Sequence<Any> aSelection;
BOOL bHasSelectionProperty = FALSE;
sal_Int32 nSelectionPos = 0;
sal_Int16 nCmdType = CommandType::TABLE;
const PropertyValue* pValues = rProperties.getConstArray();
Reference< XConnection> xConnection;
for(sal_Int32 nPos = 0; nPos < rProperties.getLength(); nPos++)
{
if(pValues[nPos].Name.equalsAsciiL(RTL_CONSTASCII_STRINGPARAM(cDataSourceName)))
pValues[nPos].Value >>= sDataSource;
else if(pValues[nPos].Name.equalsAsciiL(RTL_CONSTASCII_STRINGPARAM(cCommand)))
pValues[nPos].Value >>= sDataTableOrQuery;
else if(pValues[nPos].Name.equalsAsciiL(RTL_CONSTASCII_STRINGPARAM(cCursor)))
pValues[nPos].Value >>= xResSet;
else if(pValues[nPos].Name.equalsAsciiL(RTL_CONSTASCII_STRINGPARAM(cSelection)))
{
bHasSelectionProperty = TRUE;
nSelectionPos = nPos;
pValues[nPos].Value >>= aSelection;
}
else if(pValues[nPos].Name.equalsAsciiL(RTL_CONSTASCII_STRINGPARAM(cCommandType)))
pValues[nPos].Value >>= nCmdType;
else if(pValues[nPos].Name.equalsAsciiL(RTL_CONSTASCII_STRINGPARAM(cActiveConnection)))
pValues[nPos].Value >>= xConnection;
}
if(!sDataSource.getLength() || !sDataTableOrQuery.getLength() || !xResSet.is())
{
DBG_ERROR("PropertyValues missing or unset")
return;
}
Reference< XMultiServiceFactory > xMgr( ::comphelper::getProcessServiceFactory() );
Reference<XDataSource> xSource;
Reference<XChild> xChild(xConnection, UNO_QUERY);
if(xChild.is())
xSource = Reference<XDataSource>(xChild->getParent(), UNO_QUERY);
if(!xSource.is())
xSource = SwNewDBMgr::GetDbtoolsClient().getDataSource(sDataSource, xMgr);
Reference< XColumnsSupplier > xColSupp( xResSet, UNO_QUERY );
SwDBData aDBData;
aDBData.sDataSource = sDataSource;
aDBData.sCommand = sDataTableOrQuery;
aDBData.nCommandType = nCmdType;
SwInsertDBColAutoPilot *pDlg = new SwInsertDBColAutoPilot(
rSh.GetView(),
xSource,
xColSupp,
aDBData );
if( RET_OK == pDlg->Execute() )
{
rtl::OUString sDummy;
if(!xConnection.is())
xConnection = xSource->getConnection(sDummy, sDummy);
try
{
pDlg->DataToDoc( aSelection , xSource, xConnection, xResSet);
}
catch(Exception& )
{
DBG_ERROR("exception caught")
}
}
delete pDlg;
}
/* -----------------------------30.08.2001 12:00------------------------------
---------------------------------------------------------------------------*/
SwDbtoolsClient* SwNewDBMgr::pDbtoolsClient = NULL;
SwDbtoolsClient& SwNewDBMgr::GetDbtoolsClient()
{
if ( !pDbtoolsClient )
pDbtoolsClient = new SwDbtoolsClient;
return *pDbtoolsClient;
}
/* -----------------13.05.2003 15:34-----------------
--------------------------------------------------*/
void SwNewDBMgr::RemoveDbtoolsClient()
{
delete pDbtoolsClient;
pDbtoolsClient = 0;
}
/* -----------------------------20.08.2002 12:00------------------------------
---------------------------------------------------------------------------*/
Reference<XDataSource> SwNewDBMgr::getDataSourceAsParent(const Reference< XConnection>& _xConnection,const ::rtl::OUString& _sDataSourceName)
{
Reference<XDataSource> xSource;
try
{
Reference<XChild> xChild(_xConnection, UNO_QUERY);
if ( xChild.is() )
xSource = Reference<XDataSource>(xChild->getParent(), UNO_QUERY);
if ( !xSource.is() )
xSource = SwNewDBMgr::GetDbtoolsClient().getDataSource(_sDataSourceName, ::comphelper::getProcessServiceFactory());
}
catch(const Exception&)
{
DBG_ERROR("exception in getDataSourceAsParent caught")
}
return xSource;
}
/* -----------------------------20.08.2002 12:00------------------------------
---------------------------------------------------------------------------*/
Reference<XResultSet> SwNewDBMgr::createCursor(const ::rtl::OUString& _sDataSourceName,
const ::rtl::OUString& _sCommand,
sal_Int32 _nCommandType,
const Reference<XConnection>& _xConnection
)
{
Reference<XResultSet> xResultSet;
try
{
Reference< XMultiServiceFactory > xMgr( ::comphelper::getProcessServiceFactory() );
if( xMgr.is() )
{
Reference<XInterface> xInstance = xMgr->createInstance(
C2U( "com.sun.star.sdb.RowSet" ));
Reference <XPropertySet> xRowSetPropSet(xInstance, UNO_QUERY);
if(xRowSetPropSet.is())
{
xRowSetPropSet->setPropertyValue(C2U("DataSourceName"), makeAny(_sDataSourceName));
xRowSetPropSet->setPropertyValue(C2U("ActiveConnection"), makeAny(_xConnection));
xRowSetPropSet->setPropertyValue(C2U("Command"), makeAny(_sCommand));
xRowSetPropSet->setPropertyValue(C2U("CommandType"), makeAny(_nCommandType));
Reference< XCompletedExecution > xRowSet(xInstance, UNO_QUERY);
if ( xRowSet.is() )
{
Reference< XInteractionHandler > xHandler(xMgr->createInstance(C2U("com.sun.star.sdb.InteractionHandler")), UNO_QUERY);
xRowSet->executeWithCompletion(xHandler);
}
xResultSet = Reference<XResultSet>(xRowSet, UNO_QUERY);
}
}
}
catch(const Exception&)
{
DBG_ASSERT(0,"Catched exception while creating a new RowSet!");
}
return xResultSet;
}
/* -----------------09.12.2002 12:38-----------------
*
* --------------------------------------------------*/
SwConnectionDisposedListener_Impl::SwConnectionDisposedListener_Impl(SwNewDBMgr& rMgr) :
rDBMgr(rMgr)
{};
/* -----------------09.12.2002 12:39-----------------
*
* --------------------------------------------------*/
SwConnectionDisposedListener_Impl::~SwConnectionDisposedListener_Impl()
{};
/* -----------------09.12.2002 12:39-----------------
*
* --------------------------------------------------*/
void SwConnectionDisposedListener_Impl::disposing( const EventObject& rSource )
throw (RuntimeException)
{
::vos::OGuard aGuard(Application::GetSolarMutex());
Reference<XConnection> xSource(rSource.Source, UNO_QUERY);
for(USHORT nPos = rDBMgr.aDataSourceParams.Count(); nPos; nPos--)
{
SwDSParam* pParam = rDBMgr.aDataSourceParams[nPos - 1];
if(pParam->xConnection.is() &&
(xSource == pParam->xConnection))
{
rDBMgr.aDataSourceParams.DeleteAndDestroy(nPos - 1);
}
}
}