2009-06-29 20:53:25 +0200 fs r273484 : #i103138# Rectangle conversion 2009-06-29 20:51:50 +0200 fs r273483 : #i103138# yet more refactoring, now also setting the proper zoom level at the proper point in time 2009-06-29 13:40:26 +0200 fs r273470 : added svn:ignore to ignore output paths 2009-06-29 10:08:54 +0200 fs r273455 : #i103138# refactored the code for positioning/zooming the control Basically, we now allow adjustControlGeometry_throw (formerly known as positionControl_throw and setControlZoom) to take an additional ViewTransformation parameter, describing the transformation to obtain the actual control position/size. Consequently, positionControl itself also allows for a ViewTransformation parameter. This has become necessary since during painting, the device which we created our control for might not necessarily have a proper MapMode set. In this case, if we would use this map mode for calculating the control's position/size, this would lead to wrong results. Note that this problem was introduced by the fix for #i101398#: During the fix, we postponed the control creation to a later time (when it is really needed). At this later time, the MapMode at the device is broken, at the earlier time where we formerly crearted the control (createPrimitive2DSequence), it is not yet broken. Whether or not the MapMode is defined as "broken" might depend on one's point of view, however ... I consider it broken, since: - we need the map mode to obtain the proper zoom level, which is to be forwarded to the control - there are scenarios where the MapMode is *not* set to MAP_PIXEL (in those scenarios, everything works fine), and there are scenarios where it *is* set to MAP_PIXEL (in those the bug 103138 appears). It somehow feels wrong that one cannot rely on the device's map mode this way, but on the other hand one has no possibility to obtain the current zoom by other means. Note that one issue (still to be submitted) is left: In the page pane of a Draw/Impress document, controls have a wrong text size. This is because in this pane, the above-mentioned "broken" map mode is used, which means the controls have a zoom of "1:1" set, which is wrong here. 2009-06-25 13:41:35 +0200 msc r273380 : #100000# the tabs changed die to new properties 2009-06-24 12:42:40 +0200 msc r273330 : #102082# remove issue warning 2009-06-22 10:43:14 +0200 fs r273201 : createPrimitive2DSequence: care for being disposed 2009-06-18 12:35:13 +0200 oj r273109 : #i102305# make nooptfiles for gcc 2009-06-17 12:14:37 +0200 oj r273056 : #i102305# fix for linux 2009-06-17 07:20:22 +0200 oj r273046 : #i102305# move ValueTransfer into the for loop to avoid a crash under Linux 2009-06-17 07:17:28 +0200 oj r273045 : #i102305# use varchar 2009-06-15 14:11:27 +0200 fs r272983 : added since tag 2009-06-15 12:11:39 +0200 oj r272973 : #i102305# SAL_DLLPUBLIC_EXPORT inserted 2009-06-15 11:08:53 +0200 fs r272969 : #i10000# 2009-06-15 09:25:13 +0200 fs r272963 : merging fix for P1 issue #i102701# 2009-06-11 11:31:24 +0200 fs r272858 : #i10000# copied the fix which before the rebase was done in ../dialog/macropg.src 2009-06-11 09:38:14 +0200 fs r272846 : CWS-TOOLING: rebase CWS dba32c to trunk@272827 (milestone: DEV300:m50) 2009-06-02 09:53:10 +0200 fs r272483 : #i10000# 2009-05-29 15:55:03 +0200 fs r272465 : #i100818# 2009-05-29 12:58:43 +0200 fs r272452 : don't apply comphelper::getString on possibly VOID any 2009-05-29 10:38:35 +0200 oj r272437 : #i101519# handle where condition 2009-05-29 09:53:39 +0200 fs r272434 : #i100818# call into releaseStubs /without/ locked GlobalMutex 2009-05-28 07:53:44 +0200 oj r272375 : #i101369# parse tree changed 2009-05-27 14:53:36 +0200 fs r272347 : #i10000# 2009-05-27 09:29:15 +0200 oj r272327 : #i101626# check for double before hard cast 2009-05-27 09:13:58 +0200 oj r272326 : #i101626# handle void correctly 2009-05-27 08:04:39 +0200 oj r272321 : #i102256# wrong method signature used 2009-05-27 07:55:52 +0200 oj r272320 : #i101519# look up parameter typ if used in function 2009-05-27 06:49:07 +0200 oj r272319 : #i101519# set parameter from rowset as well 2009-05-26 13:30:56 +0200 oj r272297 : #i101987# impl XBatchExecution 2009-05-26 12:44:34 +0200 oj r272293 : #i101700# check if group is not set 2009-05-26 12:16:53 +0200 oj r272290 : #i101369# resolved some reduce7reduce problems with boolean_term and search_condition 2009-05-26 12:12:42 +0200 oj r272289 : #i101369# fix for or on one line criteria 2009-05-25 16:02:25 +0200 fs r272257 : #i999704# +PROPERTY_MOUSE_WHEEL_BEHAVIOR 2009-05-25 16:01:55 +0200 fs r272256 : merging the changes from CWS dba32b herein 2009-05-25 15:49:57 +0200 fs r272254 : #i999704# 2009-05-25 15:32:57 +0200 fs r272252 : #i99704# grid columns also to respect the MouseWheelBehavior property 2009-05-25 15:23:43 +0200 fs r272251 : don't pass empty Anys to ::comphelper::getString 2009-05-25 14:48:43 +0200 fs r272248 : merged changes from CWS dba32b herein 2009-05-25 14:44:40 +0200 fs r272247 : #i99704# support new MouseWheelBehavior property 2009-05-25 14:43:18 +0200 fs r272246 : #i99704# WheelWithoutFocus (peer property) superseded by MouseWheelBehavior (model property) 2009-05-25 14:41:03 +0200 fs r272245 : #i99704# no need to set the mouse wheel behavior at the peer, this is now a model property, having the right default 2009-05-25 14:39:31 +0200 fs r272243 : removed dead import 2009-05-25 14:35:36 +0200 fs r272242 : the new EnableVisible doesn't make sense for grid columns 2009-05-25 14:34:33 +0200 fs r272241 : #i99704# +MouseWheelBehavior - allow to enable/disable the mouse wheel for the control, or make it focus-dependent 2009-05-25 14:26:11 +0200 fs r272240 : #i99704# change MouseSettings wheel flag (NoWheelActionWithoutFocus) to a three-state option, allowing to completely ignore the mouse wheel 2009-05-23 21:35:59 +0200 fs r272213 : localize 'sub component opened/closed' event 2009-05-22 21:42:47 +0200 fs r272211 : #i102003# 2009-05-22 21:42:20 +0200 fs r272210 : grammar 2009-05-22 21:36:10 +0200 fs r272209 : #i102140# load only once, not twice, and show error messages during loading (and during any form action, that is) asynchronously 2009-05-22 21:35:11 +0200 fs r272208 : #i102140# +clear 2009-05-22 14:50:30 +0200 fs r272194 : #i102139# for newly created DB docs, set the MacroExecutionMode to USE_CONFIG 2009-05-22 12:03:42 +0200 fs r272180 : #i88878# provided by noel.power@novell.com implement a visibility property (EnableVisible) for toolkit controls, and usage in forms and UNO dialogs 2009-05-15 15:37:31 +0200 fs r271942 : #i100671# corrected some @since tags, so autodoc has better chances of correctly reading them 2009-05-15 15:33:11 +0200 fs r271940 : don't call comphelper::getFOO for VOID values 2009-05-15 15:08:31 +0200 fs r271937 : includes 2009-05-15 13:39:22 +0200 fs r271934 : #i101398# createPrimitive2DSequence: when we already have a control, use the old code. In particular, call positionControlForPaint 2009-05-15 12:33:48 +0200 fs r271933 : make the geometry a part of the ControlPrimitive2D's identity 2009-05-15 10:15:44 +0200 fs r271928 : #i10000# 2009-05-14 20:55:38 +0200 fs r271921 : #i101398# don't reuse the name PRIMITIVE_ID_CONTROLPRIMITIVE2D, make the name of our own ControlPrimitive2D unique 2009-05-14 20:55:31 +0200 fs r271920 : #i101398# don't reuse the name PRIMITIVE_ID_CONTROLPRIMITIVE2D, make the name of our own ControlPrimitive2D unique 2009-05-14 20:23:23 +0200 fs r271919 : #i101622# 2009-05-14 16:04:38 +0200 fs r271898 : don't use comphelper::getInt32 on voids 2009-05-14 16:04:12 +0200 fs r271897 : merge fix for issue whose number just slipped my memory ... (originally fixed in CWS dba32b) 2009-05-14 15:36:55 +0200 fs r271895 : merging changes from DEV300:m48 2009-05-07 14:43:19 +0200 fs r271670 : #i101477# 2009-05-07 14:37:30 +0200 fs r271668 : #i101477# 2009-05-07 09:27:30 +0200 oj r271628 : #i101343# remove pch 2009-05-06 09:36:02 +0200 fs r271568 : getFoo: diagnostics 2009-05-04 09:23:06 +0200 oj r271438 : CWS-TOOLING: rebase CWS dba32c to trunk@271427 (milestone: DEV300:m47) 2009-04-29 23:18:13 +0200 fs r271394 : #i101398# use a dedicated 2DPrimitive for UNO Controls, which is able to provide the B2DRange *without* actually creating the control 2009-04-29 13:52:25 +0200 fs r271366 : #i101308#
1607 lines
61 KiB
C++
1607 lines
61 KiB
C++
/*************************************************************************
|
||
*
|
||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||
*
|
||
* Copyright 2008 by Sun Microsystems, Inc.
|
||
*
|
||
* OpenOffice.org - a multi-platform office productivity suite
|
||
*
|
||
* $RCSfile: RowSetCache.cxx,v $
|
||
* $Revision: 1.100 $
|
||
*
|
||
* This file is part of OpenOffice.org.
|
||
*
|
||
* OpenOffice.org is free software: you can redistribute it and/or modify
|
||
* it under the terms of the GNU Lesser General Public License version 3
|
||
* only, as published by the Free Software Foundation.
|
||
*
|
||
* OpenOffice.org 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 version 3 for more details
|
||
* (a copy is included in the LICENSE file that accompanied this code).
|
||
*
|
||
* You should have received a copy of the GNU Lesser General Public License
|
||
* version 3 along with OpenOffice.org. If not, see
|
||
* <http://www.openoffice.org/license.html>
|
||
* for a copy of the LGPLv3 License.
|
||
*
|
||
************************************************************************/
|
||
|
||
// MARKER(update_precomp.py): autogen include statement, do not remove
|
||
#include "precompiled_dbaccess.hxx"
|
||
|
||
#ifndef _COMPHELPER_SEQSTREAM_HXX
|
||
#include <comphelper/seqstream.hxx>
|
||
#endif
|
||
#ifndef _COMPHELPER_UNO3_HXX_
|
||
#include <comphelper/uno3.hxx>
|
||
#endif
|
||
#ifndef _COMPHELPER_EXTRACT_HXX_
|
||
#include <comphelper/extract.hxx>
|
||
#endif
|
||
#ifndef _COM_SUN_STAR_SDBCX_XKEYSSUPPLIER_HPP_
|
||
#include <com/sun/star/sdbcx/XKeysSupplier.hpp>
|
||
#endif
|
||
#ifndef _COM_SUN_STAR_SDBCX_XTABLESSUPPLIER_HPP_
|
||
#include <com/sun/star/sdbcx/XTablesSupplier.hpp>
|
||
#endif
|
||
#ifndef _COM_SUN_STAR_SDBCX_KEYTYPE_HPP_
|
||
#include <com/sun/star/sdbcx/KeyType.hpp>
|
||
#endif
|
||
#ifndef _COM_SUN_STAR_SDBC_RESULTSETCONCURRENCY_HPP_
|
||
#include <com/sun/star/sdbc/ResultSetConcurrency.hpp>
|
||
#endif
|
||
#ifndef _COM_SUN_STAR_SDBC_COLUMNVALUE_HPP_
|
||
#include <com/sun/star/sdbc/ColumnValue.hpp>
|
||
#endif
|
||
#ifndef _COM_SUN_STAR_SDBCX_XCOLUMNSSUPPLIER_HPP_
|
||
#include <com/sun/star/sdbcx/XColumnsSupplier.hpp>
|
||
#endif
|
||
#ifndef _COM_SUN_STAR_SDBCX_PRIVILEGE_HPP_
|
||
#include <com/sun/star/sdbcx/Privilege.hpp>
|
||
#endif
|
||
#ifndef _DBACORE_DATACOLUMN_HXX_
|
||
#include "CRowSetDataColumn.hxx"
|
||
#endif
|
||
#ifndef DBACCESS_CORE_API_CROWSETCOLUMN_HXX
|
||
#include "CRowSetColumn.hxx"
|
||
#endif
|
||
#ifndef DBACCESS_CORE_API_ROWSETBASE_HXX
|
||
#include "RowSetBase.hxx"
|
||
#endif
|
||
#ifndef _DBHELPER_DBEXCEPTION_HXX_
|
||
#include <connectivity/dbexception.hxx>
|
||
#endif
|
||
#ifndef _CONNECTIVITY_SQLPARSE_HXX
|
||
#include <connectivity/sqlparse.hxx>
|
||
#endif
|
||
#ifndef _CONNECTIVITY_SQLNODE_HXX
|
||
#include <connectivity/sqlnode.hxx>
|
||
#endif
|
||
#ifndef _CONNECTIVITY_PARSE_SQLITERATOR_HXX_
|
||
#include <connectivity/sqliterator.hxx>
|
||
#endif
|
||
#ifndef _COMPHELPER_PROPERTY_HXX_
|
||
#include <comphelper/property.hxx>
|
||
#endif
|
||
#ifndef _COM_SUN_STAR_SDBCX_COMPAREBOOKMARK_HPP_
|
||
#include <com/sun/star/sdbcx/CompareBookmark.hpp>
|
||
#endif
|
||
#ifndef _TOOLS_DEBUG_HXX
|
||
#include <tools/debug.hxx>
|
||
#endif
|
||
|
||
#include <algorithm>
|
||
#ifndef DBACCESS_CORE_API_ROWSETCACHE_HXX
|
||
#include "RowSetCache.hxx"
|
||
#endif
|
||
#ifndef _DBA_CORE_RESOURCE_HXX_
|
||
#include "core_resource.hxx"
|
||
#endif
|
||
#ifndef _DBA_CORE_RESOURCE_HRC_
|
||
#include "core_resource.hrc"
|
||
#endif
|
||
#ifndef DBACCESS_CORE_API_BOOKMARKSET_HXX
|
||
#include "BookmarkSet.hxx"
|
||
#endif
|
||
#ifndef DBACCESS_CORE_API_STATICSET_HXX
|
||
#include "StaticSet.hxx"
|
||
#endif
|
||
#ifndef DBACCESS_CORE_API_KEYSET_HXX
|
||
#include "KeySet.hxx"
|
||
#endif
|
||
#ifndef DBACCESS_SHARED_DBASTRINGS_HRC
|
||
#include "dbastrings.hrc"
|
||
#endif
|
||
|
||
using namespace dbaccess;
|
||
using namespace dbtools;
|
||
using namespace connectivity;
|
||
using namespace ::com::sun::star::uno;
|
||
using namespace ::com::sun::star::beans;
|
||
using namespace ::com::sun::star::sdbc;
|
||
using namespace ::com::sun::star::sdb;
|
||
using namespace ::com::sun::star::sdbcx;
|
||
using namespace ::com::sun::star::container;
|
||
using namespace ::com::sun::star::lang;
|
||
using namespace ::cppu;
|
||
using namespace ::osl;
|
||
|
||
#define CHECK_MATRIX_POS(M) OSL_ENSURE(((M) >= static_cast<ORowSetMatrix::difference_type>(0)) && ((M) < static_cast<sal_Int32>(m_pMatrix->size())),"Position is invalid!")
|
||
|
||
DBG_NAME(ORowSetCache)
|
||
// -------------------------------------------------------------------------
|
||
ORowSetCache::ORowSetCache(const Reference< XResultSet >& _xRs,
|
||
const Reference< XSingleSelectQueryAnalyzer >& _xAnalyzer,
|
||
const ::comphelper::ComponentContext& _rContext,
|
||
const ::rtl::OUString& _rUpdateTableName,
|
||
sal_Bool& _bModified,
|
||
sal_Bool& _bNew,
|
||
const ORowSetValueVector& _aParameterValueForCache)
|
||
:m_xSet(_xRs)
|
||
,m_xMetaData(Reference< XResultSetMetaDataSupplier >(_xRs,UNO_QUERY)->getMetaData())
|
||
,m_aContext( _rContext )
|
||
,m_pCacheSet(NULL)
|
||
,m_pMatrix(NULL)
|
||
,m_pInsertMatrix(NULL)
|
||
,m_nLastColumnIndex(0)
|
||
,m_nFetchSize(0)
|
||
,m_nRowCount(0)
|
||
,m_nPrivileges( Privilege::SELECT )
|
||
,m_nPosition(0)
|
||
,m_nStartPos(0)
|
||
,m_nEndPos(0)
|
||
,m_bRowCountFinal(sal_False)
|
||
,m_bBeforeFirst(sal_True)
|
||
,m_bAfterLast( sal_False )
|
||
,m_bUpdated(sal_False)
|
||
,m_bModified(_bModified)
|
||
,m_bNew(_bNew)
|
||
{
|
||
DBG_CTOR(ORowSetCache,NULL);
|
||
|
||
// check if all keys of the updateable table are fetched
|
||
sal_Bool bAllKeysFound = sal_False;
|
||
sal_Int32 nTablesCount = 0;
|
||
|
||
Reference< XIndexAccess> xUpdateTableKeys;
|
||
::rtl::OUString aUpdateTableName = _rUpdateTableName;
|
||
Reference< XConnection> xConnection;
|
||
if(_xAnalyzer.is())
|
||
{
|
||
try
|
||
{
|
||
Reference<XTablesSupplier> xTabSup(_xAnalyzer,UNO_QUERY);
|
||
OSL_ENSURE(xTabSup.is(),"ORowSet::execute composer isn't a tablesupplier!");
|
||
Reference<XNameAccess> xTables = xTabSup->getTables();
|
||
|
||
|
||
if(_rUpdateTableName.getLength() && xTables->hasByName(_rUpdateTableName))
|
||
xTables->getByName(_rUpdateTableName) >>= m_aUpdateTable;
|
||
else if(xTables->getElementNames().getLength())
|
||
{
|
||
aUpdateTableName = xTables->getElementNames()[0];
|
||
xTables->getByName(aUpdateTableName) >>= m_aUpdateTable;
|
||
}
|
||
Reference<XIndexAccess> xIndexAccess(xTables,UNO_QUERY);
|
||
if(xIndexAccess.is())
|
||
nTablesCount = xIndexAccess->getCount();
|
||
else
|
||
nTablesCount = xTables->getElementNames().getLength();
|
||
|
||
if(m_aUpdateTable.is() && nTablesCount < 3) // for we can't handle more than 2 tables in our keyset
|
||
{
|
||
Reference<XKeysSupplier> xKeys(m_aUpdateTable,UNO_QUERY);
|
||
if(xKeys.is())
|
||
{
|
||
xUpdateTableKeys = xKeys->getKeys();
|
||
if ( xUpdateTableKeys.is() )
|
||
{
|
||
Reference<XColumnsSupplier> xColumnsSupplier;
|
||
// search the one and only primary key
|
||
const sal_Int32 nCount = xUpdateTableKeys->getCount();
|
||
for(sal_Int32 i = 0 ; i < nCount ; ++i)
|
||
{
|
||
Reference<XPropertySet> xProp(xUpdateTableKeys->getByIndex(i),UNO_QUERY);
|
||
sal_Int32 nKeyType = 0;
|
||
xProp->getPropertyValue(PROPERTY_TYPE) >>= nKeyType;
|
||
if(KeyType::PRIMARY == nKeyType)
|
||
{
|
||
xColumnsSupplier.set(xProp,UNO_QUERY);
|
||
break;
|
||
}
|
||
}
|
||
|
||
if(xColumnsSupplier.is())
|
||
{
|
||
// first we need a connection
|
||
Reference< XStatement> xStmt(_xRs->getStatement(),UNO_QUERY);
|
||
if(xStmt.is())
|
||
xConnection = xStmt->getConnection();
|
||
else
|
||
{
|
||
Reference< XPreparedStatement> xPrepStmt(_xRs->getStatement(),UNO_QUERY);
|
||
xConnection = xPrepStmt->getConnection();
|
||
}
|
||
OSL_ENSURE(xConnection.is(),"No connection!");
|
||
|
||
Reference<XNameAccess> xColumns = xColumnsSupplier->getColumns();
|
||
Reference<XColumnsSupplier> xColSup(_xAnalyzer,UNO_QUERY);
|
||
if ( xColSup.is() )
|
||
{
|
||
Reference<XNameAccess> xSelColumns = xColSup->getColumns();
|
||
Reference<XDatabaseMetaData> xMeta = xConnection->getMetaData();
|
||
SelectColumnsMetaData aColumnNames(xMeta.is() && xMeta->supportsMixedCaseQuotedIdentifiers() ? true : false);
|
||
::dbaccess::getColumnPositions(xSelColumns,xColumns,aUpdateTableName,aColumnNames);
|
||
bAllKeysFound = !aColumnNames.empty() && sal_Int32(aColumnNames.size()) == xColumns->getElementNames().getLength();
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
catch(Exception&)
|
||
{
|
||
}
|
||
}
|
||
Reference< XPropertySet> xProp(_xRs,UNO_QUERY);
|
||
|
||
sal_Bool bNeedKeySet = !(xProp->getPropertySetInfo()->hasPropertyByName(PROPERTY_ISBOOKMARKABLE) &&
|
||
any2bool(xProp->getPropertyValue(PROPERTY_ISBOOKMARKABLE)) && Reference< XRowLocate >(_xRs, UNO_QUERY).is() );
|
||
bNeedKeySet = bNeedKeySet || (xProp->getPropertySetInfo()->hasPropertyByName(PROPERTY_RESULTSETCONCURRENCY) &&
|
||
::comphelper::getINT32(xProp->getPropertyValue(PROPERTY_RESULTSETCONCURRENCY)) == ResultSetConcurrency::READ_ONLY);
|
||
|
||
// first check if resultset is bookmarkable
|
||
if(!bNeedKeySet)
|
||
{
|
||
try
|
||
{
|
||
m_pCacheSet = new OBookmarkSet();
|
||
m_xCacheSet = m_pCacheSet;
|
||
m_pCacheSet->construct(_xRs);
|
||
|
||
// check privileges
|
||
m_nPrivileges = Privilege::SELECT;
|
||
if(Reference<XResultSetUpdate>(_xRs,UNO_QUERY).is()) // this interface is optional so we have to check it
|
||
{
|
||
Reference<XPropertySet> xTable(m_aUpdateTable,UNO_QUERY);
|
||
if(xTable.is() && xTable->getPropertySetInfo()->hasPropertyByName(PROPERTY_PRIVILEGES))
|
||
{
|
||
m_nPrivileges = 0;
|
||
xTable->getPropertyValue(PROPERTY_PRIVILEGES) >>= m_nPrivileges;
|
||
if(!m_nPrivileges)
|
||
m_nPrivileges = Privilege::SELECT;
|
||
}
|
||
}
|
||
}
|
||
catch(const SQLException&)
|
||
{
|
||
bNeedKeySet = sal_True;
|
||
}
|
||
|
||
}
|
||
if(bNeedKeySet)
|
||
{
|
||
// need to check if we could handle this select clause
|
||
bAllKeysFound = bAllKeysFound && (nTablesCount == 1 || checkJoin(xConnection,_xAnalyzer,aUpdateTableName));
|
||
|
||
// || !(comphelper::hasProperty(PROPERTY_CANUPDATEINSERTEDROWS,xProp) && any2bool(xProp->getPropertyValue(PROPERTY_CANUPDATEINSERTEDROWS)))
|
||
|
||
// oj removed because keyset uses only the next// || (xProp->getPropertySetInfo()->hasPropertyByName(PROPERTY_RESULTSETTYPE) && comphelper::getINT32(xProp->getPropertyValue(PROPERTY_RESULTSETTYPE)) == ResultSetType::FORWARD_ONLY)
|
||
if(!bAllKeysFound )
|
||
{
|
||
m_pCacheSet = new OStaticSet();
|
||
m_xCacheSet = m_pCacheSet;
|
||
m_pCacheSet->construct(_xRs);
|
||
m_nPrivileges = Privilege::SELECT;
|
||
}
|
||
else
|
||
{
|
||
Reference<XDatabaseMetaData> xMeta = xConnection->getMetaData();
|
||
SelectColumnsMetaData aColumnNames(xMeta.is() && xMeta->supportsMixedCaseQuotedIdentifiers() ? true : false);
|
||
Reference<XColumnsSupplier> xColSup(_xAnalyzer,UNO_QUERY);
|
||
Reference<XNameAccess> xSelColumns = xColSup->getColumns();
|
||
Reference<XNameAccess> xColumns = m_aUpdateTable->getColumns();
|
||
::dbaccess::getColumnPositions(xSelColumns,xColumns,aUpdateTableName,aColumnNames);
|
||
|
||
// check privileges
|
||
m_nPrivileges = Privilege::SELECT;
|
||
sal_Bool bNoInsert = sal_False;
|
||
|
||
Sequence< ::rtl::OUString> aNames(xColumns->getElementNames());
|
||
const ::rtl::OUString* pIter = aNames.getConstArray();
|
||
const ::rtl::OUString* pEnd = pIter + aNames.getLength();
|
||
for(;pIter != pEnd;++pIter)
|
||
{
|
||
Reference<XPropertySet> xColumn;
|
||
::cppu::extractInterface(xColumn,xColumns->getByName(*pIter));
|
||
OSL_ENSURE(xColumn.is(),"Column in table is null!");
|
||
if(xColumn.is())
|
||
{
|
||
sal_Int32 nNullable = 0;
|
||
xColumn->getPropertyValue(PROPERTY_ISNULLABLE) >>= nNullable;
|
||
if(nNullable == ColumnValue::NO_NULLS && aColumnNames.find(*pIter) == aColumnNames.end())
|
||
{ // we found a column where null is not allowed so we can't insert new values
|
||
bNoInsert = sal_True;
|
||
break; // one column is enough
|
||
}
|
||
}
|
||
}
|
||
|
||
OKeySet* pKeySet = new OKeySet(m_aUpdateTable,xUpdateTableKeys,aUpdateTableName ,_xAnalyzer,_aParameterValueForCache);
|
||
try
|
||
{
|
||
m_pCacheSet = pKeySet;
|
||
m_xCacheSet = m_pCacheSet;
|
||
pKeySet->construct(_xRs);
|
||
|
||
if(Reference<XResultSetUpdate>(_xRs,UNO_QUERY).is()) // this interface is optional so we have to check it
|
||
{
|
||
Reference<XPropertySet> xTable(m_aUpdateTable,UNO_QUERY);
|
||
if(xTable.is() && xTable->getPropertySetInfo()->hasPropertyByName(PROPERTY_PRIVILEGES))
|
||
{
|
||
m_nPrivileges = 0;
|
||
xTable->getPropertyValue(PROPERTY_PRIVILEGES) >>= m_nPrivileges;
|
||
if(!m_nPrivileges)
|
||
m_nPrivileges = Privilege::SELECT;
|
||
}
|
||
}
|
||
if(bNoInsert)
|
||
m_nPrivileges |= ~Privilege::INSERT; // remove the insert privilege
|
||
}
|
||
catch(const SQLException&)
|
||
{
|
||
// we couldn't create a keyset here so we have to create a static cache
|
||
if ( m_pCacheSet )
|
||
m_pCacheSet = NULL;
|
||
m_xCacheSet = NULL;
|
||
m_pCacheSet = new OStaticSet();
|
||
m_xCacheSet = m_pCacheSet;
|
||
m_pCacheSet->construct(_xRs);
|
||
m_nPrivileges = Privilege::SELECT;
|
||
}
|
||
}
|
||
|
||
}
|
||
// last check
|
||
if(!bAllKeysFound && xProp->getPropertySetInfo()->hasPropertyByName(PROPERTY_RESULTSETCONCURRENCY) &&
|
||
::comphelper::getINT32(xProp->getPropertyValue(PROPERTY_RESULTSETCONCURRENCY)) == ResultSetConcurrency::READ_ONLY)
|
||
m_nPrivileges = Privilege::SELECT;
|
||
}
|
||
|
||
// -------------------------------------------------------------------------
|
||
ORowSetCache::~ORowSetCache()
|
||
{
|
||
m_pCacheSet = NULL;
|
||
m_xCacheSet = NULL;
|
||
if(m_pMatrix)
|
||
{
|
||
m_pMatrix->clear();
|
||
delete m_pMatrix;
|
||
}
|
||
|
||
if(m_pInsertMatrix)
|
||
{
|
||
m_pInsertMatrix->clear();
|
||
delete m_pInsertMatrix;
|
||
}
|
||
m_xSet = WeakReference< XResultSet>();
|
||
m_xMetaData = NULL;
|
||
m_aUpdateTable = NULL;
|
||
|
||
DBG_DTOR(ORowSetCache,NULL);
|
||
}
|
||
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::setMaxRowSize(sal_Int32 _nSize)
|
||
{
|
||
|
||
if(_nSize == m_nFetchSize)
|
||
return;
|
||
|
||
m_nFetchSize = _nSize;
|
||
if(!m_pMatrix)
|
||
{
|
||
m_pMatrix = new ORowSetMatrix(_nSize);
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
m_aMatrixEnd = m_pMatrix->end();
|
||
|
||
m_pInsertMatrix = new ORowSetMatrix(1); // a little bit overkill but ??? :-)
|
||
m_aInsertRow = m_pInsertMatrix->end();
|
||
}
|
||
else
|
||
{
|
||
// now correct the iterator in our iterator vector
|
||
::std::vector<sal_Int32> aPositions;
|
||
::std::map<sal_Int32,sal_Bool> aCacheIterToChange;
|
||
// first get the positions where they stand now
|
||
ORowSetCacheMap::iterator aCacheIter = m_aCacheIterators.begin();
|
||
ORowSetCacheMap::iterator aCacheEnd = m_aCacheIterators.end();
|
||
for(;aCacheIter != aCacheEnd;++aCacheIter)
|
||
{
|
||
aCacheIterToChange[aCacheIter->first] = sal_False;
|
||
if ( !aCacheIter->second.pRowSet->isInsertRow()
|
||
/*&& aCacheIter->second.aIterator != m_pMatrix->end()*/ && !m_bModified )
|
||
{
|
||
ptrdiff_t nDist = (aCacheIter->second.aIterator - m_pMatrix->begin());
|
||
aPositions.push_back(nDist);
|
||
aCacheIterToChange[aCacheIter->first] = sal_True;
|
||
}
|
||
}
|
||
sal_Int32 nKeyPos = (m_aMatrixIter - m_pMatrix->begin());
|
||
m_pMatrix->resize(_nSize);
|
||
|
||
if ( nKeyPos < _nSize )
|
||
m_aMatrixIter = m_pMatrix->begin() + nKeyPos;
|
||
else
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
m_aMatrixEnd = m_pMatrix->end();
|
||
|
||
// now adjust their positions because a resize invalid all iterators
|
||
::std::vector<sal_Int32>::const_iterator aIter = aPositions.begin();
|
||
::std::map<sal_Int32,sal_Bool>::const_iterator aPosChangeIter = aCacheIterToChange.begin();
|
||
for( aCacheIter = m_aCacheIterators.begin();
|
||
aPosChangeIter != aCacheIterToChange.end();
|
||
++aPosChangeIter,++aCacheIter)
|
||
{
|
||
if ( aPosChangeIter->second )
|
||
{
|
||
CHECK_MATRIX_POS(*aIter);
|
||
if ( *aIter < _nSize )
|
||
aCacheIter->second.aIterator = m_pMatrix->begin() + *aIter++;
|
||
else
|
||
aCacheIter->second.aIterator = m_pMatrix->end();
|
||
}
|
||
}
|
||
}
|
||
if(!m_nPosition)
|
||
{
|
||
sal_Int32 nNewSt = 1;
|
||
fillMatrix(nNewSt,_nSize+1);
|
||
m_nStartPos = 0;
|
||
m_nEndPos = _nSize;
|
||
}
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
|
||
// XResultSetMetaDataSupplier
|
||
Reference< XResultSetMetaData > ORowSetCache::getMetaData( )
|
||
{
|
||
return m_xMetaData;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
// ::com::sun::star::sdbcx::XRowLocate
|
||
Any ORowSetCache::getBookmark( )
|
||
{
|
||
|
||
if(m_bAfterLast)
|
||
throwFunctionSequenceException(m_xSet.get());
|
||
|
||
if ( m_aMatrixIter >= m_pMatrix->end() || m_aMatrixIter < m_pMatrix->begin() || !(*m_aMatrixIter).isValid())
|
||
{
|
||
return Any(); // this is allowed here because the rowset knowns what it is doing
|
||
}
|
||
|
||
switch(((*m_aMatrixIter)->get())[0].getTypeKind())
|
||
{
|
||
case DataType::TINYINT:
|
||
case DataType::SMALLINT:
|
||
case DataType::INTEGER:
|
||
return makeAny((sal_Int32)((*m_aMatrixIter)->get())[0]);
|
||
default:
|
||
if(((*m_aMatrixIter)->get())[0].isNull())
|
||
((*m_aMatrixIter)->get())[0] = m_pCacheSet->getBookmark();
|
||
return ((*m_aMatrixIter)->get())[0].getAny();
|
||
}
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::moveToBookmark( const Any& bookmark )
|
||
{
|
||
if ( m_pCacheSet->moveToBookmark(bookmark) )
|
||
{
|
||
m_bBeforeFirst = sal_False;
|
||
m_nPosition = m_pCacheSet->getRow();
|
||
|
||
checkPositionFlags();
|
||
|
||
if(!m_bAfterLast)
|
||
{
|
||
moveWindow();
|
||
checkPositionFlags();
|
||
if ( !m_bAfterLast )
|
||
{
|
||
m_aMatrixIter = calcPosition();
|
||
OSL_ENSURE(m_aMatrixIter->isValid(),"Iterator after moveToBookmark not valid");
|
||
}
|
||
else
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
else
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
else
|
||
return sal_False;
|
||
|
||
return m_aMatrixIter != m_pMatrix->end() && (*m_aMatrixIter).isValid();
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::moveRelativeToBookmark( const Any& bookmark, sal_Int32 rows )
|
||
{
|
||
sal_Bool bRet( moveToBookmark( bookmark ) );
|
||
if ( bRet )
|
||
{
|
||
m_nPosition = m_pCacheSet->getRow() + rows;
|
||
absolute(m_nPosition);
|
||
// for(sal_Int32 i=0;i<rows && m_aMatrixIter != m_pMatrix->end();++i,++m_aMatrixIter) ;
|
||
|
||
bRet = m_aMatrixIter != m_pMatrix->end() && (*m_aMatrixIter).isValid();
|
||
}
|
||
|
||
return bRet;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Int32 ORowSetCache::compareBookmarks( const Any& _first, const Any& _second )
|
||
{
|
||
return (!_first.hasValue() || !_second.hasValue()) ? CompareBookmark::NOT_COMPARABLE : m_pCacheSet->compareBookmarks(_first,_second);
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::hasOrderedBookmarks( )
|
||
{
|
||
|
||
return m_pCacheSet->hasOrderedBookmarks();
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Int32 ORowSetCache::hashBookmark( const Any& bookmark )
|
||
{
|
||
|
||
return m_pCacheSet->hashBookmark(bookmark);
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
// XRowUpdate
|
||
// -----------------------------------------------------------------------------
|
||
void ORowSetCache::updateValue(sal_Int32 columnIndex,const ORowSetValue& x)
|
||
{
|
||
checkUpdateConditions(columnIndex);
|
||
|
||
|
||
((*m_aInsertRow)->get())[columnIndex].setBound(sal_True);
|
||
((*m_aInsertRow)->get())[columnIndex] = x;
|
||
((*m_aInsertRow)->get())[columnIndex].setModified();
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::updateBinaryStream( sal_Int32 columnIndex, const Reference< ::com::sun::star::io::XInputStream >& x, sal_Int32 length )
|
||
{
|
||
checkUpdateConditions(columnIndex);
|
||
|
||
|
||
Sequence<sal_Int8> aSeq;
|
||
if(x.is())
|
||
x->readBytes(aSeq,length);
|
||
updateValue(columnIndex,aSeq);
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::updateCharacterStream( sal_Int32 columnIndex, const Reference< ::com::sun::star::io::XInputStream >& x, sal_Int32 length )
|
||
{
|
||
checkUpdateConditions(columnIndex);
|
||
|
||
|
||
Sequence<sal_Int8> aSeq;
|
||
if(x.is())
|
||
x->readBytes(aSeq,length);
|
||
|
||
updateValue(columnIndex,aSeq);
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::updateObject( sal_Int32 columnIndex, const Any& x )
|
||
{
|
||
checkUpdateConditions(columnIndex);
|
||
|
||
|
||
((*m_aInsertRow)->get())[columnIndex].setBound(sal_True);
|
||
((*m_aInsertRow)->get())[columnIndex] = x;
|
||
((*m_aInsertRow)->get())[columnIndex].setModified();
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::updateNumericObject( sal_Int32 columnIndex, const Any& x, sal_Int32 /*scale*/ )
|
||
{
|
||
checkUpdateConditions(columnIndex);
|
||
|
||
|
||
((*m_aInsertRow)->get())[columnIndex].setBound(sal_True);
|
||
((*m_aInsertRow)->get())[columnIndex] = x;
|
||
((*m_aInsertRow)->get())[columnIndex].setModified();
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
// XResultSet
|
||
sal_Bool ORowSetCache::next( )
|
||
{
|
||
|
||
|
||
if(!isAfterLast())
|
||
{
|
||
m_bBeforeFirst = sal_False;
|
||
++m_nPosition;
|
||
|
||
// after we increment the position we have to check if we are already after the last row
|
||
checkPositionFlags();
|
||
if(!m_bAfterLast)
|
||
{
|
||
moveWindow();
|
||
|
||
OSL_ENSURE(((m_nPosition - m_nStartPos) - 1) < (sal_Int32)m_pMatrix->size(),"Position is behind end()!");
|
||
m_aMatrixIter = calcPosition();
|
||
checkPositionFlags();
|
||
}
|
||
}
|
||
|
||
return !m_bAfterLast;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::isBeforeFirst( )
|
||
{
|
||
// return !m_nPosition;
|
||
|
||
return m_bBeforeFirst;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::isAfterLast( )
|
||
{
|
||
|
||
return m_bAfterLast;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::isFirst( )
|
||
{
|
||
|
||
return m_nPosition == 1; // ask resultset for
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::isLast( )
|
||
{
|
||
// return m_bRowCountFinal ? (m_nPosition==m_nRowCount) : m_pCacheSet->isLast();
|
||
|
||
return m_nPosition == m_nRowCount;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::beforeFirst( )
|
||
{
|
||
|
||
|
||
if(!m_bBeforeFirst)
|
||
{
|
||
m_bAfterLast = sal_False;
|
||
m_nPosition = 0;
|
||
m_bBeforeFirst = sal_True;
|
||
m_pCacheSet->beforeFirst();
|
||
moveWindow();
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
return sal_True;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::afterLast( )
|
||
{
|
||
|
||
|
||
if(!m_bAfterLast)
|
||
{
|
||
m_bBeforeFirst = sal_False;
|
||
m_bAfterLast = sal_True;
|
||
|
||
if(!m_bRowCountFinal)
|
||
{
|
||
m_pCacheSet->last();
|
||
m_bRowCountFinal = sal_True;
|
||
m_nRowCount = m_pCacheSet->getRow();// + 1 removed
|
||
}
|
||
m_pCacheSet->afterLast();
|
||
|
||
m_nPosition = 0;
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
return sal_True;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::fillMatrix(sal_Int32& _nNewStartPos,sal_Int32 _nNewEndPos)
|
||
{
|
||
OSL_ENSURE(_nNewStartPos != _nNewEndPos,"ORowSetCache::fillMatrix: StartPos and EndPos can not be equal!");
|
||
// fill the whole window with new data
|
||
ORowSetMatrix::iterator aIter = m_pMatrix->begin();
|
||
sal_Bool bCheck = m_pCacheSet->absolute(_nNewStartPos); // -1 no need to
|
||
|
||
sal_Int32 i=_nNewStartPos;
|
||
for(;i<_nNewEndPos;++i,++aIter)
|
||
{
|
||
if(bCheck)
|
||
{
|
||
if(!aIter->isValid())
|
||
*aIter = new ORowSetValueVector(m_xMetaData->getColumnCount());
|
||
m_pCacheSet->fillValueRow(*aIter,i);
|
||
}
|
||
else
|
||
{ // there are no more rows found so we can fetch some before start
|
||
|
||
if(!m_bRowCountFinal)
|
||
{
|
||
if(m_pCacheSet->previous()) // because we stand after the last row
|
||
m_nRowCount = m_pCacheSet->getRow(); // here we have the row count
|
||
if(!m_nRowCount)
|
||
m_nRowCount = i-1; // it can be that getRow return zero
|
||
m_bRowCountFinal = sal_True;
|
||
}
|
||
if(m_nRowCount > m_nFetchSize)
|
||
{
|
||
ORowSetMatrix::iterator aEnd = aIter;
|
||
ORowSetMatrix::iterator aRealEnd = m_pMatrix->end();
|
||
sal_Int32 nPos = m_nRowCount - m_nFetchSize + 1;
|
||
_nNewStartPos = nPos;
|
||
bCheck = m_pCacheSet->absolute(_nNewStartPos);
|
||
|
||
for(;bCheck && aIter != aRealEnd;++aIter)
|
||
{
|
||
if(bCheck)
|
||
{
|
||
if(!aIter->isValid())
|
||
*aIter = new ORowSetValueVector(m_xMetaData->getColumnCount());
|
||
m_pCacheSet->fillValueRow(*aIter,nPos++);
|
||
}
|
||
bCheck = m_pCacheSet->next();
|
||
}
|
||
if(aIter != aEnd)
|
||
::std::rotate(m_pMatrix->begin(),aEnd,aRealEnd);
|
||
}
|
||
break;
|
||
}
|
||
bCheck = m_pCacheSet->next();
|
||
}
|
||
// m_nStartPos = _nNewStartPos;
|
||
// we have to read one row forward to enshure that we know when we are on last row
|
||
// but only when we don't know it already
|
||
if(!m_bRowCountFinal)
|
||
{
|
||
if(!m_pCacheSet->next())
|
||
{
|
||
if(m_pCacheSet->previous()) // because we stand after the last row
|
||
m_nRowCount = m_pCacheSet->getRow(); // here we have the row count
|
||
m_bRowCountFinal = sal_True;
|
||
}
|
||
else
|
||
m_nRowCount = std::max(i,m_nRowCount);
|
||
|
||
}
|
||
return bCheck;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::moveWindow()
|
||
{
|
||
|
||
sal_Bool bRet = sal_True;
|
||
|
||
sal_Int32 nDiff = (sal_Int32)(m_nFetchSize*0.5 -0.5);
|
||
sal_Int32 nNewStartPos = (m_nPosition - nDiff);
|
||
// sal_Int32 nNewEndPos = (m_nPosition+m_nFetchSize*0.5);
|
||
sal_Int32 nNewEndPos = nNewStartPos + m_nFetchSize;
|
||
|
||
if ( m_nPosition <= m_nStartPos )
|
||
{ // the window is behind the new start pos
|
||
if(!m_nStartPos)
|
||
return sal_False;
|
||
// the new position should be the nPos - nFetchSize/2
|
||
if ( nNewEndPos > m_nStartPos )
|
||
{ // but the two regions are overlapping
|
||
// fill the rows behind the new end
|
||
|
||
ORowSetMatrix::iterator aEnd; // the iterator we need for rotate
|
||
ORowSetMatrix::iterator aIter; // the iterator we fill with new values
|
||
|
||
sal_Bool bCheck = sal_True;
|
||
if ( nNewStartPos < 1 )
|
||
{
|
||
bCheck = m_pCacheSet->first();
|
||
// aEnd = m_pMatrix->begin() + (sal_Int32)(m_nFetchSize*0.5);
|
||
OSL_ENSURE((nNewEndPos - m_nStartPos - nNewStartPos) < (sal_Int32)m_pMatrix->size(),"Position is behind end()!");
|
||
aEnd = m_pMatrix->begin() + (nNewEndPos - m_nStartPos - nNewStartPos);
|
||
aIter = aEnd;
|
||
m_nStartPos = 0;
|
||
}
|
||
else
|
||
{
|
||
OSL_ENSURE((nNewEndPos - m_nStartPos -1) < (sal_Int32)m_pMatrix->size(),"Position is behind end()!");
|
||
aEnd = m_pMatrix->begin() + ((nNewEndPos - m_nStartPos)-1);
|
||
aIter = m_pMatrix->begin() + ((nNewEndPos - m_nStartPos)-1);
|
||
bCheck = m_pCacheSet->absolute(nNewStartPos);
|
||
m_nStartPos = nNewStartPos -1;
|
||
}
|
||
|
||
if ( bCheck )
|
||
{
|
||
sal_Int32 nPos = m_nStartPos;
|
||
bCheck = fill(aIter,m_pMatrix->end(),nPos,bCheck);
|
||
|
||
::std::rotate(m_pMatrix->begin(),aEnd,m_pMatrix->end());
|
||
// now correct the iterator in our iterator vector
|
||
// rotateCacheIterator(aEnd-m_pMatrix->begin()); //can't be used because they decrement and here we need to increment
|
||
ptrdiff_t nNewDist = aEnd - m_pMatrix->begin();
|
||
ptrdiff_t nOffSet = m_pMatrix->end() - aEnd;
|
||
ORowSetCacheMap::iterator aCacheIter = m_aCacheIterators.begin();
|
||
ORowSetCacheMap::iterator aCacheEnd = m_aCacheIterators.end();
|
||
for(;aCacheIter != aCacheEnd;++aCacheIter)
|
||
{
|
||
if ( !aCacheIter->second.pRowSet->isInsertRow()
|
||
&& aCacheIter->second.aIterator != m_pMatrix->end() && !m_bModified )
|
||
{
|
||
ptrdiff_t nDist = (aCacheIter->second.aIterator - m_pMatrix->begin());
|
||
if ( nDist >= nNewDist )
|
||
{
|
||
aCacheIter->second.aIterator = m_pMatrix->end();
|
||
}
|
||
else
|
||
{
|
||
#if OSL_DEBUG_LEVEL > 0
|
||
ORowSetMatrix::iterator aOldPos;
|
||
aOldPos = aCacheIter->second.aIterator;
|
||
#endif
|
||
CHECK_MATRIX_POS( ((aOldPos - m_pMatrix->begin()) + nOffSet) );
|
||
aCacheIter->second.aIterator += nOffSet;
|
||
#if OSL_DEBUG_LEVEL > 0
|
||
ORowSetMatrix::iterator aCurrentPos;
|
||
aCurrentPos = aCacheIter->second.aIterator;
|
||
#endif
|
||
OSL_ENSURE(aCacheIter->second.aIterator >= m_pMatrix->begin()
|
||
&& aCacheIter->second.aIterator < m_pMatrix->end(),"Iterator out of area!");
|
||
}
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{ // normaly this should never happen
|
||
OSL_ENSURE(0,"What the hell is happen here!");
|
||
return sal_False;
|
||
}
|
||
}
|
||
else
|
||
{// no rows can be reused so fill again
|
||
if(nNewStartPos < 1) // special case
|
||
{
|
||
m_nStartPos = 0;
|
||
|
||
rotateCacheIterator(static_cast<sal_Int16>(m_nFetchSize+1)); // static_cast<sal_Int16>(m_nFetchSize+1)
|
||
|
||
m_pCacheSet->beforeFirst();
|
||
|
||
sal_Bool bCheck;
|
||
ORowSetMatrix::iterator aIter = m_pMatrix->begin();
|
||
for(sal_Int32 i=0;i<m_nFetchSize;++i,++aIter)
|
||
{
|
||
bCheck = m_pCacheSet->next();
|
||
if ( bCheck )
|
||
{
|
||
if(!aIter->isValid())
|
||
*aIter = new ORowSetValueVector(m_xMetaData->getColumnCount());
|
||
m_pCacheSet->fillValueRow(*aIter,i+1);
|
||
}
|
||
else
|
||
*aIter = NULL;
|
||
}
|
||
}
|
||
else
|
||
bRet = reFillMatrix(nNewStartPos,nNewEndPos);
|
||
}
|
||
}
|
||
else if(m_nPosition > m_nStartPos)
|
||
{ // the new start pos is above the startpos of the window
|
||
|
||
if(m_nPosition <= (m_nStartPos+m_nFetchSize))
|
||
{ // position in window
|
||
OSL_ENSURE((m_nPosition - m_nStartPos -1) < (sal_Int32)m_pMatrix->size(),"Position is behind end()!");
|
||
m_aMatrixIter = calcPosition();
|
||
if(!m_aMatrixIter->isValid())
|
||
{
|
||
sal_Bool bOk( m_pCacheSet->absolute( m_nPosition ) );
|
||
if ( bOk )
|
||
{
|
||
*m_aMatrixIter = new ORowSetValueVector(m_xMetaData->getColumnCount());
|
||
m_pCacheSet->fillValueRow(*m_aMatrixIter,m_nPosition);
|
||
// we have to read one row forward to enshure that we know when we are on last row
|
||
// but only when we don't know it already
|
||
if ( !m_bRowCountFinal )
|
||
{
|
||
bOk = m_pCacheSet->absolute( m_nPosition + 1 );
|
||
if ( bOk )
|
||
m_nRowCount = std::max(sal_Int32(m_nPosition+1),m_nRowCount);
|
||
}
|
||
}
|
||
if(!bOk)
|
||
{
|
||
if(!m_bRowCountFinal)
|
||
{
|
||
// because we stand after the last row
|
||
m_nRowCount = m_pCacheSet->previous() ? m_pCacheSet->getRow() : 0;// + 1 removed
|
||
m_bRowCountFinal = sal_True;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
else if(nNewStartPos < (m_nStartPos+m_nFetchSize))
|
||
{ // position behind window but the region is overlapping
|
||
// the rows from begin() to (begin + nNewStartPos - m_nStartPos) can be refilled with the new rows
|
||
// the rows behind this can be reused
|
||
ORowSetMatrix::iterator aIter = m_pMatrix->begin();
|
||
CHECK_MATRIX_POS(nNewStartPos - m_nStartPos - 1);
|
||
ORowSetMatrix::iterator aEnd = m_pMatrix->begin() + (nNewStartPos - m_nStartPos - 1);
|
||
|
||
sal_Int32 nPos = m_nStartPos + m_nFetchSize + 1;
|
||
sal_Bool bCheck = m_pCacheSet->absolute(nPos);
|
||
bCheck = fill(aIter,aEnd,nPos,bCheck); // refill the region wew don't need anymore
|
||
|
||
// // we know that this is the current maximal rowcount here
|
||
// if ( !m_bRowCountFinal && bCheck )
|
||
// m_nRowCount = std::max(nPos,m_nRowCount);
|
||
// we have to read one row forward to enshure that we know when we are on last row
|
||
// but only when we don't know it already
|
||
sal_Bool bOk = sal_True;
|
||
if(bCheck && !m_bRowCountFinal)
|
||
bOk = m_pCacheSet->next();
|
||
// bind end to front
|
||
if(bCheck)
|
||
{ // rotate the end to the front
|
||
::std::rotate(m_pMatrix->begin(),aIter,m_pMatrix->end());
|
||
// now correct the iterator in our iterator vector
|
||
rotateCacheIterator( (sal_Int16)( aIter - m_pMatrix->begin() ) );
|
||
m_nStartPos = nNewStartPos - 1; // must be -1
|
||
// now I can say how many rows we have
|
||
if(!bOk)
|
||
{
|
||
m_pCacheSet->previous(); // because we stand after the last row
|
||
m_nRowCount = nPos; // here we have the row count
|
||
m_bRowCountFinal = sal_True;
|
||
}
|
||
else if(!m_bRowCountFinal)
|
||
m_nRowCount = std::max(++nPos,m_nRowCount);
|
||
}
|
||
else
|
||
{ // the end was reached before end() so we can set the start before nNewStartPos
|
||
|
||
m_nStartPos += (aIter - m_pMatrix->begin());
|
||
// m_nStartPos = (aIter - m_pMatrix->begin());
|
||
::std::rotate(m_pMatrix->begin(),aIter,m_pMatrix->end());
|
||
// now correct the iterator in our iterator vector
|
||
rotateCacheIterator( (sal_Int16)( aIter - m_pMatrix->begin() ) );
|
||
|
||
if ( !m_bRowCountFinal )
|
||
{
|
||
m_pCacheSet->previous(); // because we stand after the last row
|
||
m_nRowCount = std::max(m_nRowCount,--nPos); // here we have the row count
|
||
OSL_ENSURE(nPos == m_pCacheSet->getRow(),"nPos isn't valid!");
|
||
m_bRowCountFinal = sal_True;
|
||
}
|
||
// TODO check
|
||
// m_nStartPos = (nNewStartPos+m_nRowCount) - m_nFetchSize ;
|
||
if(m_nStartPos < 0)
|
||
m_nStartPos = 0;
|
||
}
|
||
// here we need only to check if the begining row is valid. If not we have to fetch it.
|
||
if(!m_pMatrix->begin()->isValid())
|
||
{
|
||
aIter = m_pMatrix->begin();
|
||
|
||
nPos = m_nStartPos;
|
||
bCheck = m_pCacheSet->absolute(m_nStartPos);
|
||
for(; !aIter->isValid() && bCheck;++aIter)
|
||
{
|
||
OSL_ENSURE(aIter != m_pMatrix->end(),"Invalid iterator");
|
||
bCheck = m_pCacheSet->next();
|
||
if ( bCheck ) // resultset stands on right position
|
||
{
|
||
*aIter = new ORowSetValueVector(m_xMetaData->getColumnCount());
|
||
m_pCacheSet->fillValueRow(*aIter,++nPos);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
else // no rows can be reused so fill again
|
||
bRet = reFillMatrix(nNewStartPos,nNewEndPos);
|
||
}
|
||
|
||
if(!m_bRowCountFinal)
|
||
m_nRowCount = std::max(m_nPosition,m_nRowCount);
|
||
OSL_ENSURE(m_nStartPos >= 0,"ORowSetCache::moveWindow: m_nStartPos is less than 0!");
|
||
|
||
return bRet;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::first( )
|
||
{
|
||
// first move to the first row
|
||
// then check if the cache window is at the begining
|
||
// when not postionize the window and fill it with data
|
||
// smart moving of the window -> clear only the rows whom are out of range
|
||
sal_Bool bRet = m_pCacheSet->first();
|
||
if(bRet)
|
||
{
|
||
m_bBeforeFirst = m_bAfterLast = sal_False;
|
||
m_nPosition = 1;
|
||
moveWindow();
|
||
m_aMatrixIter = m_pMatrix->begin();
|
||
}
|
||
else
|
||
{
|
||
m_bRowCountFinal = m_bBeforeFirst = m_bAfterLast = sal_True;
|
||
m_nRowCount = m_nPosition = 0;
|
||
|
||
OSL_ENSURE(m_bBeforeFirst || m_bNew,"ORowSetCache::first return false and BeforeFirst isn't true");
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
return bRet;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::last( )
|
||
{
|
||
sal_Bool bRet = m_pCacheSet->last();
|
||
if(bRet)
|
||
{
|
||
m_bBeforeFirst = m_bAfterLast = sal_False;
|
||
if(!m_bRowCountFinal)
|
||
{
|
||
m_bRowCountFinal = sal_True;
|
||
m_nRowCount = m_nPosition = m_pCacheSet->getRow(); // not + 1
|
||
}
|
||
m_nPosition = m_pCacheSet->getRow();
|
||
moveWindow();
|
||
// we have to repositioning because moveWindow can modify the cache
|
||
m_pCacheSet->last();
|
||
// if(m_nPosition > m_nFetchSize)
|
||
// m_aMatrixIter = m_pMatrix->end() -1;
|
||
// else
|
||
// m_aMatrixIter = m_pMatrix->begin() + m_nPosition - 1;
|
||
OSL_ENSURE(((m_nPosition - m_nStartPos) - 1) < (sal_Int32)m_pMatrix->size(),"Position is behind end()!");
|
||
m_aMatrixIter = calcPosition();
|
||
}
|
||
else
|
||
{
|
||
m_bRowCountFinal = m_bBeforeFirst = m_bAfterLast = sal_True;
|
||
m_nRowCount = m_nPosition = 0;
|
||
OSL_ENSURE(m_bBeforeFirst,"ORowSetCache::last return false and BeforeFirst isn't true");
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
#if OSL_DEBUG_LEVEL > 1
|
||
if(bRet)
|
||
{
|
||
OSL_ENSURE((*m_aMatrixIter).isValid(),"ORowSetCache::last: Row not valid!");
|
||
}
|
||
#endif
|
||
|
||
return bRet;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Int32 ORowSetCache::getRow( )
|
||
{
|
||
return (isBeforeFirst() || isAfterLast()) ? 0 : m_nPosition;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::absolute( sal_Int32 row )
|
||
{
|
||
if(!row )
|
||
throw SQLException(DBACORE_RESSTRING(RID_STR_NO_ABS_ZERO),NULL,SQLSTATE_GENERAL,1000,Any() );
|
||
|
||
if(row < 0)
|
||
{
|
||
// here we have to scroll from the last row to backward so we have to go to last row and
|
||
// and two the previous
|
||
if(m_bRowCountFinal || last())
|
||
{
|
||
m_nPosition = m_nRowCount + row + 1; // + row because row is negative and +1 because row==-1 means last row
|
||
if(m_nPosition < 1)
|
||
{
|
||
m_bBeforeFirst = sal_True;
|
||
m_bAfterLast = sal_False;
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
else
|
||
{
|
||
m_bBeforeFirst = sal_False;
|
||
m_bAfterLast = m_nPosition > m_nRowCount;
|
||
moveWindow();
|
||
OSL_ENSURE(((m_nPosition - m_nStartPos) - 1) < (sal_Int32)m_pMatrix->size(),"Position is behind end()!");
|
||
m_aMatrixIter = calcPosition();
|
||
}
|
||
}
|
||
else
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
else
|
||
{
|
||
m_nPosition = row;
|
||
// the position flags
|
||
m_bBeforeFirst = sal_False;
|
||
checkPositionFlags();
|
||
|
||
if(!m_bAfterLast)
|
||
{
|
||
moveWindow();
|
||
checkPositionFlags();
|
||
if(!m_bAfterLast)
|
||
m_aMatrixIter = calcPosition();
|
||
else
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
else
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
|
||
return !(m_bAfterLast || m_bBeforeFirst);
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::relative( sal_Int32 rows )
|
||
{
|
||
sal_Bool bErg = sal_True;
|
||
if(rows)
|
||
{
|
||
sal_Int32 nNewPosition = m_nPosition + rows;
|
||
|
||
if ( m_bBeforeFirst && rows > 0 )
|
||
nNewPosition = rows;
|
||
else if ( m_bRowCountFinal && m_bAfterLast && rows < 0 )
|
||
nNewPosition = m_nRowCount + 1 + rows;
|
||
else
|
||
if ( m_bBeforeFirst || ( m_bRowCountFinal && m_bAfterLast ) )
|
||
throw SQLException( DBACORE_RESSTRING( RID_STR_NO_RELATIVE ), NULL, SQLSTATE_GENERAL, 1000, Any() );
|
||
if ( nNewPosition )
|
||
{
|
||
bErg = absolute( nNewPosition );
|
||
bErg = bErg && !isAfterLast() && !isBeforeFirst();
|
||
}
|
||
else
|
||
{
|
||
m_bBeforeFirst = sal_True;
|
||
bErg = sal_False;
|
||
}
|
||
}
|
||
return bErg;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::previous( )
|
||
{
|
||
sal_Bool bRet = sal_False;
|
||
if(!isBeforeFirst())
|
||
{
|
||
if(m_bAfterLast) // we stand after the last row so one before is the last row
|
||
bRet = last();
|
||
else
|
||
{
|
||
m_bAfterLast = sal_False;
|
||
--m_nPosition;
|
||
moveWindow();
|
||
OSL_ENSURE(((m_nPosition - m_nStartPos) - 1) < (sal_Int32)m_pMatrix->size(),"Position is behind end()!");
|
||
|
||
checkPositionFlags();
|
||
|
||
if(!m_nPosition)
|
||
{
|
||
m_bBeforeFirst = sal_True;
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
}
|
||
else
|
||
{
|
||
m_aMatrixIter = calcPosition();
|
||
bRet = (*m_aMatrixIter).isValid();
|
||
}
|
||
}
|
||
}
|
||
return bRet;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::refreshRow( )
|
||
{
|
||
if(isAfterLast())
|
||
throw SQLException(DBACORE_RESSTRING(RID_STR_NO_REFESH_AFTERLAST),NULL,SQLSTATE_GENERAL,1000,Any() );
|
||
OSL_ENSURE(m_aMatrixIter != m_pMatrix->end(),"refreshRow() called for invalid row!");
|
||
m_pCacheSet->refreshRow();
|
||
m_pCacheSet->fillValueRow(*m_aMatrixIter,m_nPosition);
|
||
if ( m_bNew )
|
||
{
|
||
cancelRowModification();
|
||
}
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::rowUpdated( )
|
||
{
|
||
return m_pCacheSet->rowUpdated();
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::rowInserted( )
|
||
{
|
||
return m_pCacheSet->rowInserted();
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
// XResultSetUpdate
|
||
sal_Bool ORowSetCache::insertRow( )
|
||
{
|
||
if ( !m_bNew || !m_aInsertRow->isValid() )
|
||
throw SQLException(DBACORE_RESSTRING(RID_STR_NO_MOVETOINSERTROW_CALLED),NULL,SQLSTATE_GENERAL,1000,Any() );
|
||
|
||
m_pCacheSet->insertRow(*m_aInsertRow,m_aUpdateTable);
|
||
|
||
sal_Bool bRet( rowInserted() );
|
||
if ( bRet )
|
||
{
|
||
++m_nRowCount;
|
||
Any aBookmark = ((*m_aInsertRow)->get())[0].makeAny();
|
||
m_bAfterLast = m_bBeforeFirst = sal_False;
|
||
if(aBookmark.hasValue())
|
||
moveToBookmark(aBookmark);
|
||
else
|
||
{
|
||
OSL_ENSURE(0,"There must be a bookmark after the row was inserted!");
|
||
}
|
||
}
|
||
return bRet;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::resetInsertRow(sal_Bool _bClearInsertRow)
|
||
{
|
||
if ( _bClearInsertRow )
|
||
clearInsertRow();
|
||
m_bNew = sal_False;
|
||
m_bModified = sal_False;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::cancelRowModification()
|
||
{
|
||
// clear the insertrow references -> implies that the current row of the rowset changes as well
|
||
ORowSetCacheMap::iterator aCacheIter = m_aCacheIterators.begin();
|
||
ORowSetCacheMap::iterator aCacheEnd = m_aCacheIterators.end();
|
||
for(;aCacheIter != aCacheEnd;++aCacheIter)
|
||
{
|
||
if ( aCacheIter->second.pRowSet->isInsertRow() && aCacheIter->second.aIterator == m_aInsertRow )
|
||
aCacheIter->second.aIterator = m_pMatrix->end();
|
||
} // for(;aCacheIter != aCacheEnd;++aCacheIter)
|
||
resetInsertRow(sal_False);
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::updateRow( ORowSetMatrix::iterator& _rUpdateRow )
|
||
{
|
||
if(isAfterLast() || isBeforeFirst())
|
||
throw SQLException(DBACORE_RESSTRING(RID_STR_NO_UPDATEROW),NULL,SQLSTATE_GENERAL,1000,Any() );
|
||
|
||
Any aBookmark = ((*_rUpdateRow)->get())[0].makeAny();
|
||
OSL_ENSURE(aBookmark.hasValue(),"Bookmark must have a value!");
|
||
// here we don't have to reposition our CacheSet, when we try to update a row,
|
||
// the row was already fetched
|
||
moveToBookmark(aBookmark);
|
||
m_pCacheSet->updateRow(*_rUpdateRow,*m_aMatrixIter,m_aUpdateTable);
|
||
// *(*m_aMatrixIter) = *(*_rUpdateRow);
|
||
// refetch the whole row
|
||
(*m_aMatrixIter) = NULL;
|
||
moveToBookmark(aBookmark);
|
||
|
||
// moveToBookmark((*(*m_aInsertRow))[0].makeAny());
|
||
// if(m_pCacheSet->rowUpdated())
|
||
// *m_aMatrixIter = m_aInsertRow;
|
||
m_bModified = sal_False;
|
||
// refreshRow( );
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
bool ORowSetCache::deleteRow( )
|
||
{
|
||
if(isAfterLast() || isBeforeFirst())
|
||
throw SQLException(DBACORE_RESSTRING(RID_STR_NO_DELETEROW),NULL,SQLSTATE_GENERAL,1000,Any() );
|
||
|
||
// m_pCacheSet->absolute(m_nPosition);
|
||
m_pCacheSet->deleteRow(*m_aMatrixIter,m_aUpdateTable);
|
||
if ( !m_pCacheSet->rowDeleted() )
|
||
return false;
|
||
|
||
--m_nRowCount;
|
||
OSL_ENSURE(((m_nPosition - m_nStartPos) - 1) < (sal_Int32)m_pMatrix->size(),"Position is behind end()!");
|
||
ORowSetMatrix::iterator aPos = calcPosition();
|
||
(*aPos) = NULL;
|
||
// (*m_pMatrix)[(m_nPosition - m_nStartPos)] = NULL; // set the deleted row to NULL
|
||
|
||
|
||
ORowSetMatrix::iterator aEnd = m_pMatrix->end();
|
||
for(++aPos;aPos != aEnd && aPos->isValid();++aPos)
|
||
{
|
||
*(aPos-1) = *aPos;
|
||
(*aPos) = NULL;
|
||
}
|
||
m_aMatrixIter = m_pMatrix->end();
|
||
|
||
--m_nPosition;
|
||
return true;
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::cancelRowUpdates( )
|
||
{
|
||
m_bNew = m_bModified = sal_False;
|
||
if(!m_nPosition)
|
||
{
|
||
OSL_ENSURE(0,"cancelRowUpdates:Invalid positions pos == 0");
|
||
::dbtools::throwFunctionSequenceException(NULL);
|
||
}
|
||
|
||
if(m_pCacheSet->absolute(m_nPosition))
|
||
m_pCacheSet->fillValueRow(*m_aMatrixIter,m_nPosition);
|
||
else
|
||
{
|
||
OSL_ENSURE(0,"cancelRowUpdates couldn't position right with absolute");
|
||
::dbtools::throwFunctionSequenceException(NULL);
|
||
}
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::moveToInsertRow( )
|
||
{
|
||
m_bNew = sal_True;
|
||
m_bUpdated = m_bAfterLast = sal_False;
|
||
|
||
m_aInsertRow = m_pInsertMatrix->begin();
|
||
if(!m_aInsertRow->isValid())
|
||
*m_aInsertRow = new ORowSetValueVector(m_xMetaData->getColumnCount());
|
||
|
||
// we don't unbound the bookmark column
|
||
ORowSetValueVector::Vector::iterator aIter = (*m_aInsertRow)->get().begin()+1;
|
||
ORowSetValueVector::Vector::iterator aEnd = (*m_aInsertRow)->get().end();
|
||
for(;aIter != aEnd;++aIter)
|
||
{
|
||
aIter->setBound(sal_False);
|
||
aIter->setModified(sal_False);
|
||
aIter->setNull();
|
||
}
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
ORowSetCacheIterator ORowSetCache::createIterator(ORowSetBase* _pRowSet)
|
||
{
|
||
|
||
ORowSetCacheIterator_Helper aHelper;
|
||
aHelper.aIterator = m_pMatrix->end();
|
||
aHelper.pRowSet = _pRowSet;
|
||
return ORowSetCacheIterator(m_aCacheIterators.insert(m_aCacheIterators.begin(),ORowSetCacheMap::value_type(m_aCacheIterators.size()+1,aHelper)),this,_pRowSet);
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
void ORowSetCache::rotateCacheIterator(ORowSetMatrix::difference_type _nDist)
|
||
{
|
||
if(_nDist)
|
||
{
|
||
// now correct the iterator in our iterator vector
|
||
ORowSetCacheMap::iterator aCacheIter = m_aCacheIterators.begin();
|
||
ORowSetCacheMap::iterator aCacheEnd = m_aCacheIterators.end();
|
||
for(;aCacheIter != aCacheEnd;++aCacheIter)
|
||
{
|
||
if ( !aCacheIter->second.pRowSet->isInsertRow()
|
||
&& aCacheIter->second.aIterator != m_pMatrix->end() && !m_bModified )
|
||
{
|
||
ptrdiff_t nDist = (aCacheIter->second.aIterator - m_pMatrix->begin());
|
||
if(nDist < _nDist)
|
||
{
|
||
aCacheIter->second.aIterator = m_pMatrix->end();
|
||
}
|
||
else
|
||
{
|
||
OSL_ENSURE((aCacheIter->second.aIterator - m_pMatrix->begin()) >= _nDist,"Invalid Dist value!");
|
||
aCacheIter->second.aIterator -= _nDist;
|
||
OSL_ENSURE(aCacheIter->second.aIterator >= m_pMatrix->begin()
|
||
&& aCacheIter->second.aIterator < m_pMatrix->end(),"Iterator out of area!");
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
// -------------------------------------------------------------------------
|
||
void ORowSetCache::setUpdateIterator(const ORowSetMatrix::iterator& _rOriginalRow)
|
||
{
|
||
m_aInsertRow = m_pInsertMatrix->begin();
|
||
if(!m_aInsertRow->isValid())
|
||
*m_aInsertRow = new ORowSetValueVector(m_xMetaData->getColumnCount());
|
||
|
||
(*(*m_aInsertRow)) = (*(*_rOriginalRow));
|
||
// we don't unbound the bookmark column
|
||
ORowSetValueVector::Vector::iterator aIter = (*m_aInsertRow)->get().begin();
|
||
ORowSetValueVector::Vector::iterator aEnd = (*m_aInsertRow)->get().end();
|
||
for(;aIter != aEnd;++aIter)
|
||
aIter->setModified(sal_False);
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
void ORowSetCache::checkPositionFlags()
|
||
{
|
||
if(m_bRowCountFinal)
|
||
{
|
||
m_bAfterLast = m_nPosition > m_nRowCount;
|
||
if(m_bAfterLast)
|
||
m_nPosition = 0;//m_nRowCount;
|
||
}
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
void ORowSetCache::checkUpdateConditions(sal_Int32 columnIndex)
|
||
{
|
||
if(m_bAfterLast || columnIndex >= (sal_Int32)(*m_aInsertRow)->get().size())
|
||
throwFunctionSequenceException(m_xSet.get());
|
||
}
|
||
//------------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::checkInnerJoin(const ::connectivity::OSQLParseNode *pNode,const Reference< XConnection>& _xConnection,const ::rtl::OUString& _sUpdateTableName)
|
||
{
|
||
sal_Bool bOk = sal_False;
|
||
if (pNode->count() == 3 && // Ausdruck is geklammert
|
||
SQL_ISPUNCTUATION(pNode->getChild(0),"(") &&
|
||
SQL_ISPUNCTUATION(pNode->getChild(2),")"))
|
||
{
|
||
bOk = checkInnerJoin(pNode->getChild(1),_xConnection,_sUpdateTableName);
|
||
}
|
||
else if ((SQL_ISRULE(pNode,search_condition) || SQL_ISRULE(pNode,boolean_term)) && // AND/OR-Verknuepfung:
|
||
pNode->count() == 3)
|
||
{
|
||
// nur AND Verkn<6B>pfung zulassen
|
||
if ( SQL_ISTOKEN(pNode->getChild(1),AND) )
|
||
bOk = checkInnerJoin(pNode->getChild(0),_xConnection,_sUpdateTableName)
|
||
&& checkInnerJoin(pNode->getChild(2),_xConnection,_sUpdateTableName);
|
||
}
|
||
else if (SQL_ISRULE(pNode,comparison_predicate))
|
||
{
|
||
// only the comparison of columns is allowed
|
||
DBG_ASSERT(pNode->count() == 3,"checkInnerJoin: Fehler im Parse Tree");
|
||
if (!(SQL_ISRULE(pNode->getChild(0),column_ref) &&
|
||
SQL_ISRULE(pNode->getChild(2),column_ref) &&
|
||
pNode->getChild(1)->getNodeType() == SQL_NODE_EQUAL))
|
||
{
|
||
bOk = sal_False;
|
||
}
|
||
::rtl::OUString sColumnName,sTableRange;
|
||
OSQLParseTreeIterator::getColumnRange( pNode->getChild(0), _xConnection, sColumnName, sTableRange );
|
||
bOk = sTableRange == _sUpdateTableName;
|
||
if ( !bOk )
|
||
{
|
||
OSQLParseTreeIterator::getColumnRange( pNode->getChild(2), _xConnection, sColumnName, sTableRange );
|
||
bOk = sTableRange == _sUpdateTableName;
|
||
}
|
||
}
|
||
return bOk;
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::checkJoin(const Reference< XConnection>& _xConnection,
|
||
const Reference< XSingleSelectQueryAnalyzer >& _xAnalyzer,
|
||
const ::rtl::OUString& _sUpdateTableName )
|
||
{
|
||
sal_Bool bOk = sal_False;
|
||
::rtl::OUString sSql = _xAnalyzer->getQuery();
|
||
::rtl::OUString sErrorMsg;
|
||
::connectivity::OSQLParser aSqlParser( m_aContext.getLegacyServiceFactory() );
|
||
::std::auto_ptr< ::connectivity::OSQLParseNode> pSqlParseNode( aSqlParser.parseTree(sErrorMsg,sSql));
|
||
if ( pSqlParseNode.get() && SQL_ISRULE(pSqlParseNode, select_statement) )
|
||
{
|
||
OSQLParseNode* pTableRefCommalist = pSqlParseNode->getByRule(::connectivity::OSQLParseNode::table_ref_commalist);
|
||
OSL_ENSURE(pTableRefCommalist,"NO tables why!?");
|
||
if(pTableRefCommalist && pTableRefCommalist->count() == 1)
|
||
{
|
||
// we found only one element so it must some kind of join here
|
||
OSQLParseNode* pJoin = pTableRefCommalist->getByRule(::connectivity::OSQLParseNode::qualified_join);
|
||
if(pJoin)
|
||
{ // we are only intereseted in qualified joins like RIGHT or LEFT
|
||
OSQLParseNode* pJoinType = pJoin->getChild(1);
|
||
OSQLParseNode* pOuterType = NULL;
|
||
if(SQL_ISRULE(pJoinType,join_type) && pJoinType->count() == 2)
|
||
pOuterType = pJoinType->getChild(0);
|
||
else if(SQL_ISRULE(pJoinType,outer_join_type))
|
||
pOuterType = pJoinType;
|
||
|
||
sal_Bool bCheck = sal_False;
|
||
sal_Bool bLeftSide = sal_False;
|
||
if(pOuterType)
|
||
{ // found outer join
|
||
bLeftSide = SQL_ISTOKEN(pOuterType->getChild(0),LEFT);
|
||
bCheck = bLeftSide || SQL_ISTOKEN(pOuterType->getChild(0),RIGHT);
|
||
}
|
||
|
||
if(bCheck)
|
||
{ // here we know that we have to check on which side our table resides
|
||
const OSQLParseNode* pTableRef = pJoin->getByRule(::connectivity::OSQLParseNode::qualified_join);
|
||
if(bLeftSide)
|
||
pTableRef = pJoin->getChild(0);
|
||
else
|
||
pTableRef = pJoin->getChild(3);
|
||
OSL_ENSURE(SQL_ISRULE(pTableRef,table_ref),"Must be a tableref here!");
|
||
|
||
::rtl::OUString sTableRange = OSQLParseNode::getTableRange(pTableRef);
|
||
if(!sTableRange.getLength())
|
||
pTableRef->getChild(0)->parseNodeToStr( sTableRange, _xConnection, NULL, sal_False, sal_False );
|
||
bOk = sTableRange == _sUpdateTableName;
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
OSQLParseNode* pWhereOpt = pSqlParseNode->getChild(3)->getChild(1);
|
||
if ( pWhereOpt && !pWhereOpt->isLeaf() )
|
||
bOk = checkInnerJoin(pWhereOpt->getChild(1),_xConnection,_sUpdateTableName);
|
||
}
|
||
}
|
||
return bOk;
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
void ORowSetCache::clearInsertRow()
|
||
{
|
||
// we don't unbound the bookmark column
|
||
if ( m_aInsertRow != m_pInsertMatrix->end() && m_aInsertRow->isValid() )
|
||
{
|
||
ORowSetValueVector::Vector::iterator aIter = (*m_aInsertRow)->get().begin()+1;
|
||
ORowSetValueVector::Vector::iterator aEnd = (*m_aInsertRow)->get().end();
|
||
for(;aIter != aEnd;++aIter)
|
||
{
|
||
aIter->setBound(sal_False);
|
||
aIter->setModified(sal_False);
|
||
aIter->setNull();
|
||
} // for(;aIter != (*m_aInsertRow)->end();++aIter)
|
||
}
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
ORowSetMatrix::iterator ORowSetCache::calcPosition() const
|
||
{
|
||
sal_Int32 nValue = (m_nPosition - m_nStartPos) - 1;
|
||
CHECK_MATRIX_POS(nValue);
|
||
return ( nValue < 0 || nValue >= static_cast<sal_Int32>(m_pMatrix->size()) ) ? m_pMatrix->end() : (m_pMatrix->begin() + nValue);
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
|
||
TORowSetOldRowHelperRef ORowSetCache::registerOldRow()
|
||
{
|
||
TORowSetOldRowHelperRef pRef = new ORowSetOldRowHelper(ORowSetRow());
|
||
m_aOldRows.push_back(pRef);
|
||
return pRef;
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
void ORowSetCache::deregisterOldRow(const TORowSetOldRowHelperRef& _rRow)
|
||
{
|
||
TOldRowSetRows::iterator aOldRowEnd = m_aOldRows.end();
|
||
for (TOldRowSetRows::iterator aOldRowIter = m_aOldRows.begin(); aOldRowIter != aOldRowEnd; ++aOldRowIter)
|
||
{
|
||
if ( aOldRowIter->getBodyPtr() == _rRow.getBodyPtr() )
|
||
{
|
||
m_aOldRows.erase(aOldRowIter);
|
||
break;
|
||
}
|
||
|
||
}
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::reFillMatrix(sal_Int32 _nNewStartPos,sal_Int32 _nNewEndPos)
|
||
{
|
||
TOldRowSetRows::iterator aOldRowEnd = m_aOldRows.end();
|
||
for (TOldRowSetRows::iterator aOldRowIter = m_aOldRows.begin(); aOldRowIter != aOldRowEnd; ++aOldRowIter)
|
||
{
|
||
if ( aOldRowIter->isValid() && aOldRowIter->getBody().getRow().isValid() )
|
||
aOldRowIter->getBody().setRow(new ORowSetValueVector(aOldRowIter->getBody().getRow().getBody()) );
|
||
}
|
||
sal_Int32 nNewSt = _nNewStartPos;
|
||
sal_Bool bRet = fillMatrix(nNewSt,_nNewEndPos);
|
||
m_nStartPos = nNewSt - 1;
|
||
rotateCacheIterator(static_cast<sal_Int16>(m_nFetchSize+1)); // forces that every iterator will be set to null
|
||
return bRet;
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
sal_Bool ORowSetCache::fill(ORowSetMatrix::iterator& _aIter,const ORowSetMatrix::iterator& _aEnd,sal_Int32& _nPos,sal_Bool _bCheck)
|
||
{
|
||
sal_Int32 nColumnCount = m_xMetaData->getColumnCount();
|
||
for(; _bCheck && _aIter != _aEnd;)
|
||
{
|
||
if ( !_aIter->isValid() )
|
||
*_aIter = new ORowSetValueVector(nColumnCount);
|
||
else
|
||
{
|
||
TOldRowSetRows::iterator aOldRowEnd = m_aOldRows.end();
|
||
for (TOldRowSetRows::iterator aOldRowIter = m_aOldRows.begin(); aOldRowIter != aOldRowEnd; ++aOldRowIter)
|
||
{
|
||
if ( aOldRowIter->getBody().getRow().isEqualBody(*_aIter) )
|
||
*_aIter = new ORowSetValueVector(nColumnCount);
|
||
}
|
||
}
|
||
m_pCacheSet->fillValueRow(*_aIter++,++_nPos);
|
||
_bCheck = m_pCacheSet->next();
|
||
}
|
||
return _bCheck;
|
||
}
|
||
// -----------------------------------------------------------------------------
|
||
|
||
|
||
|
||
|
||
|