1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_dbaccess.hxx"
26 #ifndef DBAUI_QUERYDESIGNVIEW_HXX
27 #include "QueryDesignView.hxx"
28 #endif
29 #ifndef DBAUI_QUERYTABLEVIEW_HXX
30 #include "QueryTableView.hxx"
31 #endif
32 #ifndef DBAUI_QUERY_TABLEWINDOW_HXX
33 #include "QTableWindow.hxx"
34 #endif
35 #ifndef _SV_TOOLBOX_HXX
36 #include <vcl/toolbox.hxx>
37 #endif
38 #ifndef DBAUI_QUERYCONTROLLER_HXX
39 #include "querycontroller.hxx"
40 #endif
41 #ifndef _SV_SPLIT_HXX
42 #include <vcl/split.hxx>
43 #endif
44 #ifndef _UNDO_HXX
45 #include <svl/undo.hxx>
46 #endif
47 #ifndef TOOLS_DIAGNOSE_EX_H
48 #include <tools/diagnose_ex.h>
49 #endif
50 #ifndef DBAUI_QYDLGTAB_HXX
51 #include "adtabdlg.hxx"
52 #endif
53 #ifndef _SV_SVAPP_HXX
54 #include <vcl/svapp.hxx>
55 #endif
56 #ifndef _SV_COMBOBOX_HXX
57 #include <vcl/combobox.hxx>
58 #endif
59 #ifndef _SV_MSGBOX_HXX
60 #include <vcl/msgbox.hxx>
61 #endif
62 #ifndef DBACCESS_UI_BROWSER_ID_HXX
63 #include "browserids.hxx"
64 #endif
65 #ifndef DBAUI_QUERYDESIGN_OSELECTIONBROWSEBOX_HXX
66 #include "SelectionBrowseBox.hxx"
67 #endif
68 #ifndef _DBU_QRY_HRC_
69 #include "dbu_qry.hrc"
70 #endif
71 #ifndef _UTL_CONFIGMGR_HXX_
72 #include <unotools/configmgr.hxx>
73 #endif
74 #ifndef _COMPHELPER_TYPES_HXX_
75 #include <comphelper/types.hxx>
76 #endif
77 #ifndef _CONNECTIVITY_DBTOOLS_HXX_
78 #include <connectivity/dbtools.hxx>
79 #endif
80 #ifndef _DBHELPER_DBEXCEPTION_HXX_
81 #include <connectivity/dbexception.hxx>
82 #endif
83 #ifndef _COM_SUN_STAR_I18N_XLOCALEDATA_HPP_
84 #include <com/sun/star/i18n/XLocaleData.hpp>
85 #endif
86 #ifndef _COM_SUN_STAR_SDBC_DATATYPE_HPP_
87 #include <com/sun/star/sdbc/DataType.hpp>
88 #endif
89 #ifndef _COM_SUN_STAR_CONTAINER_XNAMEACCESS_HPP_
90 #include <com/sun/star/container/XNameAccess.hpp>
91 #endif
92 #ifndef _COM_SUN_STAR_SDBC_COLUMNVALUE_HPP_
93 #include <com/sun/star/sdbc/ColumnValue.hpp>
94 #endif
95 #ifndef _CONNECTIVITY_PCOLUMN_HXX_
96 #include <connectivity/PColumn.hxx>
97 #endif
98 #ifndef DBAUI_QUERYTABLECONNECTION_HXX
99 #include "QTableConnection.hxx"
100 #endif
101 #ifndef DBAUI_CONNECTIONLINE_HXX
102 #include "ConnectionLine.hxx"
103 #endif
104 #ifndef DBAUI_CONNECTIONLINEDATA_HXX
105 #include "ConnectionLineData.hxx"
106 #endif
107 #ifndef DBAUI_QTABLECONNECTIONDATA_HXX
108 #include "QTableConnectionData.hxx"
109 #endif
110 #ifndef DBACCESS_SHARED_DBUSTRINGS_HRC
111 #include "dbustrings.hrc"
112 #endif
113 #ifndef _COMPHELPER_EXTRACT_HXX_
114 #include <comphelper/extract.hxx>
115 #endif
116 #ifndef DBAUI_TOOLS_HXX
117 #include "UITools.hxx"
118 #endif
119 #ifndef DBAUI_QUERYCONTAINERWINDOW_HXX
120 #include "querycontainerwindow.hxx"
121 #endif
122 #ifndef DBAUI_QUERYTABLEVIEW_HXX
123 #include "QueryTableView.hxx"
124 #endif
125 #ifndef _DBAUI_SQLMESSAGE_HXX_
126 #include "sqlmessage.hxx"
127 #endif
128 #ifndef INCLUDED_SVTOOLS_SYSLOCALE_HXX
129 #include <unotools/syslocale.hxx>
130 #endif
131 
132 using namespace ::dbaui;
133 using namespace ::utl;
134 using namespace ::connectivity;
135 using namespace ::dbtools;
136 using namespace ::com::sun::star::uno;
137 using namespace ::com::sun::star::lang;
138 using namespace ::com::sun::star::i18n;
139 using namespace ::com::sun::star::sdbc;
140 using namespace ::com::sun::star::beans;
141 using namespace ::com::sun::star::container;
142 
143 #define SQL_ISRULEOR2(pParseNode, e1,e2) 	((pParseNode)->isRule() && (\
144 											(pParseNode)->getRuleID() == OSQLParser::RuleID(OSQLParseNode::e1) || \
145 											(pParseNode)->getRuleID() == OSQLParser::RuleID(OSQLParseNode::e2)))
146 
147 // here we define our functions used in the anonymous namespace to get our header file smaller
148 // please look at the book LargeScale C++ to know why
149 namespace
150 {
151 	static const ::rtl::OUString C_AND = ::rtl::OUString::createFromAscii(" AND ");
152 	static const ::rtl::OUString C_OR  = ::rtl::OUString::createFromAscii(" OR ");
153 
154 	// forward declarations
155 	sal_Bool InsertJoin(	const OQueryDesignView* _pView,
156 							const ::connectivity::OSQLParseNode *pNode);
157 
158 	SqlParseError InstallFields(OQueryDesignView* _pView,
159 								const ::connectivity::OSQLParseNode* pNode,
160 								OJoinTableView::OTableWindowMap* pTabList );
161 
162 	SqlParseError GetGroupCriteria(	OQueryDesignView* _pView,
163 									OSelectionBrowseBox* _pSelectionBrw,
164 									const ::connectivity::OSQLParseNode* pSelectRoot );
165 
166 	SqlParseError GetHavingCriteria(OQueryDesignView* _pView,
167 									OSelectionBrowseBox* _pSelectionBrw,
168 									const ::connectivity::OSQLParseNode* pSelectRoot,
169 									sal_uInt16& rLevel );
170 
171 	SqlParseError GetOrderCriteria(	OQueryDesignView* _pView,
172 									OSelectionBrowseBox* _pSelectionBrw,
173 									const ::connectivity::OSQLParseNode* pParseRoot );
174 
175 	SqlParseError AddFunctionCondition(OQueryDesignView* _pView,
176 									OSelectionBrowseBox* _pSelectionBrw,
177 									const ::connectivity::OSQLParseNode * pCondition,
178 									const sal_uInt16 nLevel,
179 									sal_Bool bHaving,
180                                     bool _bAddOrOnOneLine);
181 
182 	//------------------------------------------------------------------------------
183 	::rtl::OUString quoteTableAlias(sal_Bool _bQuote, const ::rtl::OUString& _sAliasName, const ::rtl::OUString& _sQuote)
184 	{
185 		::rtl::OUString sRet;
186 		if ( _bQuote && _sAliasName.getLength() )
187 		{
188 			sRet = ::dbtools::quoteName(_sQuote,_sAliasName);
189 			const static ::rtl::OUString sTableSeparater('.');
190 			sRet += sTableSeparater;
191 		}
192 		return sRet;
193 	}
194     //------------------------------------------------------------------------------
195     ::rtl::OUString getTableRange(const OQueryDesignView* _pView,const ::connectivity::OSQLParseNode* _pTableRef)
196     {
197         Reference< XConnection> xConnection = static_cast<OQueryController&>(_pView->getController()).getConnection();
198         ::rtl::OUString sTableRange;
199         if ( _pTableRef )
200         {
201             sTableRange = ::connectivity::OSQLParseNode::getTableRange(_pTableRef);
202             if ( !sTableRange.getLength() )
203                 _pTableRef->parseNodeToStr(sTableRange,xConnection,NULL,sal_False,sal_False);
204         }
205         return sTableRange;
206     }
207     //------------------------------------------------------------------------------
208     void insertConnection(const OQueryDesignView* _pView,const EJoinType& _eJoinType,OTableFieldDescRef _aDragLeft,OTableFieldDescRef _aDragRight,bool _bNatural = false)
209     {
210         OQueryTableView* pTableView = static_cast<OQueryTableView*>(_pView->getTableView());
211 		OQueryTableConnection* pConn = static_cast<OQueryTableConnection*>( pTableView->GetTabConn(static_cast<OTableWindow*>(_aDragLeft->GetTabWindow()),static_cast<OTableWindow*>(_aDragRight->GetTabWindow()),true));
212 
213 		if ( !pConn )
214 		{
215 			OQueryTableConnectionData* pInfoData = new OQueryTableConnectionData();
216             TTableConnectionData::value_type aInfoData(pInfoData);
217 			pInfoData->InitFromDrag(_aDragLeft, _aDragRight);
218 			pInfoData->SetJoinType(_eJoinType);
219 
220             if ( _bNatural )
221             {
222                 aInfoData->ResetConnLines();
223                 pInfoData->setNatural(_bNatural);
224                 try
225                 {
226                     Reference<XNameAccess> xReferencedTableColumns(aInfoData->getReferencedTable()->getColumns());
227                     Sequence< ::rtl::OUString> aSeq = aInfoData->getReferencingTable()->getColumns()->getElementNames();
228                     const ::rtl::OUString* pIter = aSeq.getConstArray();
229                     const ::rtl::OUString* pEnd	  = pIter + aSeq.getLength();
230                     for(;pIter != pEnd;++pIter)
231                     {
232                         if ( xReferencedTableColumns->hasByName(*pIter) )
233                             aInfoData->AppendConnLine(*pIter,*pIter);
234                     }
235                 }
236                 catch( const Exception& )
237                 {
238                     DBG_UNHANDLED_EXCEPTION();
239                 }
240             }
241 
242 			OQueryTableConnection aInfo(pTableView, aInfoData);
243 			// da ein OQueryTableConnection-Objekt nie den Besitz der uebergebenen Daten uebernimmt, sondern sich nur den Zeiger merkt,
244 			// ist dieser Zeiger auf eine lokale Variable hier unkritisch, denn aInfoData und aInfo haben die selbe Lebensdauer
245 			pTableView->NotifyTabConnection( aInfo );
246 		}
247 		else
248 		{
249 			::rtl::OUString aSourceFieldName(_aDragLeft->GetField());
250 			::rtl::OUString aDestFieldName(_aDragRight->GetField());
251 			// the connection could point on the other side
252 			if(pConn->GetSourceWin() == _aDragRight->GetTabWindow())
253 			{
254 				::rtl::OUString aTmp(aSourceFieldName);
255 				aSourceFieldName = aDestFieldName;
256 				aDestFieldName = aTmp;
257 			}
258 			pConn->GetData()->AppendConnLine( aSourceFieldName,aDestFieldName);
259 			pConn->UpdateLineList();
260 			// Modified-Flag
261 			//	SetModified();
262 			// und neu zeichnen
263 			pConn->RecalcLines();
264 				// fuer das unten folgende Invalidate muss ich dieser neuen Connection erst mal die Moeglichkeit geben,
265 				// ihr BoundingRect zu ermitteln
266 			pConn->InvalidateConnection();
267 		}
268     }
269 	//------------------------------------------------------------------------------
270 	::rtl::OUString ParseCondition(	OQueryController& rController
271 									,const ::connectivity::OSQLParseNode* pCondition
272 									,const ::rtl::OUString _sDecimal
273 									,const ::com::sun::star::lang::Locale& _rLocale
274 									,sal_uInt32 _nStartIndex)
275 	{
276 		::rtl::OUString	aCondition;
277 		Reference< XConnection> xConnection = rController.getConnection();
278 		if ( xConnection.is() )
279 		{
280 			sal_uInt32 nCount = pCondition->count();
281 			for(sal_uInt32 i = _nStartIndex ; i < nCount ; ++i)
282 				pCondition->getChild(i)->parseNodeToPredicateStr(aCondition,
283 								xConnection,
284 								rController.getNumberFormatter(),
285 								_rLocale,
286 								static_cast<sal_Char>(_sDecimal.toChar()),
287 								&rController.getParser().getContext());
288 		}
289 		return aCondition;
290 	}
291 	//------------------------------------------------------------------------------
292 	SqlParseError FillOuterJoins(OQueryDesignView* _pView,
293 								const ::connectivity::OSQLParseNode* pTableRefList)
294 	{
295 		SqlParseError eErrorCode = eOk;
296 		sal_uInt32 nCount = pTableRefList->count();
297 		sal_Bool bError = sal_False;
298 		for (sal_uInt32 i=0; !bError && i < nCount; ++i)
299 		{
300 			const ::connectivity::OSQLParseNode* pParseNode = pTableRefList->getChild(i);
301 			const ::connectivity::OSQLParseNode* pJoinNode = NULL;
302 
303 			if ( SQL_ISRULE( pParseNode, qualified_join ) || SQL_ISRULE( pParseNode, joined_table ) || SQL_ISRULE( pParseNode, cross_union ) )
304 				pJoinNode = pParseNode;
305 			else if(	SQL_ISRULE(pParseNode,table_ref)
306 					&&	pParseNode->count() == 4 ) // '{' SQL_TOKEN_OJ joined_table '}'
307 				pJoinNode = pParseNode->getChild(2);
308 
309 			if ( pJoinNode )
310 			{
311 				if ( !InsertJoin(_pView,pJoinNode) )
312 					bError = sal_True;
313 			}
314 		}
315 		// check if error occurred
316 		if ( bError )
317 			eErrorCode = eIllegalJoin;
318 
319 		return eErrorCode;
320 	}
321 	// -----------------------------------------------------------------------------
322 
323 	/** FillDragInfo fills the field description out of the table
324 	*/
325 	//------------------------------------------------------------------------------
326 	SqlParseError FillDragInfo(	const OQueryDesignView* _pView,
327 							const ::connectivity::OSQLParseNode* pColumnRef,
328 							OTableFieldDescRef& _rDragInfo)
329 	{
330 		SqlParseError eErrorCode = eOk;
331 
332 		sal_Bool bErg = sal_False;
333 
334 		::rtl::OUString aTableRange,aColumnName;
335 		sal_uInt16 nCntAccount;
336 		::connectivity::OSQLParseTreeIterator& rParseIter = static_cast<OQueryController&>(_pView->getController()).getParseIterator();
337 		rParseIter.getColumnRange( pColumnRef, aColumnName, aTableRange );
338 
339 		if ( aTableRange.getLength() )
340 		{
341 			OQueryTableWindow*	pSTW = static_cast<OQueryTableView*>(_pView->getTableView())->FindTable( aTableRange );
342 			bErg = (pSTW && pSTW->ExistsField( aColumnName, _rDragInfo ) );
343 		}
344 		if ( !bErg )
345         {
346 			bErg = static_cast<OQueryTableView*>(_pView->getTableView())->FindTableFromField(aColumnName, _rDragInfo, nCntAccount);
347             if ( !bErg )
348                 bErg = _pView->HasFieldByAliasName(aColumnName, _rDragInfo);
349         }
350 		if ( !bErg )
351 		{
352 			eErrorCode = eColumnNotFound;
353 			String sError(ModuleRes(STR_QRY_COLUMN_NOT_FOUND));
354 			sError.SearchAndReplaceAscii("$name$",aColumnName);
355 			_pView->getController().appendError( sError );
356 
357 			try
358 			{
359 				Reference<XDatabaseMetaData> xMeta = _pView->getController().getConnection()->getMetaData();
360                 if ( xMeta.is() && xMeta->supportsMixedCaseQuotedIdentifiers() )
361                     _pView->getController().appendError( String( ModuleRes( STR_QRY_CHECK_CASESENSITIVE ) ) );
362 			}
363 			catch(Exception&)
364 			{
365 			}
366 		}
367 
368 		return eErrorCode;
369 	}
370 	//------------------------------------------------------------------------------
371 	::rtl::OUString BuildJoinCriteria(	const Reference< XConnection>& _xConnection,
372 										OConnectionLineDataVec* pLineDataList,
373 										OQueryTableConnectionData* pData)
374 	{
375 		::rtl::OUStringBuffer aCondition;
376 		if ( _xConnection.is() )
377         {
378 		    OConnectionLineDataVec::iterator aIter = pLineDataList->begin();
379             OConnectionLineDataVec::iterator aEnd = pLineDataList->end();
380 		    try
381 		    {
382 			    const Reference< XDatabaseMetaData >  xMetaData = _xConnection->getMetaData();
383 			    const ::rtl::OUString aQuote = xMetaData->getIdentifierQuoteString();
384                 const ::rtl::OUString sEqual(RTL_CONSTASCII_USTRINGPARAM(" = "));
385 
386 			    for(;aIter != aEnd;++aIter)
387 			    {
388 				    OConnectionLineDataRef pLineData = *aIter;
389 				    if(aCondition.getLength())
390 					    aCondition.append(C_AND);
391 				    aCondition.append(quoteTableAlias(sal_True,pData->GetAliasName(JTCS_FROM),aQuote));
392 				    aCondition.append(::dbtools::quoteName(aQuote, pLineData->GetFieldName(JTCS_FROM) ));
393 				    aCondition.append(sEqual);
394 				    aCondition.append(quoteTableAlias(sal_True,pData->GetAliasName(JTCS_TO),aQuote));
395 				    aCondition.append(::dbtools::quoteName(aQuote, pLineData->GetFieldName(JTCS_TO) ));
396 			    }
397 		    }
398 		    catch(SQLException&)
399 		    {
400 			    OSL_ASSERT(!"Failure while building Join criteria!");
401 		    }
402         }
403 
404 		return aCondition.makeStringAndClear();
405 	}
406 	//------------------------------------------------------------------------------
407 	/** JoinCycle looks for a join cycle and append it to the string
408 		@param	_xConnection	the connection
409 		@param	_pEntryConn		the table connection which holds the data
410 		@param	_pEntryTabTo	the corresponding table window
411 		@param	_rJoin			the String which will contain the resulting string
412 	*/
413 	void JoinCycle(	const Reference< XConnection>& _xConnection,
414 					OQueryTableConnection* _pEntryConn,
415 					const OQueryTableWindow* _pEntryTabTo,
416 					::rtl::OUString& _rJoin )
417 	{
418 		OSL_ENSURE(_pEntryConn,"TableConnection can not be null!");
419 
420 		OQueryTableConnectionData* pData = static_cast< OQueryTableConnectionData*>(_pEntryConn->GetData().get());
421 		if ( pData->GetJoinType() != INNER_JOIN && _pEntryTabTo->ExistsAVisitedConn() )
422 		{
423 			sal_Bool bBrace = sal_False;
424 			if(_rJoin.getLength() && _rJoin.lastIndexOf(')') == (_rJoin.getLength()-1))
425 			{
426 				bBrace = sal_True;
427 				_rJoin = _rJoin.replaceAt(_rJoin.getLength()-1,1,::rtl::OUString(' '));
428 			}
429 			(_rJoin += C_AND) += BuildJoinCriteria(_xConnection,pData->GetConnLineDataList(),pData);
430 			if(bBrace)
431 				_rJoin += ::rtl::OUString(')');
432 			_pEntryConn->SetVisited(sal_True);
433 		}
434 	}
435 	//------------------------------------------------------------------------------
436 	::rtl::OUString BuildTable(	const Reference< XConnection>& _xConnection,
437 								const OQueryTableWindow* pEntryTab,
438                                 bool _bForce = false
439 								)
440 	{
441 		::rtl::OUString aDBName(pEntryTab->GetComposedName());
442 
443 		//	Reference< XConnection> xConnection = static_cast<OQueryController&>(_pView->getController()).getConnection();
444 		if( _xConnection.is() )
445 		{
446 			try
447 			{
448 				Reference< XDatabaseMetaData >  xMetaData = _xConnection->getMetaData();
449 
450                 ::rtl::OUString sCatalog, sSchema, sTable;
451                 ::dbtools::qualifiedNameComponents( xMetaData, aDBName, sCatalog, sSchema, sTable, ::dbtools::eInDataManipulation );
452 				::rtl::OUString aTableListStr = ::dbtools::composeTableNameForSelect( _xConnection, sCatalog, sSchema, sTable );
453 
454 				::rtl::OUString aQuote = xMetaData->getIdentifierQuoteString();
455 				if ( _bForce || isAppendTableAliasEnabled( _xConnection ) || pEntryTab->GetAliasName() != aDBName )
456 				{
457                     aTableListStr += ::rtl::OUString::createFromAscii(" ");
458                     if ( generateAsBeforeTableAlias( _xConnection ) )
459                         aTableListStr += ::rtl::OUString::createFromAscii("AS ");
460 					aTableListStr += ::dbtools::quoteName( aQuote, pEntryTab->GetAliasName() );
461 				}
462 				aDBName = aTableListStr;
463 			}
464 			catch(const SQLException&)
465 			{
466                 DBG_UNHANDLED_EXCEPTION();
467 			}
468 		}
469 		return aDBName;
470 	}
471 	//------------------------------------------------------------------------------
472 	::rtl::OUString BuildJoin(	const Reference< XConnection>& _xConnection,
473 								const ::rtl::OUString& rLh,
474 								const ::rtl::OUString& rRh,
475 								OQueryTableConnectionData* pData)
476 	{
477 
478 		String aErg(rLh);
479         if ( pData->isNatural() && pData->GetJoinType() != CROSS_JOIN )
480             aErg.AppendAscii(" NATURAL ");
481 		switch(pData->GetJoinType())
482 		{
483 			case LEFT_JOIN:
484 				aErg.AppendAscii(" LEFT OUTER ");
485 				break;
486 			case RIGHT_JOIN:
487 				aErg.AppendAscii(" RIGHT OUTER ");
488 				break;
489             case CROSS_JOIN:
490                 OSL_ENSURE(!pData->isNatural(),"OQueryDesignView::BuildJoin: This should not happen!");
491 				aErg.AppendAscii(" CROSS ");
492 				break;
493 			case INNER_JOIN:
494                 OSL_ENSURE(pData->isNatural(),"OQueryDesignView::BuildJoin: This should not happen!");
495 				aErg.AppendAscii(" INNER ");
496 				break;
497 			default:
498 				aErg.AppendAscii(" FULL OUTER ");
499 				break;
500 		}
501 		aErg.AppendAscii("JOIN ");
502 		aErg += String(rRh);
503         if ( CROSS_JOIN != pData->GetJoinType() && !pData->isNatural() )
504         {
505 		    aErg.AppendAscii(" ON ");
506 		    aErg += String(BuildJoinCriteria(_xConnection,pData->GetConnLineDataList(),pData));
507         }
508 
509 		return aErg;
510 	}
511 	//------------------------------------------------------------------------------
512 	::rtl::OUString BuildJoin(	const Reference< XConnection>& _xConnection,
513 								OQueryTableWindow* pLh,
514 								OQueryTableWindow* pRh,
515 								OQueryTableConnectionData* pData
516 								)
517 	{
518         bool bForce = pData->GetJoinType() == CROSS_JOIN || pData->isNatural();
519 		return BuildJoin(_xConnection,BuildTable(_xConnection,pLh,bForce),BuildTable(_xConnection,pRh,bForce),pData);
520 	}
521 	//------------------------------------------------------------------------------
522 	::rtl::OUString BuildJoin(	const Reference< XConnection>& _xConnection,
523 								const ::rtl::OUString &rLh,
524 								OQueryTableWindow* pRh,
525 								OQueryTableConnectionData* pData
526 								)
527 	{
528 		return BuildJoin(_xConnection,rLh,BuildTable(_xConnection,pRh),pData);
529 	}
530 	//------------------------------------------------------------------------------
531 	::rtl::OUString BuildJoin(	const Reference< XConnection>& _xConnection,
532 								OQueryTableWindow* pLh,
533 								const ::rtl::OUString &rRh,
534 								OQueryTableConnectionData* pData
535 								)
536 	{
537 		return BuildJoin(_xConnection,BuildTable(_xConnection,pLh),rRh,pData);
538 	}
539 	//------------------------------------------------------------------------------
540 	void GetNextJoin(	const Reference< XConnection>& _xConnection,
541 						OQueryTableConnection* pEntryConn,
542 						OQueryTableWindow* pEntryTabTo,
543 						::rtl::OUString &aJoin)
544 	{
545 		OQueryTableConnectionData* pEntryConnData = static_cast<OQueryTableConnectionData*>(pEntryConn->GetData().get());
546         if ( pEntryConnData->GetJoinType() == INNER_JOIN && !pEntryConnData->isNatural() )
547 			return;
548 
549 		//	Reference< XConnection> xConnection = static_cast<OQueryController&>(_pView->getController()).getConnection();
550 
551 		if(!aJoin.getLength())
552 		{
553 			OQueryTableWindow* pEntryTabFrom = static_cast<OQueryTableWindow*>(pEntryConn->GetSourceWin());
554 			aJoin = BuildJoin(_xConnection,pEntryTabFrom,pEntryTabTo,pEntryConnData);
555 		}
556 		else if(pEntryTabTo == pEntryConn->GetDestWin())
557 		{
558 			aJoin = BuildJoin(_xConnection,aJoin,pEntryTabTo,pEntryConnData);
559 		}
560 		else if(pEntryTabTo == pEntryConn->GetSourceWin())
561 		{
562 			aJoin = BuildJoin(_xConnection,pEntryTabTo,aJoin,pEntryConnData);
563 		}
564 
565 		pEntryConn->SetVisited(sal_True);
566 
567 		// first search for the "to" window
568 		const ::std::vector<OTableConnection*>* pConnections = pEntryConn->GetParent()->getTableConnections();
569 		::std::vector<OTableConnection*>::const_iterator aIter = pConnections->begin();
570         ::std::vector<OTableConnection*>::const_iterator aEnd = pConnections->end();
571 		for(;aIter != aEnd;++aIter)
572 		{
573 			OQueryTableConnection* pNext = static_cast<OQueryTableConnection*>(*aIter);
574 			if(!pNext->IsVisited() && (pNext->GetSourceWin() == pEntryTabTo || pNext->GetDestWin() == pEntryTabTo))
575 			{
576 				OQueryTableWindow* pEntryTab = pNext->GetSourceWin() == pEntryTabTo ? static_cast<OQueryTableWindow*>(pNext->GetDestWin()) : static_cast<OQueryTableWindow*>(pNext->GetSourceWin());
577 				// exists there a connection to a OQueryTableWindow that holds a connection that has been already visited
578 				JoinCycle(_xConnection,pNext,pEntryTab,aJoin);
579 				if(!pNext->IsVisited())
580 					GetNextJoin(_xConnection,pNext,pEntryTab,aJoin);
581 			}
582 		}
583 
584 		// when nothing found found look for the "from" window
585 		if(aIter == aEnd)
586 		{
587 			OQueryTableWindow* pEntryTabFrom = static_cast<OQueryTableWindow*>(pEntryConn->GetSourceWin());
588 			aIter = pConnections->begin();
589 			for(;aIter != aEnd;++aIter)
590 			{
591 				OQueryTableConnection* pNext = static_cast<OQueryTableConnection*>(*aIter);
592 				if(!pNext->IsVisited() && (pNext->GetSourceWin() == pEntryTabFrom || pNext->GetDestWin() == pEntryTabFrom))
593 				{
594 					OQueryTableWindow* pEntryTab = pNext->GetSourceWin() == pEntryTabFrom ? static_cast<OQueryTableWindow*>(pNext->GetDestWin()) : static_cast<OQueryTableWindow*>(pNext->GetSourceWin());
595 					// exists there a connection to a OQueryTableWindow that holds a connection that has been already visited
596 					JoinCycle(_xConnection,pNext,pEntryTab,aJoin);
597 					if(!pNext->IsVisited())
598 						GetNextJoin(_xConnection,pNext,pEntryTab,aJoin);
599 				}
600 			}
601 		}
602 	}
603 	//------------------------------------------------------------------------------
604 	SqlParseError InsertJoinConnection(	const OQueryDesignView* _pView,
605 									const ::connectivity::OSQLParseNode *pNode,
606 									const EJoinType& _eJoinType,
607                                     const ::connectivity::OSQLParseNode *pLeftTable,
608                                     const ::connectivity::OSQLParseNode *pRightTable)
609 	{
610 		SqlParseError eErrorCode = eOk;
611 		if (pNode->count() == 3 &&	// Ausdruck is geklammert
612 			SQL_ISPUNCTUATION(pNode->getChild(0),"(") &&
613 			SQL_ISPUNCTUATION(pNode->getChild(2),")"))
614 		{
615 			eErrorCode = InsertJoinConnection(_pView,pNode->getChild(1), _eJoinType,pLeftTable,pRightTable);
616 		}
617 		else if (SQL_ISRULEOR2(pNode,search_condition,boolean_term)	&&			// AND/OR-Verknuepfung:
618 				 pNode->count() == 3)
619 		{
620 			// nur AND Verkn�pfung zulassen
621 			if (!SQL_ISTOKEN(pNode->getChild(1),AND))
622 				eErrorCode = eIllegalJoinCondition;
623 			else if ( eOk == (eErrorCode = InsertJoinConnection(_pView,pNode->getChild(0), _eJoinType,pLeftTable,pRightTable)) )
624 					eErrorCode = InsertJoinConnection(_pView,pNode->getChild(2), _eJoinType,pLeftTable,pRightTable);
625 		}
626 		else if (SQL_ISRULE(pNode,comparison_predicate))
627 		{
628 			// only the comparison of columns is allowed
629 			DBG_ASSERT(pNode->count() == 3,"OQueryDesignView::InsertJoinConnection: Fehler im Parse Tree");
630 			if (!(SQL_ISRULE(pNode->getChild(0),column_ref) &&
631 				  SQL_ISRULE(pNode->getChild(2),column_ref) &&
632 				   pNode->getChild(1)->getNodeType() == SQL_NODE_EQUAL))
633 			{
634 				String sError(ModuleRes(STR_QRY_JOIN_COLUMN_COMPARE));
635 				_pView->getController().appendError( sError );
636 				return eIllegalJoin;
637 			}
638 
639 			OTableFieldDescRef aDragLeft  = new OTableFieldDesc();
640 			OTableFieldDescRef aDragRight = new OTableFieldDesc();
641 			if ( eOk != ( eErrorCode = FillDragInfo(_pView,pNode->getChild(0),aDragLeft)) ||
642 				eOk != ( eErrorCode = FillDragInfo(_pView,pNode->getChild(2),aDragRight)))
643 				return eErrorCode;
644 
645             if ( pLeftTable )
646             {
647                 OQueryTableWindow*	pLeftWindow = static_cast<OQueryTableView*>(_pView->getTableView())->FindTable( getTableRange(_pView,pLeftTable->getByRule(OSQLParseNode::table_ref) ));
648                 // OQueryTableWindow*	pRightWindow = static_cast<OQueryTableView*>(_pView->getTableView())->FindTable( getTableRange(_pView,pRightTable->getByRule(OSQLParseNode::table_ref) ));
649                 if ( pLeftWindow == aDragLeft->GetTabWindow() )
650                     insertConnection(_pView,_eJoinType,aDragLeft,aDragRight);
651                 else
652                     insertConnection(_pView,_eJoinType,aDragRight,aDragLeft);
653             }
654             else
655                 insertConnection(_pView,_eJoinType,aDragLeft,aDragRight);
656 		}
657 		else
658 			eErrorCode = eIllegalJoin;
659 		return eErrorCode;
660 	}
661 	//------------------------------------------------------------------------------
662 	sal_Bool GetInnerJoinCriteria(	const OQueryDesignView* _pView,
663 									const ::connectivity::OSQLParseNode *pCondition)
664 	{
665 		return InsertJoinConnection(_pView,pCondition, INNER_JOIN,NULL,NULL) != eOk;
666 	}
667 	//------------------------------------------------------------------------------
668 	::rtl::OUString GenerateSelectList(	const OQueryDesignView* _pView,
669 										OTableFields&	_rFieldList,
670 										sal_Bool bAlias)
671 	{
672         Reference< XConnection> xConnection = static_cast<OQueryController&>(_pView->getController()).getConnection();
673 		if ( !xConnection.is() )
674 			return ::rtl::OUString();
675 
676 		::rtl::OUStringBuffer aTmpStr,aFieldListStr;
677 
678 		sal_Bool bAsterix = sal_False;
679 		int nVis = 0;
680 		OTableFields::iterator aIter = _rFieldList.begin();
681         OTableFields::iterator aEnd = _rFieldList.end();
682 		for(;aIter != aEnd;++aIter)
683 		{
684 			OTableFieldDescRef pEntryField = *aIter;
685 			if ( pEntryField->IsVisible() )
686 			{
687 				if ( pEntryField->GetField().toChar() == '*' )
688 					bAsterix = sal_True;
689 				++nVis;
690 			}
691 		}
692 		if(nVis == 1)
693 			bAsterix = sal_False;
694 
695 		try
696 		{
697 			const Reference< XDatabaseMetaData >  xMetaData = xConnection->getMetaData();
698 			const ::rtl::OUString aQuote = xMetaData->getIdentifierQuoteString();
699 
700 			OJoinTableView::OTableWindowMap* pTabList = _pView->getTableView()->GetTabWinMap();
701 
702 			const static ::rtl::OUString sFieldSeparator(RTL_CONSTASCII_USTRINGPARAM(", "));
703             const static ::rtl::OUString s_sAs(RTL_CONSTASCII_USTRINGPARAM(" AS "));
704 
705 			aIter = _rFieldList.begin();
706 		    for(;aIter != aEnd;++aIter)
707 			{
708 				OTableFieldDescRef pEntryField = *aIter;
709 				::rtl::OUString rFieldName = pEntryField->GetField();
710 				if ( rFieldName.getLength() && pEntryField->IsVisible() )
711 				{
712 					aTmpStr = ::rtl::OUString();
713 					const ::rtl::OUString rAlias = pEntryField->GetAlias();
714 					const ::rtl::OUString rFieldAlias = pEntryField->GetFieldAlias();
715 
716 					aTmpStr.append(quoteTableAlias((bAlias || bAsterix),rAlias,aQuote));
717 
718 					// if we have a none numeric field, the table alias could be in the name
719 					// otherwise we are not allowed to do this (e.g. 0.1 * PRICE )
720 					if  ( !pEntryField->isOtherFunction() )
721 					{
722 						// we have to look if we have alias.* here but before we have to check if the column doesn't already exist
723 						String sTemp = rFieldName;
724 						OTableFieldDescRef	aInfo = new OTableFieldDesc();
725 						OJoinTableView::OTableWindowMap::iterator tableIter = pTabList->begin();
726                         OJoinTableView::OTableWindowMap::iterator tableEnd = pTabList->end();
727 						sal_Bool bFound = sal_False;
728 						for(;!bFound && tableIter != tableEnd ;++tableIter)
729 						{
730 							OQueryTableWindow* pTabWin = static_cast<OQueryTableWindow*>(tableIter->second);
731 
732                             bFound = pTabWin->ExistsField( rFieldName, aInfo );
733 							if ( bFound )
734 								rFieldName = aInfo->GetField();
735 						}
736 						if ( ( rFieldName.toChar() != '*' ) && ( rFieldName.indexOf( aQuote ) == -1 ) )
737 						{
738 							OSL_ENSURE(pEntryField->GetTable().getLength(),"No table field name!");
739 							aTmpStr.append(::dbtools::quoteName(aQuote, rFieldName));
740 						}
741 						else
742 							aTmpStr.append(rFieldName);
743 					}
744 					else
745 						aTmpStr.append(rFieldName);
746 
747 					if  ( pEntryField->isAggreateFunction() )
748 					{
749 						DBG_ASSERT(pEntryField->GetFunction().getLength(),"Functionname darf hier nicht leer sein! ;-(");
750 						::rtl::OUStringBuffer aTmpStr2( pEntryField->GetFunction());
751 						aTmpStr2.appendAscii("(");
752 						aTmpStr2.append(aTmpStr.makeStringAndClear());
753 						aTmpStr2.appendAscii(")");
754 						aTmpStr = aTmpStr2;
755 					}
756 
757 					if (rFieldAlias.getLength()							&&
758 						(rFieldName.toChar() != '*'						||
759 						pEntryField->isNumericOrAggreateFunction()		||
760 						pEntryField->isOtherFunction()))
761 					{
762 						aTmpStr.append(s_sAs);
763 						aTmpStr.append(::dbtools::quoteName(aQuote, rFieldAlias));
764 					}
765 					aFieldListStr.append(aTmpStr.makeStringAndClear());
766 					aFieldListStr.append(sFieldSeparator);
767 				}
768 			}
769 			if(aFieldListStr.getLength())
770 				aFieldListStr.setLength(aFieldListStr.getLength()-2);
771 		}
772 		catch(SQLException&)
773 		{
774 			OSL_ASSERT(!"Failure while building select list!");
775 		}
776 		return aFieldListStr.makeStringAndClear();
777 	}
778 	//------------------------------------------------------------------------------
779 	sal_Bool GenerateCriterias(	OQueryDesignView* _pView,
780 								::rtl::OUStringBuffer& rRetStr,
781 								::rtl::OUStringBuffer& rHavingStr,
782 								OTableFields& _rFieldList,
783 								sal_Bool bMulti )
784 	{
785 		// * darf keine Filter enthalten : habe ich die entsprechende Warnung schon angezeigt ?
786 		sal_Bool bCritsOnAsterikWarning = sal_False;		// ** TMFS **
787 
788 		::rtl::OUString aFieldName,aCriteria,aWhereStr,aHavingStr,aWork/*,aOrderStr*/;
789 		// Zeilenweise werden die Ausdr"ucke mit AND verknuepft
790 		sal_uInt16 nMaxCriteria = 0;
791 		OTableFields::iterator aIter = _rFieldList.begin();
792 		OTableFields::iterator aEnd = _rFieldList.end();
793 		for(;aIter != aEnd;++aIter)
794 		{
795 			nMaxCriteria = ::std::max<sal_uInt16>(nMaxCriteria,(sal_uInt16)(*aIter)->GetCriteria().size());
796 		}
797 		Reference< XConnection> xConnection = static_cast<OQueryController&>(_pView->getController()).getConnection();
798 		if(!xConnection.is())
799 			return sal_False;
800 		try
801 		{
802 			const Reference< XDatabaseMetaData >  xMetaData = xConnection->getMetaData();
803 			const ::rtl::OUString aQuote = xMetaData->getIdentifierQuoteString();
804 			const IParseContext& rContext = static_cast<OQueryController&>(_pView->getController()).getParser().getContext();
805 
806 			for (sal_uInt16 i=0 ; i < nMaxCriteria ; i++)
807 			{
808 				aHavingStr = aWhereStr = ::rtl::OUString();
809 
810 				for(aIter = _rFieldList.begin();aIter != aEnd;++aIter)
811 				{
812 					OTableFieldDescRef	pEntryField = *aIter;
813 					aFieldName = pEntryField->GetField();
814 
815 					if (!aFieldName.getLength())
816 						continue;
817 					aCriteria = pEntryField->GetCriteria( i );
818 					if ( aCriteria.getLength() )
819 					{
820 						// * is not allowed to contain any filter, only when used in combination an aggregate function
821 						if ( aFieldName.toChar() == '*' && pEntryField->isNoneFunction() )
822 						{
823 							// only show the messagebox the first time
824 							if (!bCritsOnAsterikWarning)
825 								ErrorBox(_pView, ModuleRes( ERR_QRY_CRITERIA_ON_ASTERISK)).Execute();
826 							bCritsOnAsterikWarning = sal_True;
827 							continue;
828 						}
829 						aWork = ::rtl::OUString();
830 
831 
832 						aWork += quoteTableAlias(bMulti,pEntryField->GetAlias(),aQuote);
833 
834 						if ( (pEntryField->GetFunctionType() & (FKT_OTHER|FKT_NUMERIC)) || (aFieldName.toChar() == '*') )
835 							aWork += aFieldName;
836 						else
837 							aWork += ::dbtools::quoteName(aQuote, aFieldName);
838 
839 						if ( pEntryField->isAggreateFunction() || pEntryField->IsGroupBy() )
840 						{
841 							if (!aHavingStr.getLength())			// noch keine Kriterien
842 								aHavingStr += ::rtl::OUString('(');			// Klammern
843 							else
844 								aHavingStr += C_AND;
845 
846 							if ( pEntryField->isAggreateFunction() )
847 							{
848 								OSL_ENSURE(pEntryField->GetFunction().getLength(),"No function name for aggregate given!");
849 								aHavingStr += pEntryField->GetFunction();
850 								aHavingStr += ::rtl::OUString('(');			// Klammern
851 								aHavingStr += aWork;
852 								aHavingStr += ::rtl::OUString(')');			// Klammern
853 							}
854 							else
855 								aHavingStr += aWork;
856 
857 							::rtl::OUString aTmp = aCriteria;
858 							::rtl::OUString aErrorMsg;
859 							Reference<XPropertySet> xColumn;
860                             ::std::auto_ptr< ::connectivity::OSQLParseNode> pParseNode(_pView->getPredicateTreeFromEntry(pEntryField,aTmp,aErrorMsg,xColumn));
861 							if (pParseNode.get())
862 							{
863 								if (bMulti && !(pEntryField->isOtherFunction() || (aFieldName.toChar() == '*')))
864 									pParseNode->replaceNodeValue(pEntryField->GetAlias(),aFieldName);
865 								::rtl::OUString sHavingStr = aHavingStr;
866 
867 								sal_uInt32 nCount = pParseNode->count();
868 								for( sal_uInt32 node = 1 ; node < nCount ; ++node)
869 									pParseNode->getChild(node)->parseNodeToStr(	sHavingStr,
870 																xConnection,
871 																&rContext,
872 																sal_False,
873 																!pEntryField->isOtherFunction());
874 								aHavingStr = sHavingStr;
875 							}
876 							else
877 								aHavingStr += aCriteria;
878 						}
879 						else
880 						{
881 							if ( !aWhereStr.getLength() )			// noch keine Kriterien
882 								aWhereStr += ::rtl::OUString('(');			// Klammern
883 							else
884 								aWhereStr += C_AND;
885 
886 							aWhereStr += ::rtl::OUString(' ');
887 							// aCriteria could have some german numbers so I have to be sure here
888 							::rtl::OUString aTmp = aCriteria;
889 							::rtl::OUString aErrorMsg;
890 							Reference<XPropertySet> xColumn;
891 							::std::auto_ptr< ::connectivity::OSQLParseNode> pParseNode( _pView->getPredicateTreeFromEntry(pEntryField,aTmp,aErrorMsg,xColumn));
892 							if (pParseNode.get())
893 							{
894 								if (bMulti && !(pEntryField->isOtherFunction() || (aFieldName.toChar() == '*')))
895 									pParseNode->replaceNodeValue(pEntryField->GetAlias(),aFieldName);
896 								::rtl::OUString aWhere = aWhereStr;
897 								pParseNode->parseNodeToStr(	aWhere,
898 															xConnection,
899 															&rContext,
900 															sal_False,
901 															!pEntryField->isOtherFunction() );
902 								aWhereStr = aWhere;
903 							}
904 							else
905 							{
906 								aWhereStr += aWork;
907 								aWhereStr += ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("="));
908 								aWhereStr += aCriteria;
909 							}
910 						}
911 					}
912 					// nur einmal f�r jedes Feld
913 					else if ( !i && pEntryField->isCondition() )
914 					{
915 						if (!aWhereStr.getLength())			// noch keine Kriterien
916 							aWhereStr += ::rtl::OUString('(');			// Klammern
917 						else
918 							aWhereStr += C_AND;
919 						aWhereStr += pEntryField->GetField();
920 					}
921 				}
922 				if (aWhereStr.getLength())
923 				{
924 					aWhereStr += ::rtl::OUString(')');						// Klammern zu fuer 'AND' Zweig
925 					if (rRetStr.getLength())							// schon Feldbedingungen ?
926 						rRetStr.append(C_OR);
927 					else										// Klammern auf fuer 'OR' Zweig
928 						rRetStr.append(sal_Unicode('('));
929 					rRetStr.append(aWhereStr);
930 				}
931 				if (aHavingStr.getLength())
932 				{
933 					aHavingStr += ::rtl::OUString(')');						// Klammern zu fuer 'AND' Zweig
934 					if (rHavingStr.getLength())							// schon Feldbedingungen ?
935 						rHavingStr.append(C_OR);
936 					else										// Klammern auf fuer 'OR' Zweig
937 						rHavingStr.append(sal_Unicode('('));
938 					rHavingStr.append(aHavingStr);
939 				}
940 			}
941 
942 			if (rRetStr.getLength())
943 				rRetStr.append(sal_Unicode(')'));								// Klammern zu fuer 'OR' Zweig
944 			if (rHavingStr.getLength())
945 				rHavingStr.append(sal_Unicode(')'));								// Klammern zu fuer 'OR' Zweig
946 		}
947 		catch(SQLException&)
948 		{
949 			OSL_ASSERT(!"Failure while building where clause!");
950 		}
951 		return sal_True;
952 	}
953 	//------------------------------------------------------------------------------
954 	SqlParseError GenerateOrder(	OQueryDesignView* _pView,
955 									OTableFields& _rFieldList,
956 									sal_Bool bMulti,
957 									::rtl::OUString& _rsRet)
958 	{
959         const OQueryController& rController = static_cast<OQueryController&>(_pView->getController());
960 		Reference< XConnection> xConnection = rController.getConnection();
961 		if ( !xConnection.is() )
962 			return eNoConnection;
963 
964 		SqlParseError eErrorCode = eOk;
965 
966 		::rtl::OUString aColumnName;
967 		::rtl::OUString aWorkStr;
968 		try
969 		{
970             const bool bColumnAliasInOrderBy = rController.getSdbMetaData().supportsColumnAliasInOrderBy();
971 			Reference< XDatabaseMetaData >  xMetaData = xConnection->getMetaData();
972 			::rtl::OUString aQuote = xMetaData->getIdentifierQuoteString();
973 			// * darf keine Filter enthalten : habe ich die entsprechende Warnung schon angezeigt ?
974 			sal_Bool bCritsOnAsterikWarning = sal_False;		// ** TMFS **
975 			OTableFields::iterator aIter = _rFieldList.begin();
976 			OTableFields::iterator aEnd = _rFieldList.end();
977 		    for(;aIter != aEnd;++aIter)
978 			{
979 				OTableFieldDescRef	pEntryField = *aIter;
980 				EOrderDir eOrder = pEntryField->GetOrderDir();
981 
982 				// nur wenn eine Sortierung und ein Tabellenname vorhanden ist-> erzeugen
983 				// sonst werden die Expressions vom Order By im GenerateCriteria mit erzeugt
984 				if ( eOrder != ORDER_NONE )
985 				{
986 					aColumnName = pEntryField->GetField();
987 					if(aColumnName.toChar() == '*')
988 					{
989 						// die entsprechende MessageBox nur beim ersten mal anzeigen
990 						if (!bCritsOnAsterikWarning)
991 							ErrorBox(_pView, ModuleRes( ERR_QRY_ORDERBY_ON_ASTERISK)).Execute();
992 						bCritsOnAsterikWarning = sal_True;
993 						continue;
994 					}
995 
996 					if ( bColumnAliasInOrderBy && pEntryField->GetFieldAlias().getLength() )
997 					{
998 						aWorkStr += ::dbtools::quoteName(aQuote, pEntryField->GetFieldAlias());
999 					}
1000 					else if ( pEntryField->isNumericOrAggreateFunction() )
1001 					{
1002 						DBG_ASSERT(pEntryField->GetFunction().getLength(),"Functionname darf hier nicht leer sein! ;-(");
1003 						aWorkStr += pEntryField->GetFunction();
1004 						aWorkStr +=  ::rtl::OUString('(');
1005 						aWorkStr += quoteTableAlias(bMulti,pEntryField->GetAlias(),aQuote);
1006 						// only quote column name when we don't have a numeric
1007 						if ( pEntryField->isNumeric() )
1008 							aWorkStr += aColumnName;
1009 						else
1010 							aWorkStr += ::dbtools::quoteName(aQuote, aColumnName);
1011 
1012 						aWorkStr +=  ::rtl::OUString(')');
1013 					}
1014                     else if ( pEntryField->isOtherFunction() )
1015                     {
1016                         aWorkStr += aColumnName;
1017                     }
1018 					else
1019 					{
1020 						aWorkStr += quoteTableAlias(bMulti,pEntryField->GetAlias(),aQuote);
1021 						aWorkStr += ::dbtools::quoteName(aQuote, aColumnName);
1022 					}
1023 					aWorkStr += ::rtl::OUString(' ');
1024 					aWorkStr += String::CreateFromAscii( ";ASC;DESC" ).GetToken( (sal_uInt16)eOrder );
1025 					aWorkStr += ::rtl::OUString(',');
1026 				}
1027 			}
1028 
1029 			{
1030 				String sTemp(aWorkStr);
1031 				sTemp.EraseTrailingChars( ',' );
1032 				aWorkStr = sTemp;
1033 			}
1034 
1035 			if ( aWorkStr.getLength() )
1036 			{
1037 				const sal_Int32 nMaxOrder = xMetaData->getMaxColumnsInOrderBy();
1038 				String sToken(aWorkStr);
1039 				if ( nMaxOrder && nMaxOrder < sToken.GetTokenCount(',') )
1040 					eErrorCode = eStatementTooLong;
1041 				else
1042 				{
1043 					_rsRet = ::rtl::OUString::createFromAscii(" ORDER BY ");
1044 					_rsRet += aWorkStr;
1045 				}
1046 			}
1047 		}
1048 		catch(SQLException&)
1049 		{
1050 			OSL_ASSERT(!"Failure while building group by!");
1051 		}
1052 
1053 		return eErrorCode;
1054 	}
1055 
1056 	//------------------------------------------------------------------------------
1057 	void GenerateInnerJoinCriterias(const Reference< XConnection>& _xConnection,
1058 									::rtl::OUString& _rJoinCrit,
1059 									const ::std::vector<OTableConnection*>* _pConnList)
1060 	{
1061 		::std::vector<OTableConnection*>::const_iterator aIter = _pConnList->begin();
1062         ::std::vector<OTableConnection*>::const_iterator aEnd = _pConnList->end();
1063 		for(;aIter != aEnd;++aIter)
1064 		{
1065 			const OQueryTableConnection* pEntryConn = static_cast<const OQueryTableConnection*>(*aIter);
1066 			OQueryTableConnectionData* pEntryConnData = static_cast<OQueryTableConnectionData*>(pEntryConn->GetData().get());
1067             if ( pEntryConnData->GetJoinType() == INNER_JOIN && !pEntryConnData->isNatural() )
1068 			{
1069 				if(_rJoinCrit.getLength())
1070 					_rJoinCrit += C_AND;
1071 				_rJoinCrit += BuildJoinCriteria(_xConnection,pEntryConnData->GetConnLineDataList(),pEntryConnData);
1072 			}
1073 		}
1074 	}
1075 	//------------------------------------------------------------------------------
1076 	void searchAndAppendName(const Reference< XConnection>& _xConnection,
1077 							 const OQueryTableWindow* _pTableWindow,
1078 							 ::std::map< ::rtl::OUString,sal_Bool,::comphelper::UStringMixLess>& _rTableNames,
1079 							 ::rtl::OUString& _rsTableListStr
1080 							 )
1081 	{
1082 		::rtl::OUString sTabName(BuildTable(_xConnection,_pTableWindow));
1083 
1084 		if(_rTableNames.find(sTabName) == _rTableNames.end())
1085 		{
1086 			_rTableNames[sTabName] = sal_True;
1087 			_rsTableListStr += sTabName;
1088 			_rsTableListStr += ::rtl::OUString(',');
1089 		}
1090 	}
1091 	//------------------------------------------------------------------------------
1092 	::rtl::OUString GenerateFromClause(	const Reference< XConnection>& _xConnection,
1093 										const OQueryTableView::OTableWindowMap*	pTabList,
1094 										const ::std::vector<OTableConnection*>*	pConnList
1095 										)
1096 	{
1097 
1098 		::rtl::OUString aTableListStr;
1099 		// wird gebraucht um sicher zustelllen das eine Tabelle nicht doppelt vorkommt
1100 		::std::map< ::rtl::OUString,sal_Bool,::comphelper::UStringMixLess> aTableNames;
1101 
1102 		// generate outer join clause in from
1103 		if(!pConnList->empty())
1104 		{
1105 			::std::vector<OTableConnection*>::const_iterator aIter = pConnList->begin();
1106             ::std::vector<OTableConnection*>::const_iterator aEnd = pConnList->end();
1107             ::std::map<OTableWindow*,sal_Int32> aConnectionCount;
1108 			for(;aIter != aEnd;++aIter)
1109             {
1110 				static_cast<OQueryTableConnection*>(*aIter)->SetVisited(sal_False);
1111                 if ( aConnectionCount.find((*aIter)->GetSourceWin()) == aConnectionCount.end() )
1112                     aConnectionCount.insert(::std::map<OTableWindow*,sal_Int32>::value_type((*aIter)->GetSourceWin(),0));
1113                 else
1114                     aConnectionCount[(*aIter)->GetSourceWin()]++;
1115                 if ( aConnectionCount.find((*aIter)->GetDestWin()) == aConnectionCount.end() )
1116                     aConnectionCount.insert(::std::map<OTableWindow*,sal_Int32>::value_type((*aIter)->GetDestWin(),0));
1117                 else
1118                     aConnectionCount[(*aIter)->GetDestWin()]++;
1119             }
1120             ::std::multimap<sal_Int32 , OTableWindow*> aMulti;
1121             ::std::map<OTableWindow*,sal_Int32>::iterator aCountIter = aConnectionCount.begin();
1122             ::std::map<OTableWindow*,sal_Int32>::iterator aCountEnd = aConnectionCount.end();
1123             for(;aCountIter != aCountEnd;++aCountIter)
1124             {
1125                 aMulti.insert(::std::multimap<sal_Int32 , OTableWindow*>::value_type(aCountIter->second,aCountIter->first));
1126             }
1127 
1128             const sal_Bool bUseEscape = ::dbtools::getBooleanDataSourceSetting( _xConnection, PROPERTY_OUTERJOINESCAPE );
1129             ::std::multimap<sal_Int32 , OTableWindow*>::reverse_iterator aRIter = aMulti.rbegin();
1130             ::std::multimap<sal_Int32 , OTableWindow*>::reverse_iterator aREnd = aMulti.rend();
1131             for(;aRIter != aREnd;++aRIter)
1132             {
1133                 ::std::vector<OTableConnection*>::const_iterator aConIter = aRIter->second->getTableView()->getTableConnections(aRIter->second);
1134                 for(;aConIter != aEnd;++aConIter)
1135                 {
1136                     OQueryTableConnection* pEntryConn = static_cast<OQueryTableConnection*>(*aConIter);
1137 				    if(!pEntryConn->IsVisited() && pEntryConn->GetSourceWin() == aRIter->second )
1138 				    {
1139 					    ::rtl::OUString aJoin;
1140 					    GetNextJoin(_xConnection,pEntryConn,static_cast<OQueryTableWindow*>(pEntryConn->GetDestWin()),aJoin);
1141 
1142 					    if(aJoin.getLength())
1143 					    {
1144 						    // insert tables into table list to avoid double entries
1145 						    OQueryTableWindow* pEntryTabFrom = static_cast<OQueryTableWindow*>(pEntryConn->GetSourceWin());
1146 						    OQueryTableWindow* pEntryTabTo = static_cast<OQueryTableWindow*>(pEntryConn->GetDestWin());
1147 
1148 						    ::rtl::OUString sTabName(BuildTable(_xConnection,pEntryTabFrom));
1149 						    if(aTableNames.find(sTabName) == aTableNames.end())
1150 							    aTableNames[sTabName] = sal_True;
1151 						    sTabName = BuildTable(_xConnection,pEntryTabTo);
1152 						    if(aTableNames.find(sTabName) == aTableNames.end())
1153 							    aTableNames[sTabName] = sal_True;
1154 
1155                             ::rtl::OUString aStr;
1156                             switch(static_cast<OQueryTableConnectionData*>(pEntryConn->GetData().get())->GetJoinType())
1157 		                    {
1158 			                    case LEFT_JOIN:
1159 			                    case RIGHT_JOIN:
1160                                 case FULL_JOIN:
1161                                     {
1162 						                // create outer join
1163 						                if ( bUseEscape )
1164 							                aStr += ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("{ OJ "));
1165 						                aStr += aJoin;
1166 						                if ( bUseEscape )
1167 							                aStr += ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" }"));
1168                                     }
1169                                     break;
1170                                 default:
1171                                     aStr += aJoin;
1172                                     break;
1173                             }
1174 						    aStr += ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(","));
1175 						    aTableListStr += aStr;
1176 					    }
1177 				    }
1178 			    }
1179             }
1180 
1181 			// and now all inner joins
1182 			aIter = pConnList->begin();
1183 			for(;aIter != aEnd;++aIter)
1184 			{
1185 				OQueryTableConnection* pEntryConn = static_cast<OQueryTableConnection*>(*aIter);
1186 				if(!pEntryConn->IsVisited())
1187 				{
1188 					searchAndAppendName(_xConnection,
1189 										static_cast<OQueryTableWindow*>(pEntryConn->GetSourceWin()),
1190 										aTableNames,
1191 										aTableListStr);
1192 
1193 					searchAndAppendName(_xConnection,
1194 										static_cast<OQueryTableWindow*>(pEntryConn->GetDestWin()),
1195 										aTableNames,
1196 										aTableListStr);
1197 				}
1198 			}
1199 		}
1200 		// all tables that haven't a connection to anyone
1201 		OQueryTableView::OTableWindowMap::const_iterator aTabIter = pTabList->begin();
1202         OQueryTableView::OTableWindowMap::const_iterator aTabEnd = pTabList->end();
1203 		for(;aTabIter != aTabEnd;++aTabIter)
1204 		{
1205 			const OQueryTableWindow* pEntryTab = static_cast<const OQueryTableWindow*>(aTabIter->second);
1206 			if(!pEntryTab->ExistsAConn())
1207 			{
1208 				aTableListStr += BuildTable(_xConnection,pEntryTab);
1209 				aTableListStr += ::rtl::OUString(',');
1210 			}
1211 		}
1212 
1213 		if(aTableListStr.getLength())
1214 			aTableListStr = aTableListStr.replaceAt(aTableListStr.getLength()-1,1, ::rtl::OUString() );
1215 		return aTableListStr;
1216 	}
1217 	//------------------------------------------------------------------------------
1218 	::rtl::OUString GenerateGroupBy(const OQueryDesignView* _pView,OTableFields& _rFieldList, sal_Bool bMulti )
1219 	{
1220         OQueryController& rController = static_cast<OQueryController&>(_pView->getController());
1221 		const Reference< XConnection> xConnection = rController.getConnection();
1222 		if(!xConnection.is())
1223 			return ::rtl::OUString();
1224 
1225         ::std::map< rtl::OUString,bool> aGroupByNames;
1226 
1227 		::rtl::OUString aGroupByStr;
1228 		try
1229 		{
1230 			const Reference< XDatabaseMetaData >  xMetaData = xConnection->getMetaData();
1231 			const ::rtl::OUString aQuote = xMetaData->getIdentifierQuoteString();
1232 
1233 			OTableFields::iterator aIter = _rFieldList.begin();
1234 			OTableFields::iterator aEnd = _rFieldList.end();
1235 		    for(;aIter != aEnd;++aIter)
1236 			{
1237 				OTableFieldDescRef	pEntryField = *aIter;
1238 				if ( pEntryField->IsGroupBy() )
1239 				{
1240 					DBG_ASSERT(pEntryField->GetField().getLength(),"Kein FieldName vorhanden!;-(");
1241 					::rtl::OUString sGroupByPart = quoteTableAlias(bMulti,pEntryField->GetAlias(),aQuote);
1242 
1243 					// only quote the field name when it isn't calculated
1244 					if ( pEntryField->isNoneFunction() )
1245                     {
1246 						sGroupByPart += ::dbtools::quoteName(aQuote, pEntryField->GetField());
1247                     }
1248 					else
1249                     {
1250                         ::rtl::OUString aTmp = pEntryField->GetField();
1251 						::rtl::OUString aErrorMsg;
1252 						Reference<XPropertySet> xColumn;
1253                         ::std::auto_ptr< ::connectivity::OSQLParseNode> pParseNode(_pView->getPredicateTreeFromEntry(pEntryField,aTmp,aErrorMsg,xColumn));
1254 						if (pParseNode.get())
1255 						{
1256 							::rtl::OUString sGroupBy;
1257 							pParseNode->getChild(0)->parseNodeToStr(	sGroupBy,
1258 														xConnection,
1259 														&rController.getParser().getContext(),
1260 														sal_False,
1261 														!pEntryField->isOtherFunction());
1262 							sGroupByPart += sGroupBy;
1263 						}
1264 						else
1265 							sGroupByPart += pEntryField->GetField();
1266                     }
1267                     if ( aGroupByNames.find(sGroupByPart) == aGroupByNames.end() )
1268                     {
1269                         aGroupByNames.insert(::std::map< rtl::OUString,bool>::value_type(sGroupByPart,true));
1270                         aGroupByStr += sGroupByPart;
1271 					    aGroupByStr += ::rtl::OUString(',');
1272                     }
1273 				}
1274 			}
1275 			if ( aGroupByStr.getLength() )
1276 			{
1277 				aGroupByStr = aGroupByStr.replaceAt(aGroupByStr.getLength()-1,1, ::rtl::OUString(' ') );
1278 				::rtl::OUString aGroupByStr2 = ::rtl::OUString::createFromAscii(" GROUP BY ");
1279 				aGroupByStr2 += aGroupByStr;
1280 				aGroupByStr = aGroupByStr2;
1281 			}
1282 		}
1283 		catch(SQLException&)
1284 		{
1285 			OSL_ASSERT(!"Failure while building group by!");
1286 		}
1287 		return aGroupByStr;
1288 	}
1289 	// -----------------------------------------------------------------------------
1290 	SqlParseError GetORCriteria(OQueryDesignView* _pView,
1291 								OSelectionBrowseBox* _pSelectionBrw,
1292 								const ::connectivity::OSQLParseNode * pCondition,
1293 								sal_uInt16& nLevel ,
1294 								sal_Bool bHaving = sal_False,
1295                                 bool bAddOrOnOneLine = false);
1296 	// -----------------------------------------------------------------------------
1297 	SqlParseError GetSelectionCriteria(	OQueryDesignView* _pView,
1298 										OSelectionBrowseBox* _pSelectionBrw,
1299 										const ::connectivity::OSQLParseNode* pNode,
1300 										sal_uInt16& rLevel )
1301 	{
1302 		if (!SQL_ISRULE(pNode, select_statement))
1303 			return eNoSelectStatement;
1304 
1305 		// nyi: mehr Pruefung auf korrekte Struktur!
1306 		pNode = pNode ? pNode->getChild(3)->getChild(1) : NULL;
1307 		// no where clause found
1308 		if (!pNode || pNode->isLeaf())
1309 			return eOk;
1310 
1311 		// Naechster freier Satz ...
1312 		SqlParseError eErrorCode = eOk;
1313 		::connectivity::OSQLParseNode * pCondition = pNode->getChild(1);
1314 		if ( pCondition ) // no where clause
1315 		{
1316 			// now we have to chech the other conditions
1317 			// first make the logical easier
1318 			::connectivity::OSQLParseNode::negateSearchCondition(pCondition);
1319 			::connectivity::OSQLParseNode *pNodeTmp = pNode->getChild(1);
1320 
1321 			::connectivity::OSQLParseNode::disjunctiveNormalForm(pNodeTmp);
1322 			pNodeTmp = pNode->getChild(1);
1323 			::connectivity::OSQLParseNode::absorptions(pNodeTmp);
1324 			pNodeTmp = pNode->getChild(1);
1325             // compress sort the criteria @see http://www.openoffice.org/issues/show_bug.cgi?id=24079
1326             OSQLParseNode::compress(pNodeTmp);
1327             pNodeTmp = pNode->getChild(1);
1328 
1329 			// first extract the inner joins conditions
1330 			GetInnerJoinCriteria(_pView,pNodeTmp);
1331 			// now simplify again, join are checked in ComparisonPredicate
1332 			::connectivity::OSQLParseNode::absorptions(pNodeTmp);
1333 			pNodeTmp = pNode->getChild(1);
1334 
1335 			// it could happen that pCondition is not more valid
1336 			eErrorCode = GetORCriteria(_pView,_pSelectionBrw,pNodeTmp, rLevel);
1337 		}
1338 		return eErrorCode;
1339 	}
1340 	//------------------------------------------------------------------------------
1341 	SqlParseError GetANDCriteria(	OQueryDesignView* _pView,
1342 									OSelectionBrowseBox* _pSelectionBrw,
1343 									const  ::connectivity::OSQLParseNode * pCondition,
1344 									sal_uInt16& nLevel,
1345 									sal_Bool bHaving,
1346                                     bool bAddOrOnOneLine);
1347 	//------------------------------------------------------------------------------
1348 	SqlParseError ComparisonPredicate(OQueryDesignView* _pView,
1349 							OSelectionBrowseBox* _pSelectionBrw,
1350 							const ::connectivity::OSQLParseNode * pCondition,
1351 							const sal_uInt16 nLevel,
1352 							sal_Bool bHaving,
1353                             bool bAddOrOnOneLine);
1354 	//------------------------------------------------------------------------------
1355 	SqlParseError GetORCriteria(OQueryDesignView* _pView,
1356 								OSelectionBrowseBox* _pSelectionBrw,
1357 								const ::connectivity::OSQLParseNode * pCondition,
1358 								sal_uInt16& nLevel ,
1359 								sal_Bool bHaving,
1360                                 bool bAddOrOnOneLine)
1361 	{
1362 		SqlParseError eErrorCode = eOk;
1363 
1364 		// Runde Klammern um den Ausdruck
1365 		if (pCondition->count() == 3 &&
1366 			SQL_ISPUNCTUATION(pCondition->getChild(0),"(") &&
1367 			SQL_ISPUNCTUATION(pCondition->getChild(2),")"))
1368 		{
1369 			eErrorCode = GetORCriteria(_pView,_pSelectionBrw,pCondition->getChild(1),nLevel,bHaving,bAddOrOnOneLine);
1370 		}
1371 		// oder Verknuepfung
1372 		// a searchcondition can only look like this: search_condition SQL_TOKEN_OR boolean_term
1373 		else if (SQL_ISRULE(pCondition,search_condition))
1374 		{
1375 			for (int i = 0; i < 3 && eErrorCode == eOk ; i+=2)
1376 			{
1377                 const  ::connectivity::OSQLParseNode* pChild = pCondition->getChild(i);
1378 				if ( SQL_ISRULE(pChild,search_condition) )
1379 					eErrorCode = GetORCriteria(_pView,_pSelectionBrw,pChild,nLevel,bHaving,bAddOrOnOneLine);
1380 				else
1381 				{
1382                     eErrorCode = GetANDCriteria(_pView,_pSelectionBrw,pChild, nLevel,bHaving, i == 0 ? false : bAddOrOnOneLine);
1383 					if ( !bAddOrOnOneLine)
1384 						nLevel++;
1385 				}
1386 			}
1387 		}
1388 		else
1389 			eErrorCode = GetANDCriteria( _pView,_pSelectionBrw,pCondition, nLevel, bHaving,bAddOrOnOneLine );
1390 
1391 		return eErrorCode;
1392 	}
1393     //--------------------------------------------------------------------------------------------------
1394     bool CheckOrCriteria(const ::connectivity::OSQLParseNode* _pCondition,::connectivity::OSQLParseNode* _pFirstColumnRef)
1395     {
1396         bool bRet = true;
1397         ::connectivity::OSQLParseNode* pFirstColumnRef = _pFirstColumnRef;
1398         for (int i = 0; i < 3 && bRet; i+=2)
1399 		{
1400             const  ::connectivity::OSQLParseNode* pChild = _pCondition->getChild(i);
1401             if ( SQL_ISRULE(pChild,search_condition) )
1402                 bRet = CheckOrCriteria(pChild,pFirstColumnRef);
1403             else
1404             {
1405                 // this is a simple way to test columns are the same, may be we have to adjust this algo a little bit in future. :-)
1406                 ::connectivity::OSQLParseNode* pSecondColumnRef = pChild->getByRule(::connectivity::OSQLParseNode::column_ref);
1407                 if ( pFirstColumnRef && pSecondColumnRef )
1408                     bRet = *pFirstColumnRef == *pSecondColumnRef;
1409                 else if ( !pFirstColumnRef )
1410                     pFirstColumnRef = pSecondColumnRef;
1411             }
1412         }
1413         return bRet;
1414     }
1415 	//--------------------------------------------------------------------------------------------------
1416 	SqlParseError GetANDCriteria(	OQueryDesignView* _pView,
1417 									OSelectionBrowseBox* _pSelectionBrw,
1418 									const  ::connectivity::OSQLParseNode * pCondition,
1419 									sal_uInt16& nLevel,
1420 									sal_Bool bHaving,
1421                                     bool bAddOrOnOneLine)
1422 	{
1423 		const ::com::sun::star::lang::Locale	aLocale = _pView->getLocale();
1424 		const ::rtl::OUString sDecimal = _pView->getDecimalSeparator();
1425 
1426 		// ich werde ein paar Mal einen gecasteten Pointer auf meinen ::com::sun::star::sdbcx::Container brauchen
1427 		OQueryController& rController = static_cast<OQueryController&>(_pView->getController());
1428 		SqlParseError eErrorCode = eOk;
1429 
1430 		// Runde Klammern
1431 		if (SQL_ISRULE(pCondition,boolean_primary))
1432 		{
1433             // check if we have to put the or criteria on one line.
1434 			const  ::connectivity::OSQLParseNode* pSearchCondition = pCondition->getChild(1);
1435             bool bMustAddOrOnOneLine = CheckOrCriteria(pSearchCondition,NULL);
1436 			if ( SQL_ISRULE( pSearchCondition, search_condition) ) // we have a or
1437 			{
1438 				_pSelectionBrw->DuplicateConditionLevel( nLevel);
1439 				eErrorCode = GetORCriteria(_pView,_pSelectionBrw,pSearchCondition->getChild(0), nLevel,bHaving,bMustAddOrOnOneLine );
1440 				++nLevel;
1441 				eErrorCode = GetORCriteria(_pView,_pSelectionBrw,pSearchCondition->getChild(2), nLevel,bHaving,bMustAddOrOnOneLine );
1442 			}
1443 			else
1444 				eErrorCode = GetORCriteria(_pView,_pSelectionBrw,pSearchCondition, nLevel,bHaving,bMustAddOrOnOneLine );
1445 		}
1446 		// Das erste Element ist (wieder) eine AND-Verknuepfung
1447 		else if ( SQL_ISRULE(pCondition,boolean_term) )
1448 		{
1449             OSL_ENSURE(pCondition->count() == 3,"Illegal definifiton of boolean_term");
1450 			eErrorCode = GetANDCriteria(_pView,_pSelectionBrw,pCondition->getChild(0), nLevel,bHaving,bAddOrOnOneLine );
1451 			if ( eErrorCode == eOk )
1452 				eErrorCode = GetANDCriteria(_pView,_pSelectionBrw,pCondition->getChild(2), nLevel,bHaving,bAddOrOnOneLine );
1453 		}
1454 		else if (SQL_ISRULE( pCondition, comparison_predicate))
1455 		{
1456 			eErrorCode = ComparisonPredicate(_pView,_pSelectionBrw,pCondition,nLevel,bHaving,bAddOrOnOneLine);
1457 		}
1458 		else if( SQL_ISRULE(pCondition,like_predicate) )
1459 		{
1460             const  ::connectivity::OSQLParseNode* pValueExp = pCondition->getChild(0);
1461 			if (SQL_ISRULE(pValueExp, column_ref ) )
1462 			{
1463 				::rtl::OUString aColumnName;
1464 				::rtl::OUString	aCondition;
1465 				Reference< XConnection> xConnection = rController.getConnection();
1466 				if ( xConnection.is() )
1467 				{
1468 					Reference< XDatabaseMetaData >  xMetaData = xConnection->getMetaData();
1469 					// the international doesn't matter I have a string
1470 					pCondition->parseNodeToPredicateStr(aCondition,
1471 														xConnection,
1472 														rController.getNumberFormatter(),
1473 														aLocale,
1474 														static_cast<sal_Char>(sDecimal.toChar()),
1475 														&rController.getParser().getContext());
1476 
1477 					pValueExp->parseNodeToPredicateStr(	aColumnName,
1478 														xConnection,
1479 														rController.getNumberFormatter(),
1480 														aLocale,
1481 														static_cast<sal_Char>(sDecimal.toChar()),
1482 														&rController.getParser().getContext());
1483 
1484 					// don't display the column name
1485 					aCondition = aCondition.copy(aColumnName.getLength());
1486 					aCondition = aCondition.trim();
1487 				}
1488 
1489 				OTableFieldDescRef aDragLeft = new OTableFieldDesc();
1490 				if ( eOk == ( eErrorCode = FillDragInfo(_pView,pValueExp,aDragLeft) ))
1491                 {
1492                     if ( bHaving )
1493 					    aDragLeft->SetGroupBy(sal_True);
1494 					_pSelectionBrw->AddCondition(aDragLeft, aCondition, nLevel,bAddOrOnOneLine);
1495                 }
1496 			}
1497             else if(SQL_ISRULEOR2(pValueExp,general_set_fct ,set_fct_spec)	        ||
1498 							SQL_ISRULEOR2(pValueExp,position_exp,extract_exp)	||
1499 							SQL_ISRULEOR2(pValueExp,fold,char_substring_fct)	||
1500 							SQL_ISRULEOR2(pValueExp,length_exp,char_value_fct))
1501 		    {
1502 			    AddFunctionCondition(	_pView,
1503 									    _pSelectionBrw,
1504 									    pCondition,
1505 									    nLevel,
1506 									    bHaving,
1507                                         bAddOrOnOneLine);
1508 		    }
1509 			else
1510 			{
1511 				eErrorCode = eNoColumnInLike;
1512 				String sError(ModuleRes(STR_QRY_LIKE_LEFT_NO_COLUMN));
1513 				_pView->getController().appendError( sError );
1514 			}
1515 		}
1516 		else if(	SQL_ISRULEOR2(pCondition,test_for_null,in_predicate)
1517 				||	SQL_ISRULEOR2(pCondition,all_or_any_predicate,between_predicate))
1518 		{
1519 			if ( SQL_ISRULEOR2(pCondition->getChild(0), set_fct_spec , general_set_fct ) )
1520 			{
1521 				AddFunctionCondition(	_pView,
1522 										_pSelectionBrw,
1523 										pCondition,
1524 										nLevel,
1525 										bHaving,
1526                                         bAddOrOnOneLine);
1527 			}
1528 			else if ( SQL_ISRULE(pCondition->getChild(0), column_ref ) )
1529 			{
1530 				// parse condition
1531 				::rtl::OUString sCondition = ParseCondition(rController,pCondition,sDecimal,aLocale,1);
1532 				OTableFieldDescRef	aDragLeft = new OTableFieldDesc();
1533 				if ( eOk == ( eErrorCode = FillDragInfo(_pView,pCondition->getChild(0),aDragLeft)) )
1534                 {
1535                     if ( bHaving )
1536 					    aDragLeft->SetGroupBy(sal_True);
1537 					_pSelectionBrw->AddCondition(aDragLeft, sCondition, nLevel,bAddOrOnOneLine);
1538                 }
1539 			}
1540 			else
1541 			{
1542 				// Funktions-Bedingung parsen
1543 				::rtl::OUString	sCondition = ParseCondition(rController,pCondition,sDecimal,aLocale,1);
1544 				Reference< XConnection> xConnection = rController.getConnection();
1545 				Reference< XDatabaseMetaData >  xMetaData = xConnection->getMetaData();
1546 					// the international doesn't matter I have a string
1547 				::rtl::OUString sName;
1548 				pCondition->getChild(0)->parseNodeToPredicateStr(sName,
1549 													xConnection,
1550 													rController.getNumberFormatter(),
1551 													aLocale,
1552 													static_cast<sal_Char>(sDecimal.toChar()),
1553 													&rController.getParser().getContext());
1554 
1555 				OTableFieldDescRef aDragLeft = new OTableFieldDesc();
1556 				aDragLeft->SetField(sName);
1557 				aDragLeft->SetFunctionType(FKT_OTHER);
1558 
1559 				if ( bHaving )
1560 					aDragLeft->SetGroupBy(sal_True);
1561 				_pSelectionBrw->AddCondition(aDragLeft, sCondition, nLevel,bAddOrOnOneLine);
1562 			}
1563 		}
1564 		else if( SQL_ISRULEOR2(pCondition,existence_test,unique_test) )
1565 		{
1566 			// Funktions-Bedingung parsen
1567 			::rtl::OUString	aCondition = ParseCondition(rController,pCondition,sDecimal,aLocale,0);
1568 
1569 			OTableFieldDescRef aDragLeft = new OTableFieldDesc();
1570 			aDragLeft->SetField(aCondition);
1571 			aDragLeft->SetFunctionType(FKT_CONDITION);
1572 
1573 			eErrorCode = _pSelectionBrw->InsertField(aDragLeft,BROWSER_INVALIDID,sal_False,sal_True).isValid() ? eOk : eTooManyColumns;
1574 		}
1575 		else //! TODO not supported yet
1576 			eErrorCode = eStatementTooComplex;
1577 		// Fehler einfach weiterreichen.
1578 		return eErrorCode;
1579 	}
1580 	//------------------------------------------------------------------------------
1581 	SqlParseError AddFunctionCondition(OQueryDesignView* _pView,
1582 							OSelectionBrowseBox* _pSelectionBrw,
1583 							const ::connectivity::OSQLParseNode * pCondition,
1584 							const sal_uInt16 nLevel,
1585 							sal_Bool bHaving,
1586                             bool bAddOrOnOneLine)
1587 	{
1588 		SqlParseError eErrorCode = eOk;
1589 		OQueryController& rController = static_cast<OQueryController&>(_pView->getController());
1590 
1591         OSQLParseNode* pFunction = pCondition->getChild(0);
1592 
1593         OSL_ENSURE(SQL_ISRULEOR2(pFunction,general_set_fct ,set_fct_spec)	        ||
1594 							SQL_ISRULEOR2(pFunction,position_exp,extract_exp)	||
1595 							SQL_ISRULEOR2(pFunction,fold,char_substring_fct)	||
1596 							SQL_ISRULEOR2(pFunction,length_exp,char_value_fct),"Illegal call!");
1597 		::rtl::OUString	aCondition;
1598 		OTableFieldDescRef aDragLeft = new OTableFieldDesc();
1599 
1600 		::rtl::OUString aColumnName;
1601 		Reference< XConnection> xConnection = rController.getConnection();
1602 		if(xConnection.is())
1603 		{
1604 			Reference< XDatabaseMetaData >  xMetaData = xConnection->getMetaData();
1605 			pCondition->parseNodeToPredicateStr(aCondition,
1606 												xConnection,
1607 												rController.getNumberFormatter(),
1608 												_pView->getLocale(),
1609 												static_cast<sal_Char>(_pView->getDecimalSeparator().toChar()),
1610 												&rController.getParser().getContext());
1611 
1612             pFunction->parseNodeToStr(	aColumnName,
1613 										xConnection,
1614 										&rController.getParser().getContext(),
1615 										sal_True,
1616 										sal_True); // quote is to true because we need quoted elements inside the function
1617             // i75557
1618 			//pFunction->parseNodeToPredicateStr(aColumnName,
1619 			//									xConnection,
1620 			//									rController.getNumberFormatter(),
1621 			//									_pView->getLocale(),
1622 			//									static_cast<sal_Char>(_pView->getDecimalSeparator().toChar()),
1623 			//									&rController.getParser().getContext());
1624 			// don't display the column name
1625 			aCondition = aCondition.copy(aColumnName.getLength());
1626 			aCondition = aCondition.trim();
1627 			if ( aCondition.indexOf('=',0) == 0 ) // ignore the equal sign
1628 				aCondition = aCondition.copy(1);
1629 
1630 
1631 			if ( SQL_ISRULE(pFunction, general_set_fct ) )
1632 			{
1633 				sal_Int32 nFunctionType = FKT_AGGREGATE;
1634 				OSQLParseNode* pParamNode = pFunction->getChild(pFunction->count()-2);
1635 				if ( pParamNode && pParamNode->getTokenValue().toChar() == '*' )
1636 				{
1637 					OJoinTableView::OTableWindowMap* pTabList = _pView->getTableView()->GetTabWinMap();
1638 					OJoinTableView::OTableWindowMap::iterator aIter = pTabList->begin();
1639                     OJoinTableView::OTableWindowMap::iterator aTabEnd = pTabList->end();
1640 					for(;aIter != aTabEnd;++aIter)
1641 					{
1642 						OQueryTableWindow* pTabWin = static_cast<OQueryTableWindow*>(aIter->second);
1643 						if (pTabWin->ExistsField( ::rtl::OUString::createFromAscii("*"), aDragLeft ))
1644 						{
1645 							aDragLeft->SetAlias(String());
1646 							aDragLeft->SetTable(String());
1647 							break;
1648 						}
1649 					}
1650 				}
1651 				else if( eOk != ( eErrorCode = FillDragInfo(_pView,pParamNode,aDragLeft))
1652 						&& SQL_ISRULE(pParamNode,num_value_exp) )
1653 				{
1654 					::rtl::OUString sParameterValue;
1655 					pParamNode->parseNodeToStr(	sParameterValue,
1656 												xConnection,
1657 												&rController.getParser().getContext());
1658 					nFunctionType |= FKT_NUMERIC;
1659 					aDragLeft->SetField(sParameterValue);
1660 					eErrorCode = eOk;
1661 				}
1662 				aDragLeft->SetFunctionType(nFunctionType);
1663 				if ( bHaving )
1664 					aDragLeft->SetGroupBy(sal_True);
1665 				sal_Int32 nIndex = 0;
1666 				aDragLeft->SetFunction(aColumnName.getToken(0,'(',nIndex));
1667 			}
1668 			else
1669 			{
1670 				// bei unbekannten Funktionen wird der gesamte Text in das Field gechrieben
1671 				aDragLeft->SetField(aColumnName);
1672 				if(bHaving)
1673 					aDragLeft->SetGroupBy(sal_True);
1674 				aDragLeft->SetFunctionType(FKT_OTHER|FKT_NUMERIC);
1675 			}
1676 			_pSelectionBrw->AddCondition(aDragLeft, aCondition, nLevel,bAddOrOnOneLine);
1677 		}
1678 
1679 		return eErrorCode;
1680 	}
1681 	//------------------------------------------------------------------------------
1682 	SqlParseError ComparisonPredicate(OQueryDesignView* _pView,
1683 							OSelectionBrowseBox* _pSelectionBrw,
1684 							const ::connectivity::OSQLParseNode * pCondition,
1685 							const sal_uInt16 nLevel,
1686 							sal_Bool bHaving
1687                             ,bool bAddOrOnOneLine)
1688 	{
1689 		SqlParseError eErrorCode = eOk;
1690 		OQueryController& rController = static_cast<OQueryController&>(_pView->getController());
1691 
1692 		DBG_ASSERT(SQL_ISRULE( pCondition, comparison_predicate),"ComparisonPredicate: pCondition ist kein ComparisonPredicate");
1693 		if ( SQL_ISRULE(pCondition->getChild(0), column_ref )
1694 			|| SQL_ISRULE(pCondition->getChild(pCondition->count()-1), column_ref) )
1695 		{
1696 			::rtl::OUString	aCondition;
1697 			OTableFieldDescRef aDragLeft = new OTableFieldDesc();
1698 
1699 			if ( SQL_ISRULE(pCondition->getChild(0), column_ref ) && SQL_ISRULE(pCondition->getChild(pCondition->count()-1), column_ref ) )
1700 			{
1701 				OTableFieldDescRef aDragRight = new OTableFieldDesc();
1702 				if (eOk != ( eErrorCode = FillDragInfo(_pView,pCondition->getChild(0),aDragLeft)) ||
1703 					eOk != ( eErrorCode = FillDragInfo(_pView,pCondition->getChild(2),aDragRight)))
1704 					return eErrorCode;
1705 
1706                 OQueryTableConnection* pConn = static_cast<OQueryTableConnection*>(
1707                                                     _pView->getTableView()->GetTabConn(static_cast<OQueryTableWindow*>(aDragLeft->GetTabWindow()),
1708 																					   static_cast<OQueryTableWindow*>(aDragRight->GetTabWindow()),
1709                                                                                        true));
1710 				if ( pConn )
1711 				{
1712 					OConnectionLineDataVec* pLineDataList = pConn->GetData()->GetConnLineDataList();
1713 					OConnectionLineDataVec::iterator aIter = pLineDataList->begin();
1714                     OConnectionLineDataVec::iterator aEnd = pLineDataList->end();
1715 					for(;aIter != aEnd;++aIter)
1716 					{
1717 						if((*aIter)->GetSourceFieldName() == aDragLeft->GetField() ||
1718 						   (*aIter)->GetDestFieldName() == aDragLeft->GetField() )
1719 							break;
1720 					}
1721 					if(aIter != aEnd)
1722 						return eOk;
1723 				}
1724 			}
1725 
1726 			sal_uInt32 nPos = 0;
1727 			if(SQL_ISRULE(pCondition->getChild(0), column_ref ))
1728 			{
1729 				nPos = 0;
1730 				sal_uInt32 i=1;
1731 
1732 				// don't display the equal
1733 				if (pCondition->getChild(i)->getNodeType() == SQL_NODE_EQUAL)
1734 					i++;
1735 
1736 				// Bedingung parsen
1737 				aCondition = ParseCondition(rController
1738 											,pCondition
1739 											,_pView->getDecimalSeparator()
1740 											,_pView->getLocale()
1741 											,i);
1742 			}
1743 			else if( SQL_ISRULE(pCondition->getChild(pCondition->count()-1), column_ref ) )
1744 			{
1745 				nPos = pCondition->count()-1;
1746 
1747 				sal_Int32 i = static_cast<sal_Int32>(pCondition->count() - 2);
1748 				switch (pCondition->getChild(i)->getNodeType())
1749 				{
1750 					case SQL_NODE_EQUAL:
1751 						// don't display the equal
1752 						i--;
1753 						break;
1754 					case SQL_NODE_LESS:
1755 						// take the opposite as we change the order
1756 						i--;
1757 						aCondition = aCondition + ::rtl::OUString::createFromAscii(">");
1758 						break;
1759 					case SQL_NODE_LESSEQ:
1760 						// take the opposite as we change the order
1761 						i--;
1762 						aCondition = aCondition + ::rtl::OUString::createFromAscii(">=");
1763 						break;
1764 					case SQL_NODE_GREAT:
1765 						// take the opposite as we change the order
1766 						i--;
1767 						aCondition = aCondition + ::rtl::OUString::createFromAscii("<");
1768 						break;
1769 					case SQL_NODE_GREATEQ:
1770 						// take the opposite as we change the order
1771 						i--;
1772 						aCondition = aCondition + ::rtl::OUString::createFromAscii("<=");
1773 						break;
1774                     default:
1775                         break;
1776 				}
1777 
1778 				// go backward
1779 				Reference< XConnection> xConnection = rController.getConnection();
1780 				if(xConnection.is())
1781 				{
1782 					Reference< XDatabaseMetaData >  xMetaData = xConnection->getMetaData();
1783 					for (; i >= 0; i--)
1784 						pCondition->getChild(i)->parseNodeToPredicateStr(aCondition,
1785 												xConnection,
1786 												rController.getNumberFormatter(),
1787 												_pView->getLocale(),
1788 												static_cast<sal_Char>(_pView->getDecimalSeparator().toChar()),
1789 												&rController.getParser().getContext());
1790 				}
1791 			}
1792 			// else ???
1793 
1794 
1795 			if( eOk == ( eErrorCode = FillDragInfo(_pView,pCondition->getChild(nPos),aDragLeft)))
1796 			{
1797 				if(bHaving)
1798 					aDragLeft->SetGroupBy(sal_True);
1799 				_pSelectionBrw->AddCondition(aDragLeft, aCondition, nLevel,bAddOrOnOneLine);
1800 			}
1801 		}
1802 		else if( SQL_ISRULEOR2(pCondition->getChild(0), set_fct_spec , general_set_fct ) )
1803 		{
1804 			AddFunctionCondition(	_pView,
1805 									_pSelectionBrw,
1806 									pCondition,
1807 									nLevel,
1808 									bHaving,
1809                                     bAddOrOnOneLine);
1810 		}
1811 		else // kann sich nur um einen Expr. Ausdruck handeln
1812 		{
1813 			::rtl::OUString aName,aCondition;
1814 
1815 			::connectivity::OSQLParseNode *pLhs = pCondition->getChild(0);
1816 			::connectivity::OSQLParseNode *pRhs = pCondition->getChild(2);
1817 			// Feldnamen
1818 			Reference< XConnection> xConnection = rController.getConnection();
1819 			if(xConnection.is())
1820 			{
1821 				pLhs->parseNodeToStr(aName,
1822 									 xConnection,
1823 									 &rController.getParser().getContext(),
1824 									 sal_True);
1825 				// Kriterium
1826 				aCondition = pCondition->getChild(1)->getTokenValue();
1827 				pRhs->parseNodeToPredicateStr(aCondition,
1828 															xConnection,
1829 															rController.getNumberFormatter(),
1830 															_pView->getLocale(),
1831 															static_cast<sal_Char>(_pView->getDecimalSeparator().toChar()),
1832 															&rController.getParser().getContext());
1833 			}
1834 
1835 			OTableFieldDescRef aDragLeft = new OTableFieldDesc();
1836 			aDragLeft->SetField(aName);
1837 			aDragLeft->SetFunctionType(FKT_OTHER|FKT_NUMERIC);
1838 			// und anh"angen
1839 			_pSelectionBrw->AddCondition(aDragLeft, aCondition, nLevel,bAddOrOnOneLine);
1840 		}
1841 		return eErrorCode;
1842 	}
1843 
1844     //------------------------------------------------------------------------------
1845     namespace
1846     {
1847         OQueryTableWindow* lcl_findColumnInTables( const ::rtl::OUString& _rColumName, const OJoinTableView::OTableWindowMap& _rTabList, OTableFieldDescRef& _rInfo )
1848         {
1849 			OJoinTableView::OTableWindowMap::const_iterator aIter = _rTabList.begin();
1850             OJoinTableView::OTableWindowMap::const_iterator aEnd = _rTabList.end();
1851 			for ( ; aIter != aEnd; ++aIter )
1852 			{
1853 				OQueryTableWindow* pTabWin = static_cast< OQueryTableWindow* >( aIter->second );
1854 				if ( pTabWin && pTabWin->ExistsField( _rColumName, _rInfo ) )
1855 					return pTabWin;
1856 			}
1857             return NULL;
1858         }
1859     }
1860 
1861     //------------------------------------------------------------------------------
1862 	void InsertColumnRef(const OQueryDesignView* _pView,
1863 						const ::connectivity::OSQLParseNode * pColumnRef,
1864 						::rtl::OUString& aColumnName,
1865 						const ::rtl::OUString& aColumnAlias,
1866 						::rtl::OUString& aTableRange,
1867 						OTableFieldDescRef& _raInfo,
1868 						OJoinTableView::OTableWindowMap* pTabList)
1869 	{
1870 
1871 		// Tabellennamen zusammen setzen
1872 		::connectivity::OSQLParseTreeIterator& rParseIter = static_cast<OQueryController&>(_pView->getController()).getParseIterator();
1873 		rParseIter.getColumnRange( pColumnRef, aColumnName, aTableRange );
1874 
1875 		sal_Bool bFound(sal_False);
1876 		DBG_ASSERT(aColumnName.getLength(),"Columnname darf nicht leer sein");
1877 		if (!aTableRange.getLength())
1878 		{
1879 			// SELECT column, ...
1880             bFound = NULL != lcl_findColumnInTables( aColumnName, *pTabList, _raInfo );
1881 			if ( bFound && ( aColumnName.toChar() != '*' ) )
1882 				_raInfo->SetFieldAlias(aColumnAlias);
1883 		}
1884 		else
1885 		{
1886 			// SELECT range.column, ...
1887 			OQueryTableWindow* pTabWin = static_cast<OQueryTableView*>(_pView->getTableView())->FindTable(aTableRange);
1888 
1889 			if (pTabWin && pTabWin->ExistsField(aColumnName, _raInfo))
1890 			{
1891 				if(aColumnName.toChar() != '*')
1892 					_raInfo->SetFieldAlias(aColumnAlias);
1893 				bFound = sal_True;
1894 			}
1895 		}
1896 		if (!bFound)
1897 		{
1898 			_raInfo->SetTable(::rtl::OUString());
1899 			_raInfo->SetAlias(::rtl::OUString());
1900 			_raInfo->SetField(aColumnName);
1901 			_raInfo->SetFieldAlias(aColumnAlias);	// nyi : hier ein fortlaufendes Expr_1, Expr_2 ...
1902 			_raInfo->SetFunctionType(FKT_OTHER);
1903 		}
1904 	}
1905 	//-----------------------------------------------------------------------------
1906 	sal_Bool checkJoinConditions(	const OQueryDesignView* _pView,
1907 									const ::connectivity::OSQLParseNode* _pNode )
1908 	{
1909 		const ::connectivity::OSQLParseNode* pJoinNode = NULL;
1910 		sal_Bool bRet = sal_True;
1911 		if (SQL_ISRULE(_pNode,qualified_join))
1912 			pJoinNode = _pNode;
1913         else if (SQL_ISRULE(_pNode,table_ref)
1914                 &&  _pNode->count() == 3
1915                 &&  SQL_ISPUNCTUATION(_pNode->getChild(0),"(")
1916                 &&  SQL_ISPUNCTUATION(_pNode->getChild(2),")") ) // '(' joined_table ')'
1917 			pJoinNode = _pNode->getChild(1);
1918 		else if (! ( SQL_ISRULE(_pNode, table_ref) && _pNode->count() == 2) ) // table_node table_primary_as_range_column
1919 			bRet = sal_False;
1920 
1921 		if (pJoinNode && !InsertJoin(_pView,pJoinNode))
1922 			bRet = sal_False;
1923 		return bRet;
1924 	}
1925 	//-----------------------------------------------------------------------------
1926 	sal_Bool InsertJoin(const OQueryDesignView* _pView,
1927 						const ::connectivity::OSQLParseNode *pNode)
1928 	{
1929 		DBG_ASSERT( SQL_ISRULE( pNode, qualified_join ) || SQL_ISRULE( pNode, joined_table ) || SQL_ISRULE( pNode, cross_union ),
1930 			"OQueryDesignView::InsertJoin: Fehler im Parse Tree");
1931 
1932         if (SQL_ISRULE(pNode,joined_table))
1933             return InsertJoin(_pView,pNode->getChild(1));
1934 
1935 		// first check the left and right side
1936         const ::connectivity::OSQLParseNode* pRightTableRef = pNode->getChild(3); // table_ref
1937         if ( SQL_ISRULE(pNode, qualified_join) && SQL_ISTOKEN(pNode->getChild(1),NATURAL) )
1938             pRightTableRef = pNode->getChild(4); // table_ref
1939 
1940 		if ( !checkJoinConditions(_pView,pNode->getChild(0)) || !checkJoinConditions(_pView,pRightTableRef))
1941 			return sal_False;
1942 
1943 		// named column join wird sp�ter vieleicht noch implementiert
1944 		// SQL_ISRULE(pNode->getChild(4),named_columns_join)
1945         EJoinType eJoinType = INNER_JOIN;
1946         bool bNatural = false;
1947 		if ( SQL_ISRULE(pNode, qualified_join) )
1948 		{
1949 			::connectivity::OSQLParseNode* pJoinType = pNode->getChild(1); // join_type
1950             if ( SQL_ISTOKEN(pJoinType,NATURAL) )
1951             {
1952                 bNatural = true;
1953                 pJoinType = pNode->getChild(2);
1954             }
1955 
1956 			if (SQL_ISRULE(pJoinType,join_type) && (!pJoinType->count() || SQL_ISTOKEN(pJoinType->getChild(0),INNER)))
1957 			{
1958 				eJoinType = INNER_JOIN;
1959 			}
1960 			else
1961 			{
1962 				if (SQL_ISRULE(pJoinType,join_type))	   // eine Ebene tiefer
1963 					pJoinType = pJoinType->getChild(0);
1964 
1965 				if (SQL_ISTOKEN(pJoinType->getChild(0),LEFT))
1966 					eJoinType = LEFT_JOIN;
1967 				else if(SQL_ISTOKEN(pJoinType->getChild(0),RIGHT))
1968 					eJoinType = RIGHT_JOIN;
1969 				else
1970 					eJoinType = FULL_JOIN;
1971 			}
1972             if ( SQL_ISRULE(pNode->getChild(4),join_condition) )
1973             {
1974 			    if ( InsertJoinConnection(_pView,pNode->getChild(4)->getChild(1), eJoinType,pNode->getChild(0),pRightTableRef) != eOk )
1975 				    return sal_False;
1976             }
1977 		}
1978         else if ( SQL_ISRULE(pNode, cross_union) )
1979         {
1980             eJoinType = CROSS_JOIN;
1981             pRightTableRef = pNode->getChild(pNode->count() - 1);
1982         }
1983         else
1984 			return sal_False;
1985 
1986         if ( eJoinType == CROSS_JOIN || bNatural )
1987         {
1988 
1989             OQueryTableWindow*	pLeftWindow = static_cast<OQueryTableView*>(_pView->getTableView())->FindTable( getTableRange(_pView,pNode->getChild(0)) );
1990             OQueryTableWindow*	pRightWindow = static_cast<OQueryTableView*>(_pView->getTableView())->FindTable( getTableRange(_pView,pRightTableRef) );
1991             OSL_ENSURE(pLeftWindow && pRightWindow,"Table Windows could not be found!");
1992             if ( !pLeftWindow || !pRightWindow )
1993                 return sal_False;
1994 
1995             OTableFieldDescRef aDragLeft  = new OTableFieldDesc();
1996             aDragLeft->SetTabWindow(pLeftWindow);
1997             aDragLeft->SetTable(pLeftWindow->GetTableName());
1998             aDragLeft->SetAlias(pLeftWindow->GetAliasName());
1999 
2000 			OTableFieldDescRef aDragRight = new OTableFieldDesc();
2001             aDragRight->SetTabWindow(pRightWindow);
2002             aDragRight->SetTable(pRightWindow->GetTableName());
2003             aDragRight->SetAlias(pRightWindow->GetAliasName());
2004 
2005             insertConnection(_pView,eJoinType,aDragLeft,aDragRight,bNatural);
2006         }
2007 
2008 
2009 		return sal_True;
2010 	}
2011 	//------------------------------------------------------------------------------
2012 	void insertUnUsedFields(OQueryDesignView* _pView,OSelectionBrowseBox* _pSelectionBrw)
2013 	{
2014 		// now we have to insert the fields which aren't in the statement
2015 		OQueryController& rController = static_cast<OQueryController&>(_pView->getController());
2016 		OTableFields& rUnUsedFields = rController.getUnUsedFields();
2017         OTableFields::iterator aEnd = rUnUsedFields.end();
2018 		for(OTableFields::iterator aIter = rUnUsedFields.begin();aIter != aEnd;++aIter)
2019 			if(_pSelectionBrw->InsertField(*aIter,BROWSER_INVALIDID,sal_False,sal_False).isValid())
2020 				(*aIter) = NULL;
2021 		OTableFields().swap( rUnUsedFields );
2022 	}
2023 
2024 	//------------------------------------------------------------------------------
2025 	SqlParseError InitFromParseNodeImpl(OQueryDesignView* _pView,OSelectionBrowseBox* _pSelectionBrw)
2026 	{
2027 		SqlParseError eErrorCode = eOk;
2028 
2029 		OQueryController& rController = static_cast<OQueryController&>(_pView->getController());
2030 
2031 		_pSelectionBrw->PreFill();
2032 		_pSelectionBrw->SetReadOnly(rController.isReadOnly());
2033 		_pSelectionBrw->Fill();
2034 
2035 
2036 		::connectivity::OSQLParseTreeIterator& aIterator = rController.getParseIterator();
2037 		const ::connectivity::OSQLParseNode* pParseTree = aIterator.getParseTree();
2038 
2039         do
2040         {
2041             if ( !pParseTree )
2042             {
2043 			    // now we have to insert the fields which aren't in the statement
2044 			    insertUnUsedFields(_pView,_pSelectionBrw);
2045                 break;
2046             }
2047 
2048             if ( !rController.isEsacpeProcessing() ) // not allowed in this mode
2049             {
2050 			    eErrorCode = eNativeMode;
2051                 break;
2052             }
2053 
2054 			if ( !( SQL_ISRULE( pParseTree, select_statement ) ) )
2055             {
2056 				eErrorCode = eNoSelectStatement;
2057                 break;
2058             }
2059 
2060             const OSQLParseNode* pTableExp = pParseTree->getChild(3);
2061             if ( pTableExp->getChild(6)->count() > 0 || pTableExp->getChild(7)->count() > 0 || pTableExp->getChild(8)->count() > 0)
2062             {
2063 				eErrorCode = eStatementTooComplex;
2064                 break;
2065             }
2066 
2067 			Reference< XConnection> xConnection = rController.getConnection();
2068 			if ( !xConnection.is() )
2069             {
2070                 DBG_ERROR( "InitFromParseNodeImpl: no connection? no connection!" );
2071                 break;
2072             }
2073 
2074 			const OSQLTables& aMap = aIterator.getTables();
2075 			::comphelper::UStringMixLess aTmp(aMap.key_comp());
2076 			::comphelper::UStringMixEqual aKeyComp( aTmp.isCaseSensitive() );
2077 
2078 			Reference< XDatabaseMetaData >  xMetaData = xConnection->getMetaData();
2079 			try
2080 			{
2081 			    sal_Int32 nMax = xMetaData->getMaxTablesInSelect();
2082                 if ( nMax && nMax < (sal_Int32)aMap.size() )
2083                 {
2084 					eErrorCode = eTooManyTables;
2085                     break;
2086                 }
2087 
2088 				::rtl::OUString sComposedName;
2089 				::rtl::OUString aQualifierName;
2090 				::rtl::OUString sAlias;
2091 
2092 				OQueryTableView* pTableView = static_cast<OQueryTableView*>(_pView->getTableView());
2093                 pTableView->clearLayoutInformation();
2094 				OSQLTables::const_iterator aIter = aMap.begin();
2095                 OSQLTables::const_iterator aEnd = aMap.end();
2096 				for(;aIter != aEnd;++aIter)
2097 				{
2098 					OSQLTable xTable = aIter->second;
2099                     Reference< XPropertySet > xTableProps( xTable, UNO_QUERY_THROW );
2100 
2101 					sAlias = aIter->first;
2102 
2103                     // check whether this is a query
2104                     Reference< XPropertySetInfo > xPSI = xTableProps->getPropertySetInfo();
2105                     bool bIsQuery = xPSI.is() && xPSI->hasPropertyByName( PROPERTY_COMMAND );
2106 
2107                     if ( bIsQuery )
2108                         OSL_VERIFY( xTableProps->getPropertyValue( PROPERTY_NAME ) >>= sComposedName );
2109                     else
2110                     {
2111                         sComposedName = ::dbtools::composeTableName( xMetaData, xTableProps, ::dbtools::eInDataManipulation, false, false, false );
2112 
2113                         // if the alias is the complete (composed) table, then shorten it
2114                         if ( aKeyComp( sComposedName, aIter->first ) )
2115 					    {
2116 						    ::rtl::OUString sCatalog, sSchema, sTable;
2117 						    ::dbtools::qualifiedNameComponents( xMetaData, sComposedName, sCatalog, sSchema, sTable, ::dbtools::eInDataManipulation );
2118 						    sAlias = sTable;
2119 					    }
2120                     }
2121 
2122                     // find the existent window for this alias
2123 					OQueryTableWindow* pExistentWin	= pTableView->FindTable( sAlias );
2124 					if ( !pExistentWin )
2125 					{
2126 						pTableView->AddTabWin( sComposedName, sAlias, sal_False );  // don't create data here
2127 					}
2128 					else
2129 					{
2130 						// there already exists a window for this alias ....
2131 						if ( !aKeyComp( pExistentWin->GetData()->GetComposedName(), sComposedName ) )
2132 							// ... but for another complete table name -> new window
2133 							pTableView->AddTabWin(sComposedName, sAlias);
2134 					}
2135 				}
2136 
2137 				// now delete the data for which we haven't any tablewindow
2138 				OJoinTableView::OTableWindowMap aTableMap(*pTableView->GetTabWinMap());
2139 				OJoinTableView::OTableWindowMap::iterator aIterTableMap = aTableMap.begin();
2140                 OJoinTableView::OTableWindowMap::iterator aIterTableEnd = aTableMap.end();
2141 				for(;aIterTableMap != aIterTableEnd;++aIterTableMap)
2142 				{
2143 					if(aMap.find(aIterTableMap->second->GetComposedName())	== aMap.end() &&
2144 						aMap.find(aIterTableMap->first)						== aMap.end())
2145 						pTableView->RemoveTabWin(aIterTableMap->second);
2146 				}
2147 
2148 				if ( eOk == (eErrorCode = FillOuterJoins(_pView,pTableExp->getChild(0)->getChild(1))) )
2149 				{
2150 					// check if we have a distinct statement
2151 					if(SQL_ISTOKEN(pParseTree->getChild(1),DISTINCT))
2152 					{
2153 						rController.setDistinct(sal_True);
2154 						rController.InvalidateFeature(SID_QUERY_DISTINCT_VALUES);
2155 					}
2156 					if ( (eErrorCode = InstallFields(_pView,pParseTree, pTableView->GetTabWinMap())) == eOk )
2157 					{
2158 						// GetSelectionCriteria must be called before GetHavingCriteria
2159 						sal_uInt16 nLevel=0;
2160 
2161 						if ( eOk == (eErrorCode = GetSelectionCriteria(_pView,_pSelectionBrw,pParseTree,nLevel)) )
2162 						{
2163 							if ( eOk == (eErrorCode = GetGroupCriteria(_pView,_pSelectionBrw,pParseTree)) )
2164 							{
2165 								if ( eOk == (eErrorCode = GetHavingCriteria(_pView,_pSelectionBrw,pParseTree,nLevel)) )
2166 								{
2167 									if ( eOk == (eErrorCode = GetOrderCriteria(_pView,_pSelectionBrw,pParseTree)) )
2168 										insertUnUsedFields(_pView,_pSelectionBrw);
2169 								}
2170 							}
2171 						}
2172 					}
2173 				}
2174 			}
2175 			catch(SQLException&)
2176 			{
2177 				OSL_ASSERT(!"getMaxTablesInSelect!");
2178 			}
2179 		}
2180         while ( false );
2181 
2182 		// Durch das Neuerzeugen wurden wieder Undo-Actions in den Manager gestellt
2183 		rController.ClearUndoManager();
2184 		_pSelectionBrw->Invalidate();
2185 		return eErrorCode;
2186 	}
2187 	//------------------------------------------------------------------------------
2188 	/** fillSelectSubList
2189 		@return
2190 			<TRUE/> when columns could be inserted otherwise <FALSE/>
2191 	*/
2192 	//------------------------------------------------------------------------------
2193 	SqlParseError fillSelectSubList(	OQueryDesignView* _pView,
2194 								OJoinTableView::OTableWindowMap* _pTabList)
2195 	{
2196 		SqlParseError eErrorCode = eOk;
2197 		sal_Bool bFirstField = sal_True;
2198 		::rtl::OUString sAsterix(RTL_CONSTASCII_USTRINGPARAM("*"));
2199 		OJoinTableView::OTableWindowMap::iterator aIter = _pTabList->begin();
2200         OJoinTableView::OTableWindowMap::iterator aEnd = _pTabList->end();
2201 		for(;aIter != aEnd && eOk == eErrorCode ;++aIter)
2202 		{
2203 			OQueryTableWindow* pTabWin = static_cast<OQueryTableWindow*>(aIter->second);
2204             OTableFieldDescRef	aInfo = new OTableFieldDesc();
2205 			if (pTabWin->ExistsField( sAsterix, aInfo ))
2206 			{
2207 				eErrorCode = _pView->InsertField(aInfo, sal_True, bFirstField);
2208 				bFirstField = sal_False;
2209 			}
2210 		}
2211 		return eErrorCode;
2212 	}
2213 	//------------------------------------------------------------------------------
2214 	SqlParseError InstallFields(OQueryDesignView* _pView,
2215 								const ::connectivity::OSQLParseNode* pNode,
2216 								OJoinTableView::OTableWindowMap* pTabList )
2217 	{
2218 		if( pNode==0 || !SQL_ISRULE(pNode,select_statement))
2219 			return eNoSelectStatement;
2220 
2221 		::connectivity::OSQLParseNode* pParseTree = pNode->getChild(2); // selection
2222 		sal_Bool bFirstField = sal_True;	// bei der Initialisierung mu� auf alle Faelle das erste Feld neu aktiviert werden
2223 
2224 		SqlParseError eErrorCode = eOk;
2225 
2226 		if ( pParseTree->isRule() && SQL_ISPUNCTUATION(pParseTree->getChild(0),"*") )
2227 		{
2228 			// SELECT * ...
2229 			eErrorCode = fillSelectSubList(_pView,pTabList);
2230 		}
2231 		else if (SQL_ISRULE(pParseTree,scalar_exp_commalist) )
2232 		{
2233 			// SELECT column, ...
2234 			OQueryController& rController = static_cast<OQueryController&>(_pView->getController());
2235 			Reference< XConnection> xConnection = rController.getConnection();
2236 
2237 			::rtl::OUString aColumnName,aTableRange;
2238 			for (sal_uInt32 i = 0; i < pParseTree->count() && eOk == eErrorCode ; ++i)
2239 			{
2240 				::connectivity::OSQLParseNode * pColumnRef = pParseTree->getChild(i);
2241 
2242                 do {
2243 
2244                 if ( SQL_ISRULE(pColumnRef,select_sublist) )
2245 				{
2246 					 eErrorCode = fillSelectSubList(_pView,pTabList);
2247                      break;
2248 				}
2249 
2250                 if ( SQL_ISRULE(pColumnRef,derived_column) )
2251 				{
2252 					::rtl::OUString aColumnAlias(rController.getParseIterator().getColumnAlias(pColumnRef)); // kann leer sein
2253 					pColumnRef = pColumnRef->getChild(0);
2254 					OTableFieldDescRef aInfo = new OTableFieldDesc();
2255 
2256 					if (	pColumnRef->count() == 3 &&
2257 							SQL_ISPUNCTUATION(pColumnRef->getChild(0),"(") &&
2258 							SQL_ISPUNCTUATION(pColumnRef->getChild(2),")")
2259 						)
2260 						pColumnRef = pColumnRef->getChild(1);
2261 
2262 					if (SQL_ISRULE(pColumnRef,column_ref))
2263 					{
2264 						InsertColumnRef(_pView,pColumnRef,aColumnName,aColumnAlias,aTableRange,aInfo,pTabList);
2265 						eErrorCode = _pView->InsertField(aInfo, sal_True, bFirstField);
2266 						bFirstField = sal_False;
2267 					}
2268 					else if(SQL_ISRULEOR2(pColumnRef,general_set_fct ,set_fct_spec)	||
2269 							SQL_ISRULEOR2(pColumnRef,position_exp,extract_exp)		||
2270 							SQL_ISRULEOR2(pColumnRef,fold,char_substring_fct)		||
2271 							SQL_ISRULEOR2(pColumnRef,length_exp,char_value_fct))
2272 					{
2273 						::rtl::OUString aColumns;
2274                         pColumnRef->parseNodeToPredicateStr(aColumns,
2275 												            xConnection,
2276 												            rController.getNumberFormatter(),
2277 												            _pView->getLocale(),
2278 												            static_cast<sal_Char>(_pView->getDecimalSeparator().toChar()),
2279 												            &rController.getParser().getContext());
2280 						//pColumnRef->parseNodeToStr(	aColumns,
2281 						//							xConnection,
2282 						//							&rController.getParser().getContext(),
2283 						//							sal_True,
2284 						//							sal_True); // quote is to true because we need quoted elements inside the function
2285 
2286 						sal_Int32 nFunctionType = FKT_NONE;
2287 						::connectivity::OSQLParseNode* pParamRef = NULL;
2288 						sal_Int32 nColumnRefPos = pColumnRef->count() - 2;
2289 						if ( nColumnRefPos >= 0 && static_cast<sal_uInt32>(nColumnRefPos) < pColumnRef->count() )
2290 							pParamRef = pColumnRef->getChild(nColumnRefPos);
2291 
2292 						if (	SQL_ISRULE(pColumnRef,general_set_fct)
2293 							&&	SQL_ISRULE(pParamRef,column_ref) )
2294 						{
2295 							// Parameter auf Columnref pr"ufen
2296 							InsertColumnRef(_pView,pParamRef,aColumnName,aColumnAlias,aTableRange,aInfo,pTabList);
2297 						}
2298 						else if ( SQL_ISRULE(pColumnRef,general_set_fct) )
2299 						{
2300 							if ( pParamRef && pParamRef->getTokenValue().toChar() == '*' )
2301 							{
2302 								OJoinTableView::OTableWindowMap::iterator aIter = pTabList->begin();
2303                                 OJoinTableView::OTableWindowMap::iterator aEnd  = pTabList->end();
2304 								for(;aIter != aEnd;++aIter)
2305 								{
2306 									OQueryTableWindow* pTabWin = static_cast<OQueryTableWindow*>(aIter->second);
2307 									if (pTabWin->ExistsField( ::rtl::OUString::createFromAscii("*"), aInfo ))
2308 									{
2309 										aInfo->SetAlias(String());
2310 										aInfo->SetTable(String());
2311 										break;
2312 									}
2313 								}
2314 							}
2315 							else
2316 							{
2317 								::rtl::OUString sFieldName = aColumns;
2318 								if ( pParamRef )
2319 								{	// we got an aggregate function but without column name inside
2320 									// so we set the whole argument of the function as field name
2321 									nFunctionType |= FKT_NUMERIC;
2322 									sFieldName = ::rtl::OUString();
2323 									pParamRef->parseNodeToStr(	sFieldName,
2324 														xConnection,
2325 														&rController.getParser().getContext(),
2326 														sal_True,
2327 														sal_True); // quote is to true because we need quoted elements inside the function
2328 								}
2329 								aInfo->SetDataType(DataType::DOUBLE);
2330 								aInfo->SetFieldType(TAB_NORMAL_FIELD);
2331 								aInfo->SetField(sFieldName);
2332 							}
2333 							aInfo->SetTabWindow(NULL);
2334 							aInfo->SetFieldAlias(aColumnAlias);
2335 						}
2336 						else
2337 						{
2338                             _pView->fillFunctionInfo(pColumnRef,aColumns,aInfo);
2339 							aInfo->SetFieldAlias(aColumnAlias);
2340 						}
2341 
2342 						if ( SQL_ISRULE(pColumnRef,general_set_fct) )
2343 						{
2344 							aInfo->SetFunctionType(nFunctionType|FKT_AGGREGATE);
2345 							String aCol(aColumns);
2346 							aInfo->SetFunction(aCol.GetToken(0,'(').EraseTrailingChars(' '));
2347 						}
2348 						else
2349 							aInfo->SetFunctionType(nFunctionType|FKT_OTHER);
2350 
2351 						eErrorCode = _pView->InsertField(aInfo, sal_True, bFirstField);
2352 						bFirstField = sal_False;
2353 					}
2354 					else //if(SQL_ISRULE(pColumnRef,num_value_exp)	|| SQL_ISRULE(pColumnRef,term))
2355 					{
2356 						::rtl::OUString aColumns;
2357 						pColumnRef->parseNodeToStr(	aColumns,
2358 													xConnection,
2359 													&rController.getParser().getContext(),
2360 													sal_True,
2361 													sal_True); // quote is to true because we need quoted elements inside the function
2362 
2363                         aInfo->SetTabWindow( NULL );
2364 
2365                         // since we support queries in queries, the thingie might belong to an existing "table"
2366                         OQueryTableWindow* pExistingTable = lcl_findColumnInTables( aColumns, *pTabList, aInfo );
2367                         if ( pExistingTable )
2368                         {
2369                             aInfo->SetTabWindow( pExistingTable );
2370 					        aInfo->SetTable( pExistingTable->GetTableName() );
2371 					        aInfo->SetAlias( pExistingTable->GetAliasName() );
2372                         }
2373 
2374                         aInfo->SetDataType(DataType::DOUBLE);
2375 						aInfo->SetFieldType(TAB_NORMAL_FIELD);
2376 						aInfo->SetField(aColumns);
2377 						aInfo->SetFieldAlias(aColumnAlias);
2378 						aInfo->SetFunctionType(FKT_NUMERIC | FKT_OTHER);
2379 
2380 						eErrorCode = _pView->InsertField(aInfo, sal_True, bFirstField);
2381 						bFirstField = sal_False;
2382 					}
2383 
2384                     break;
2385 				}
2386 
2387                 DBG_ERROR( "InstallFields: don't know how to interpret this parse node!" );
2388 
2389                 } while ( false );
2390 			}
2391 		}
2392 		else
2393 			eErrorCode = eStatementTooComplex;
2394 
2395 		return eErrorCode;
2396 	}
2397 	//------------------------------------------------------------------------------
2398 	SqlParseError GetOrderCriteria(	OQueryDesignView* _pView,
2399 							OSelectionBrowseBox* _pSelectionBrw,
2400 							const ::connectivity::OSQLParseNode* pParseRoot )
2401 	{
2402 		SqlParseError eErrorCode = eOk;
2403 		if (!pParseRoot->getChild(3)->getChild(4)->isLeaf())
2404 		{
2405 			::connectivity::OSQLParseNode* pNode = pParseRoot->getChild(3)->getChild(4)->getChild(2);
2406 			::connectivity::OSQLParseNode* pParamRef = NULL;
2407 
2408             OQueryController& rController = static_cast<OQueryController&>(_pView->getController());
2409 			EOrderDir eOrderDir;
2410 			OTableFieldDescRef aDragLeft = new OTableFieldDesc();
2411 			for( sal_uInt32 i=0 ; i<pNode->count() ; i++ )
2412 			{
2413 				eOrderDir = ORDER_ASC;
2414 				::connectivity::OSQLParseNode*	pChild = pNode->getChild( i );
2415 
2416 				if (SQL_ISTOKEN( pChild->getChild(1), DESC ) )
2417 					eOrderDir = ORDER_DESC;
2418 
2419                 ::connectivity::OSQLParseNode* pArgument = pChild->getChild(0);
2420 
2421 				if(SQL_ISRULE(pArgument,column_ref))
2422 				{
2423 					if( eOk == FillDragInfo(_pView,pArgument,aDragLeft))
2424 						_pSelectionBrw->AddOrder( aDragLeft, eOrderDir, i);
2425 					else // it could be a alias name for a field
2426 					{
2427 						::rtl::OUString aTableRange,aColumnName;
2428 						::connectivity::OSQLParseTreeIterator& rParseIter = rController.getParseIterator();
2429 						rParseIter.getColumnRange( pArgument, aColumnName, aTableRange );
2430 
2431 						OTableFields& aList = rController.getTableFieldDesc();
2432 						OTableFields::iterator aIter = aList.begin();
2433                         OTableFields::iterator aEnd = aList.end();
2434 						for(;aIter != aEnd;++aIter)
2435 						{
2436 							OTableFieldDescRef pEntry = *aIter;
2437 							if(pEntry.isValid() && pEntry->GetFieldAlias() == aColumnName)
2438 								pEntry->SetOrderDir( eOrderDir );
2439 						}
2440 					}
2441 				}
2442 				else if(SQL_ISRULE(pArgument, general_set_fct ) &&
2443 						SQL_ISRULE(pParamRef = pArgument->getChild(pArgument->count()-2),column_ref) &&
2444 						eOk == FillDragInfo(_pView,pParamRef,aDragLeft))
2445 					_pSelectionBrw->AddOrder( aDragLeft, eOrderDir, i );
2446                 else if( SQL_ISRULE(pArgument, set_fct_spec ) )
2447                 {
2448 
2449 		            Reference< XConnection> xConnection = rController.getConnection();
2450 		            if(xConnection.is())
2451 		            {
2452                         ::rtl::OUString sCondition;
2453 			            pArgument->parseNodeToPredicateStr(sCondition,
2454 												            xConnection,
2455 												            rController.getNumberFormatter(),
2456 												            _pView->getLocale(),
2457 												            static_cast<sal_Char>(_pView->getDecimalSeparator().toChar()),
2458 												            &rController.getParser().getContext());
2459                         _pView->fillFunctionInfo(pArgument,sCondition,aDragLeft);
2460                         aDragLeft->SetFunctionType(FKT_OTHER);
2461                         aDragLeft->SetOrderDir(eOrderDir);
2462                         aDragLeft->SetVisible(sal_False);
2463                         _pSelectionBrw->AddOrder( aDragLeft, eOrderDir, i );
2464                     }
2465                     else
2466                         eErrorCode = eColumnNotFound;
2467                 }
2468 				else
2469 					eErrorCode = eColumnNotFound;
2470 			}
2471 		}
2472 		return eErrorCode;
2473 	}
2474 	//------------------------------------------------------------------------------
2475 	SqlParseError GetHavingCriteria(	OQueryDesignView* _pView,
2476 							OSelectionBrowseBox* _pSelectionBrw,
2477 							const ::connectivity::OSQLParseNode* pSelectRoot,
2478 							sal_uInt16& rLevel )
2479 	{
2480 		SqlParseError eErrorCode = eOk;
2481 		if (!pSelectRoot->getChild(3)->getChild(3)->isLeaf())
2482 			eErrorCode = GetORCriteria(_pView,_pSelectionBrw,pSelectRoot->getChild(3)->getChild(3)->getChild(1),rLevel, sal_True);
2483 		return eErrorCode;
2484 	}
2485 	//------------------------------------------------------------------------------
2486 	SqlParseError GetGroupCriteria(	OQueryDesignView* _pView,
2487 							OSelectionBrowseBox* _pSelectionBrw,
2488 							const ::connectivity::OSQLParseNode* pSelectRoot )
2489 	{
2490 		SqlParseError eErrorCode = eOk;
2491 		if (!pSelectRoot->getChild(3)->getChild(2)->isLeaf()) // opt_group_by_clause
2492 		{
2493             OQueryController& rController = static_cast<OQueryController&>(_pView->getController());
2494 			::connectivity::OSQLParseNode* pGroupBy = pSelectRoot->getChild(3)->getChild(2)->getChild(2);
2495 
2496 			for( sal_uInt32 i=0 ; i < pGroupBy->count() && eOk == eErrorCode; ++i )
2497 			{
2498                 OTableFieldDescRef aDragInfo = new OTableFieldDesc();
2499                 ::connectivity::OSQLParseNode* pParamRef = NULL;
2500 				::connectivity::OSQLParseNode* pArgument = pGroupBy->getChild( i );
2501 				if(SQL_ISRULE(pArgument,column_ref))
2502 				{
2503 					if ( eOk == (eErrorCode = FillDragInfo(_pView,pArgument,aDragInfo)) )
2504 					{
2505 						aDragInfo->SetGroupBy(sal_True);
2506 						_pSelectionBrw->AddGroupBy(aDragInfo,i);
2507 					}
2508 				}
2509                 else if(SQL_ISRULE(pArgument, general_set_fct ) &&
2510 						SQL_ISRULE(pParamRef = pArgument->getChild(pArgument->count()-2),column_ref) &&
2511 						eOk == FillDragInfo(_pView,pParamRef,aDragInfo))
2512                 {
2513                     aDragInfo->SetGroupBy(sal_True);
2514 					_pSelectionBrw->AddGroupBy( aDragInfo, i );
2515                 }
2516                 else if( SQL_ISRULE(pArgument, set_fct_spec ) )
2517                 {
2518 		            Reference< XConnection> xConnection = rController.getConnection();
2519 		            if(xConnection.is())
2520 		            {
2521                         ::rtl::OUString sGroupByExpression;
2522 			            pArgument->parseNodeToStr(	sGroupByExpression,
2523 													xConnection,
2524 													&rController.getParser().getContext(),
2525 													sal_True,
2526 													sal_True); // quote is to true because we need quoted elements inside the function
2527                         _pView->fillFunctionInfo(pArgument,sGroupByExpression,aDragInfo);
2528                         aDragInfo->SetFunctionType(FKT_OTHER);
2529                         aDragInfo->SetGroupBy(sal_True);
2530                         aDragInfo->SetVisible(sal_False);
2531                         _pSelectionBrw->AddGroupBy( aDragInfo, i );
2532                     }
2533                     else
2534                         eErrorCode = eColumnNotFound;
2535                 }
2536 			}
2537 		}
2538 		return eErrorCode;
2539 	}
2540 
2541     //------------------------------------------------------------------------------
2542 	String getParseErrorMessage( SqlParseError _eErrorCode )
2543 	{
2544 		sal_uInt16 nResId;
2545 		switch(_eErrorCode)
2546 		{
2547 			case eIllegalJoin:
2548 				nResId = STR_QRY_ILLEGAL_JOIN;
2549 				break;
2550 			case eStatementTooLong:
2551 				nResId = STR_QRY_TOO_LONG_STATEMENT;
2552 				break;
2553 			case eNoConnection:
2554 				nResId = STR_QRY_SYNTAX;
2555 				break;
2556 			case eNoSelectStatement:
2557 				nResId = STR_QRY_NOSELECT;
2558 				break;
2559 			case eColumnInLikeNotFound:
2560 				nResId = STR_QRY_SYNTAX;
2561 				break;
2562 			case eNoColumnInLike:
2563 				nResId = STR_QRY_SYNTAX;
2564 				break;
2565 			case eColumnNotFound:
2566 				nResId = STR_QRY_SYNTAX;
2567 				break;
2568 			case eNativeMode:
2569 				nResId = STR_QRY_NATIVE;
2570 				break;
2571 			case eTooManyTables:
2572 				nResId = STR_QRY_TOO_MANY_TABLES;
2573 				break;
2574 			case eTooManyConditions:
2575 				nResId = STR_QRY_TOOMANYCOND;
2576 				break;
2577 			case eTooManyColumns:
2578 				nResId = STR_QRY_TOO_MANY_COLUMNS;
2579 				break;
2580 			case eStatementTooComplex:
2581 				nResId = STR_QRY_TOOCOMPLEX;
2582 				break;
2583 			default:
2584 				nResId = STR_QRY_SYNTAX;
2585 				break;
2586 		}
2587 		;
2588         return String( ModuleRes( nResId ) );
2589 	}
2590 
2591 	//------------------------------------------------------------------------------
2592 	//------------------------------------------------------------------------------
2593 }
2594 // end of anonymouse namespace
2595 DBG_NAME(OQueryDesignView)
2596 
2597 OQueryDesignView::OQueryDesignView(	OQueryContainerWindow* _pParent,
2598 									OQueryController& _rController,
2599 									const Reference< XMultiServiceFactory >& _rFactory)
2600 	:OQueryView( _pParent, _rController, _rFactory )
2601 	,m_aSplitter( this )
2602 	,m_eChildFocus(NONE)
2603 	,m_bInKeyEvent(sal_False)
2604 	,m_bInSplitHandler( sal_False )
2605 {
2606     DBG_CTOR(OQueryDesignView,NULL);
2607 
2608 	try
2609 	{
2610 		SvtSysLocale aSysLocale;
2611 		m_aLocale = aSysLocale.GetLocaleData().getLocale();
2612 		m_sDecimalSep = aSysLocale.GetLocaleData().getNumDecimalSep();
2613 	}
2614 	catch(Exception&)
2615 	{
2616 	}
2617 
2618 	m_pSelectionBox = new OSelectionBrowseBox(this);
2619 
2620 	setNoneVisbleRow(static_cast<OQueryController&>(getController()).getVisibleRows());
2621 	m_pSelectionBox->Show();
2622 	// Splitter einrichten
2623 	m_aSplitter.SetSplitHdl(LINK(this, OQueryDesignView,SplitHdl));
2624 	m_aSplitter.Show();
2625 
2626 }
2627 // -----------------------------------------------------------------------------
2628 OQueryDesignView::~OQueryDesignView()
2629 {
2630 	if ( m_pTableView )
2631 		::dbaui::notifySystemWindow(this,m_pTableView,::comphelper::mem_fun(&TaskPaneList::RemoveWindow));
2632 	::std::auto_ptr<Window> aTemp(m_pSelectionBox);
2633 	m_pSelectionBox = NULL;
2634 
2635     DBG_DTOR(OQueryDesignView,NULL);
2636 }
2637 //------------------------------------------------------------------------------
2638 IMPL_LINK( OQueryDesignView, SplitHdl, void*, /*p*/ )
2639 {
2640 	if (!getController().isReadOnly())
2641 	{
2642 		m_bInSplitHandler = sal_True;
2643 		m_aSplitter.SetPosPixel( Point( m_aSplitter.GetPosPixel().X(),m_aSplitter.GetSplitPosPixel() ) );
2644 		static_cast<OQueryController&>(getController()).setSplitPos(m_aSplitter.GetSplitPosPixel());
2645 		static_cast<OQueryController&>(getController()).setModified( sal_True );
2646 		Resize();
2647 		m_bInSplitHandler = sal_True;
2648 	}
2649 	return 0L;
2650 }
2651 // -------------------------------------------------------------------------
2652 void OQueryDesignView::Construct()
2653 {
2654 	m_pTableView = new OQueryTableView(m_pScrollWindow,this);
2655 	::dbaui::notifySystemWindow(this,m_pTableView,::comphelper::mem_fun(&TaskPaneList::AddWindow));
2656 	OQueryView::Construct();
2657 }
2658 // -----------------------------------------------------------------------------
2659 void OQueryDesignView::initialize()
2660 {
2661 	if(static_cast<OQueryController&>(getController()).getSplitPos() != -1)
2662 	{
2663 		m_aSplitter.SetPosPixel( Point( m_aSplitter.GetPosPixel().X(),static_cast<OQueryController&>(getController()).getSplitPos() ) );
2664 		m_aSplitter.SetSplitPosPixel(static_cast<OQueryController&>(getController()).getSplitPos());
2665 	}
2666 	m_pSelectionBox->initialize();
2667 	reset();
2668 }
2669 // -------------------------------------------------------------------------
2670 void OQueryDesignView::resizeDocumentView(Rectangle& _rPlayground)
2671 {
2672 	Point aPlaygroundPos( _rPlayground.TopLeft() );
2673 	Size aPlaygroundSize( _rPlayground.GetSize() );
2674 
2675 	// calc the split pos, and forward it to the controller
2676 	sal_Int32 nSplitPos = static_cast<OQueryController&>(getController()).getSplitPos();
2677 	if ( 0 != aPlaygroundSize.Height() )
2678 	{
2679 		if	(	( -1 == nSplitPos )
2680 			||	( nSplitPos >= aPlaygroundSize.Height() )
2681 			)
2682 		{
2683 			// let the selection browse box determine an optimal size
2684 			Size aSelectionBoxSize = m_pSelectionBox->CalcOptimalSize( aPlaygroundSize );
2685 			nSplitPos = aPlaygroundSize.Height() - aSelectionBoxSize.Height() - m_aSplitter.GetSizePixel().Height();
2686 			// still an invalid size?
2687 			if ( nSplitPos == -1 || nSplitPos >= aPlaygroundSize.Height() )
2688 				nSplitPos = sal_Int32(aPlaygroundSize.Height()*0.6);
2689 
2690 			static_cast<OQueryController&>(getController()).setSplitPos(nSplitPos);
2691 		}
2692 
2693 		if ( !m_bInSplitHandler )
2694 		{	// the resize is triggered by something else than the split handler
2695 			// our main focus is to try to preserve the size of the selectionbrowse box
2696 			Size aSelBoxSize = m_pSelectionBox->GetSizePixel();
2697 			if ( aSelBoxSize.Height() )
2698 			{
2699 				// keep the size of the sel box constant
2700 				nSplitPos = aPlaygroundSize.Height() - m_aSplitter.GetSizePixel().Height() - aSelBoxSize.Height();
2701 
2702 				// and if the box is smaller than the optimal size, try to do something about it
2703 				Size aSelBoxOptSize = m_pSelectionBox->CalcOptimalSize( aPlaygroundSize );
2704 				if ( aSelBoxOptSize.Height() > aSelBoxSize.Height() )
2705 				{
2706 					nSplitPos = aPlaygroundSize.Height() - m_aSplitter.GetSizePixel().Height() - aSelBoxOptSize.Height();
2707 				}
2708 
2709 				static_cast< OQueryController& >(getController()).setSplitPos( nSplitPos );
2710 			}
2711 		}
2712 	}
2713 
2714 	// normalize the split pos
2715 	Point	aSplitPos		= Point( _rPlayground.Left(), nSplitPos );
2716 	Size	aSplitSize		= Size( _rPlayground.GetSize().Width(), m_aSplitter.GetSizePixel().Height() );
2717 
2718 	if( ( aSplitPos.Y() + aSplitSize.Height() ) > ( aPlaygroundSize.Height() ))
2719 		aSplitPos.Y() = aPlaygroundSize.Height() - aSplitSize.Height();
2720 
2721 	if( aSplitPos.Y() <= aPlaygroundPos.Y() )
2722 		aSplitPos.Y() = aPlaygroundPos.Y() + sal_Int32(aPlaygroundSize.Height() * 0.2);
2723 
2724 	// position the table
2725 	Size aTableViewSize(aPlaygroundSize.Width(), aSplitPos.Y() - aPlaygroundPos.Y());
2726 	m_pScrollWindow->SetPosSizePixel(aPlaygroundPos, aTableViewSize);
2727 
2728 	// position the selection browse box
2729 	Point aPos( aPlaygroundPos.X(), aSplitPos.Y() + aSplitSize.Height() );
2730 	m_pSelectionBox->SetPosSizePixel( aPos, Size( aPlaygroundSize.Width(), aPlaygroundSize.Height() - aSplitSize.Height() - aTableViewSize.Height() ));
2731 
2732 	// set the size of the splitter
2733 	m_aSplitter.SetPosSizePixel( aSplitPos, aSplitSize );
2734 	m_aSplitter.SetDragRectPixel( _rPlayground );
2735 
2736 	// just for completeness: there is no space left, we occupied it all ...
2737 	_rPlayground.SetPos( _rPlayground.BottomRight() );
2738 	_rPlayground.SetSize( Size( 0, 0 ) );
2739 }
2740 // -----------------------------------------------------------------------------
2741 void OQueryDesignView::setReadOnly(sal_Bool _bReadOnly)
2742 {
2743 	m_pSelectionBox->SetReadOnly(_bReadOnly);
2744 }
2745 // -----------------------------------------------------------------------------
2746 void OQueryDesignView::clear()
2747 {
2748 	m_pSelectionBox->ClearAll(); // clear the whole selection
2749 	m_pTableView->ClearAll();
2750 }
2751 // -----------------------------------------------------------------------------
2752 void OQueryDesignView::setStatement(const ::rtl::OUString& /*_rsStatement*/)
2753 {
2754 }
2755 // -----------------------------------------------------------------------------
2756 void OQueryDesignView::copy()
2757 {
2758 	if( m_eChildFocus == SELECTION)
2759 		m_pSelectionBox->copy();
2760 }
2761 // -----------------------------------------------------------------------------
2762 sal_Bool OQueryDesignView::isCutAllowed()
2763 {
2764 	sal_Bool bAllowed = sal_False;
2765 	if ( SELECTION == m_eChildFocus )
2766 		bAllowed = m_pSelectionBox->isCutAllowed();
2767 	return bAllowed;
2768 }
2769 // -----------------------------------------------------------------------------
2770 sal_Bool OQueryDesignView::isPasteAllowed()
2771 {
2772 	sal_Bool bAllowed = sal_False;
2773 	if ( SELECTION == m_eChildFocus )
2774 		bAllowed = m_pSelectionBox->isPasteAllowed();
2775 	return bAllowed;
2776 }
2777 // -----------------------------------------------------------------------------
2778 sal_Bool OQueryDesignView::isCopyAllowed()
2779 {
2780 	sal_Bool bAllowed = sal_False;
2781 	if ( SELECTION == m_eChildFocus )
2782 		bAllowed = m_pSelectionBox->isCopyAllowed();
2783 	return bAllowed;
2784 }
2785 // -----------------------------------------------------------------------------
2786 void OQueryDesignView::stopTimer()
2787 {
2788 	m_pSelectionBox->stopTimer();
2789 }
2790 // -----------------------------------------------------------------------------
2791 void OQueryDesignView::startTimer()
2792 {
2793 	m_pSelectionBox->startTimer();
2794 }
2795 // -----------------------------------------------------------------------------
2796 void OQueryDesignView::cut()
2797 {
2798 	if( m_eChildFocus == SELECTION)
2799 	{
2800 		m_pSelectionBox->cut();
2801 		static_cast<OQueryController&>(getController()).setModified(sal_True);
2802 	}
2803 }
2804 // -----------------------------------------------------------------------------
2805 void OQueryDesignView::paste()
2806 {
2807 	if( m_eChildFocus == SELECTION)
2808 	{
2809 		m_pSelectionBox->paste();
2810 		static_cast<OQueryController&>(getController()).setModified(sal_True);
2811 	}
2812 }
2813 // -----------------------------------------------------------------------------
2814 void OQueryDesignView::TableDeleted(const ::rtl::OUString& rAliasName)
2815 {
2816 	// Nachricht, dass Tabelle aus dem Fenster gel"oscht wurde
2817 	DeleteFields(rAliasName);
2818 	static_cast<OQueryController&>(getController()).InvalidateFeature(ID_BROWSER_ADDTABLE);	// view nochmal bescheid sagen
2819 }
2820 //------------------------------------------------------------------------------
2821 void OQueryDesignView::DeleteFields( const ::rtl::OUString& rAliasName )
2822 {
2823 	m_pSelectionBox->DeleteFields( rAliasName );
2824 }
2825 // -----------------------------------------------------------------------------
2826 bool OQueryDesignView::HasFieldByAliasName(const ::rtl::OUString& rFieldName, OTableFieldDescRef& rInfo)  const
2827 {
2828     return m_pSelectionBox->HasFieldByAliasName( rFieldName, rInfo);
2829 }
2830 // -----------------------------------------------------------------------------
2831 SqlParseError OQueryDesignView::InsertField( const OTableFieldDescRef& rInfo, sal_Bool bVis, sal_Bool bActivate)
2832 {
2833 	return m_pSelectionBox->InsertField( rInfo, BROWSER_INVALIDID,bVis, bActivate ).isValid() ? eOk : eTooManyColumns;
2834 }
2835 // -----------------------------------------------------------------------------
2836 sal_Int32 OQueryDesignView::getColWidth(sal_uInt16 _nColPos) const
2837 {
2838     static sal_Int32 s_nDefaultWidth = GetTextWidth(String(RTL_CONSTASCII_USTRINGPARAM("0"))) * 15;
2839     sal_Int32 nWidth = static_cast<OQueryController&>(getController()).getColWidth(_nColPos);
2840     if ( !nWidth )
2841         nWidth = s_nDefaultWidth;
2842 	return nWidth;
2843 }
2844 //------------------------------------------------------------------------------
2845 void OQueryDesignView::fillValidFields(const ::rtl::OUString& sAliasName, ComboBox* pFieldList)
2846 {
2847 	DBG_ASSERT(pFieldList != NULL, "OQueryDesignView::FillValidFields : What the hell do you think I can do with a NULL-ptr ? This will crash !");
2848 	pFieldList->Clear();
2849 
2850 	sal_Bool bAllTables = sAliasName.getLength() == 0;
2851 
2852 	OJoinTableView::OTableWindowMap* pTabWins = m_pTableView->GetTabWinMap();
2853 	::rtl::OUString strCurrentPrefix;
2854 	::std::vector< ::rtl::OUString> aFields;
2855 	OJoinTableView::OTableWindowMap::iterator aIter = pTabWins->begin();
2856     OJoinTableView::OTableWindowMap::iterator aEnd  = pTabWins->end();
2857     for(;aIter != aEnd;++aIter)
2858 	{
2859 		OQueryTableWindow* pCurrentWin = static_cast<OQueryTableWindow*>(aIter->second);
2860 		if (bAllTables || (pCurrentWin->GetAliasName() == sAliasName))
2861 		{
2862 			strCurrentPrefix = pCurrentWin->GetAliasName();
2863 			strCurrentPrefix += ::rtl::OUString('.');
2864 
2865 			pCurrentWin->EnumValidFields(aFields);
2866 
2867 			::std::vector< ::rtl::OUString>::iterator aStrIter = aFields.begin();
2868             ::std::vector< ::rtl::OUString>::iterator aStrEnd = aFields.end();
2869 			for(;aStrIter != aStrEnd;++aStrIter)
2870 			{
2871 				if (bAllTables || aStrIter->toChar() == '*')
2872 					pFieldList->InsertEntry(::rtl::OUString(strCurrentPrefix) += *aStrIter);
2873 				else
2874 					pFieldList->InsertEntry(*aStrIter);
2875 			}
2876 
2877 			if (!bAllTables)
2878 				// das heisst, dass ich in diesen Block kam, weil der Tabellenname genau der gesuchte war, also bin ich fertig
2879 				// (dadurch verhindere ich auch das doppelte Einfuegen von Feldern, wenn eine Tabelle mehrmals als TabWin vorkommt)
2880 				break;
2881 		}
2882 	}
2883 }
2884 // -----------------------------------------------------------------------------
2885 long OQueryDesignView::PreNotify(NotifyEvent& rNEvt)
2886 {
2887 	switch (rNEvt.GetType())
2888 	{
2889 		case EVENT_GETFOCUS:
2890 #if OSL_DEBUG_LEVEL > 0
2891 			{
2892 				Window* pFocus = Application::GetFocusWindow();
2893                 (void)pFocus;
2894 			}
2895 #endif
2896 
2897 			if ( m_pSelectionBox && m_pSelectionBox->HasChildPathFocus() )
2898 				m_eChildFocus = SELECTION;
2899 			else
2900 				m_eChildFocus = TABLEVIEW;
2901 			break;
2902 	}
2903 
2904 	return OQueryView::PreNotify(rNEvt);
2905 }
2906 //------------------------------------------------------------------------------
2907 
2908 
2909 // -----------------------------------------------------------------------------
2910 // check if the statement is correct when not returning false
2911 sal_Bool OQueryDesignView::checkStatement()
2912 {
2913 	sal_Bool bRet = sal_True;
2914 	if ( m_pSelectionBox )
2915 		bRet = m_pSelectionBox->Save(); // a error occurred so we return no
2916 	return bRet;
2917 }
2918 //-------------------------------------------------------------------------------
2919 ::rtl::OUString OQueryDesignView::getStatement()
2920 {
2921 	OQueryController& rController = static_cast<OQueryController&>(getController());
2922 	m_rController.clearError();
2923 	// used for fields which aren't any longer in the statement
2924 	OTableFields& rUnUsedFields = rController.getUnUsedFields();
2925 	OTableFields().swap( rUnUsedFields );
2926 
2927 	// create the select columns
2928 	sal_uInt32 nFieldcount = 0;
2929 	OTableFields& rFieldList = rController.getTableFieldDesc();
2930 	OTableFields::iterator aIter = rFieldList.begin();
2931     OTableFields::iterator aEnd = rFieldList.end();
2932 	for(;aIter != aEnd;++aIter)
2933 	{
2934 		OTableFieldDescRef pEntryField = *aIter;
2935 		if ( pEntryField->GetField().getLength() && pEntryField->IsVisible() )
2936 			++nFieldcount;
2937 		else if (pEntryField->GetField().getLength()			&&
2938 				!pEntryField->HasCriteria()					&&
2939 				pEntryField->isNoneFunction()				&&
2940 				pEntryField->GetOrderDir() == ORDER_NONE	&&
2941                 !pEntryField->IsGroupBy()                   &&
2942 				!pEntryField->GetFunction().getLength() )
2943 			rUnUsedFields.push_back(pEntryField);
2944 	}
2945 	if ( !nFieldcount )	// keine Felder sichtbar also zur"uck
2946 	{
2947 		rUnUsedFields = rFieldList;
2948 		return ::rtl::OUString();
2949 	}
2950 
2951 	OQueryTableView::OTableWindowMap* pTabList   = m_pTableView->GetTabWinMap();
2952 	sal_uInt32 nTabcount		= pTabList->size();
2953 
2954 	::rtl::OUString	aFieldListStr(GenerateSelectList(this,rFieldList,nTabcount>1));
2955 	if( !aFieldListStr.getLength() )
2956 		return ::rtl::OUString();
2957 	// Ausnahmebehandlung, wenn keine Felder angegeben worden sind
2958 	// Dann darf die Tabpage nicht gewechselt werden
2959 	// Im TabBarSelectHdl wird der SQL-::rtl::OUString auf STATEMENT_NOFIELDS abgefragt
2960 	// und eine Errormeldung erzeugt
2961 	// ----------------- Tabellenliste aufbauen ----------------------
2962 
2963 	const ::std::vector<OTableConnection*>*	pConnList = m_pTableView->getTableConnections();
2964 	Reference< XConnection> xConnection = rController.getConnection();
2965 	::rtl::OUString aTableListStr(GenerateFromClause(xConnection,pTabList,pConnList));
2966 	DBG_ASSERT(aTableListStr.getLength(), "OQueryDesignView::getStatement() : unerwartet : habe Felder, aber keine Tabellen !");
2967 	// wenn es Felder gibt, koennen die nur durch Einfuegen aus einer schon existenten Tabelle entstanden sein; wenn andererseits
2968 	// eine Tabelle geloescht wird, verschwinden auch die zugehoerigen Felder -> ergo KANN es das nicht geben, dass Felder
2969 	// existieren, aber keine Tabellen (und aFieldListStr hat schon eine Laenge, das stelle ich oben sicher)
2970 	::rtl::OUStringBuffer aHavingStr,aCriteriaListStr;
2971 	// ----------------- Kriterien aufbauen ----------------------
2972 	if (!GenerateCriterias(this,aCriteriaListStr,aHavingStr,rFieldList, nTabcount > 1))
2973 		return ::rtl::OUString();
2974 
2975 	::rtl::OUString aJoinCrit;
2976 	GenerateInnerJoinCriterias(xConnection,aJoinCrit,pConnList);
2977 	if(aJoinCrit.getLength())
2978 	{
2979 		::rtl::OUString aTmp = ::rtl::OUString::createFromAscii("( ");
2980 		aTmp += aJoinCrit;
2981 		aTmp += ::rtl::OUString::createFromAscii(" )");
2982 		if(aCriteriaListStr.getLength())
2983 		{
2984 			aTmp += C_AND;
2985 			aTmp += aCriteriaListStr.makeStringAndClear();
2986 		}
2987 		aCriteriaListStr = aTmp;
2988 	}
2989 	// ----------------- Statement aufbauen ----------------------
2990 	::rtl::OUStringBuffer aSqlCmd(::rtl::OUString::createFromAscii("SELECT "));
2991 	if(static_cast<OQueryController&>(getController()).isDistinct())
2992 		aSqlCmd.append(::rtl::OUString::createFromAscii(" DISTINCT "));
2993 	aSqlCmd.append(aFieldListStr);
2994 	aSqlCmd.append(::rtl::OUString::createFromAscii(" FROM "));
2995 	aSqlCmd.append(aTableListStr);
2996 
2997 	if (aCriteriaListStr.getLength())
2998 	{
2999 		aSqlCmd.append(::rtl::OUString::createFromAscii(" WHERE "));
3000 		aSqlCmd.append(aCriteriaListStr.makeStringAndClear());
3001 	}
3002 	// ----------------- GroupBy aufbauen und Anh"angen ------------
3003 	Reference<XDatabaseMetaData> xMeta;
3004 	if ( xConnection.is() )
3005 		xMeta = xConnection->getMetaData();
3006 	sal_Bool bUseAlias = nTabcount > 1;
3007 	if ( xMeta.is() )
3008 		bUseAlias = bUseAlias || !xMeta->supportsGroupByUnrelated();
3009 
3010 	aSqlCmd.append(GenerateGroupBy(this,rFieldList,bUseAlias));
3011 	// ----------------- having Anh"angen ------------
3012 	if(aHavingStr.getLength())
3013 	{
3014 		aSqlCmd.append(::rtl::OUString::createFromAscii(" HAVING "));
3015 		aSqlCmd.append(aHavingStr.makeStringAndClear());
3016 	}
3017 	// ----------------- Sortierung aufbauen und Anh"angen ------------
3018 	::rtl::OUString sOrder;
3019 	SqlParseError eErrorCode = eOk;
3020 	if ( (eErrorCode = GenerateOrder(this,rFieldList,nTabcount > 1,sOrder)) == eOk)
3021 		aSqlCmd.append(sOrder);
3022 	else
3023 	{
3024 		if ( !m_rController.hasError() )
3025             m_rController.appendError( getParseErrorMessage( eErrorCode ) );
3026 
3027         m_rController.displayError();
3028 	}
3029 
3030     ::rtl::OUString sSQL = aSqlCmd.makeStringAndClear();
3031     if ( xConnection.is() )
3032     {
3033         ::connectivity::OSQLParser& rParser( rController.getParser() );
3034         ::rtl::OUString sErrorMessage;
3035         ::std::auto_ptr<OSQLParseNode> pParseNode( rParser.parseTree( sErrorMessage, sSQL, sal_True ) );
3036         if ( pParseNode.get() )
3037         {
3038             OSQLParseNode* pNode = pParseNode->getChild(3)->getChild(1);
3039             if ( pNode->count() > 1 )
3040             {
3041 		        ::connectivity::OSQLParseNode * pCondition = pNode->getChild(1);
3042 		        if ( pCondition ) // no where clause
3043                 {
3044                     OSQLParseNode::compress(pCondition);
3045                     ::rtl::OUString sTemp;
3046                     pParseNode->parseNodeToStr(sTemp,xConnection);
3047                     sSQL = sTemp;
3048                 }
3049             }
3050         }
3051     }
3052 	return sSQL;
3053 }
3054 // -----------------------------------------------------------------------------
3055 // -----------------------------------------------------------------------------
3056 void OQueryDesignView::setSlotEnabled(sal_Int32 _nSlotId,sal_Bool _bEnable)
3057 {
3058 	sal_uInt16 nRow;
3059 	switch (_nSlotId)
3060 	{
3061 		case SID_QUERY_VIEW_FUNCTIONS:
3062 			nRow = BROW_FUNCTION_ROW;
3063 			break;
3064 		case SID_QUERY_VIEW_TABLES:
3065 			nRow = BROW_TABLE_ROW;
3066 			break;
3067 		case SID_QUERY_VIEW_ALIASES:
3068 			nRow = BROW_COLUMNALIAS_ROW;
3069 			break;
3070 		default:
3071 			// ????????????
3072 			nRow = 0;
3073 			break;
3074 	}
3075 	m_pSelectionBox->SetRowVisible(nRow,_bEnable);
3076 	m_pSelectionBox->Invalidate();
3077 }
3078 // -----------------------------------------------------------------------------
3079 sal_Bool OQueryDesignView::isSlotEnabled(sal_Int32 _nSlotId)
3080 {
3081 	sal_uInt16 nRow;
3082 	switch (_nSlotId)
3083 	{
3084 		case SID_QUERY_VIEW_FUNCTIONS:
3085 			nRow = BROW_FUNCTION_ROW;
3086 			break;
3087 		case SID_QUERY_VIEW_TABLES:
3088 			nRow = BROW_TABLE_ROW;
3089 			break;
3090 		case SID_QUERY_VIEW_ALIASES:
3091 			nRow = BROW_COLUMNALIAS_ROW;
3092 			break;
3093 		default:
3094 			// ?????????
3095 			nRow = 0;
3096 			break;
3097 	}
3098 	return m_pSelectionBox->IsRowVisible(nRow);
3099 }
3100 // -----------------------------------------------------------------------------
3101 void OQueryDesignView::SaveUIConfig()
3102 {
3103 	OQueryController& rCtrl = static_cast<OQueryController&>(getController());
3104 	rCtrl.SaveTabWinsPosSize( m_pTableView->GetTabWinMap(), m_pScrollWindow->GetHScrollBar()->GetThumbPos(), m_pScrollWindow->GetVScrollBar()->GetThumbPos() );
3105 	//	rCtrl.SaveTabFieldsWidth( m_pSelectionBox );
3106 	rCtrl.setVisibleRows( m_pSelectionBox->GetNoneVisibleRows() );
3107     if ( m_aSplitter.GetSplitPosPixel() != 0 )
3108 	    rCtrl.setSplitPos( m_aSplitter.GetSplitPosPixel() );
3109 }
3110 // -----------------------------------------------------------------------------
3111 OSQLParseNode* OQueryDesignView::getPredicateTreeFromEntry(OTableFieldDescRef pEntry,
3112 														   const String& _sCriteria,
3113 														   ::rtl::OUString& _rsErrorMessage,
3114 														   Reference<XPropertySet>& _rxColumn) const
3115 {
3116 	OSL_ENSURE(pEntry.isValid(),"Entry is null!");
3117 	if(!pEntry.isValid())
3118 		return NULL;
3119 	Reference< XConnection> xConnection = static_cast<OQueryController&>(getController()).getConnection();
3120 	if(!xConnection.is())
3121 		return NULL;
3122 
3123 	::connectivity::OSQLParser& rParser( static_cast<OQueryController&>(getController()).getParser() );
3124 	OQueryTableWindow* pWin = static_cast<OQueryTableWindow*>(pEntry->GetTabWindow());
3125 
3126 	String sTest(_sCriteria);
3127 	// special handling for functions
3128 	if ( pEntry->GetFunctionType() & (FKT_OTHER | FKT_AGGREGATE | FKT_NUMERIC) )
3129 	{
3130 		// we have a function here so we have to distinguish the type of return value
3131 		String sFunction;
3132 		if ( pEntry->isNumericOrAggreateFunction() )
3133 			sFunction = pEntry->GetFunction();
3134 
3135         if ( !sFunction.Len() )
3136 			sFunction = pEntry->GetField();
3137 
3138 		if(sFunction.GetTokenCount('(') > 1)
3139 			sFunction = sFunction.GetToken(0,'('); // this should be the name of the function
3140 
3141 		sal_Int32 nType = ::connectivity::OSQLParser::getFunctionReturnType(sFunction,&rParser.getContext());
3142 		if ( nType == DataType::OTHER || (!sFunction.Len() && pEntry->isNumericOrAggreateFunction()) )
3143 		{
3144 			// first try the international version
3145 			::rtl::OUString sSql;
3146 			sSql += ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("SELECT * "));
3147 			sSql += ::rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(" FROM x WHERE "));
3148 			sSql += pEntry->GetField();
3149 			sSql += _sCriteria;
3150             ::std::auto_ptr<OSQLParseNode> pParseNode( rParser.parseTree( _rsErrorMessage, sSql, sal_True ) );
3151 			nType = DataType::DOUBLE;
3152 			if ( pParseNode.get() )
3153 			{
3154 				OSQLParseNode* pColumnRef = pParseNode->getByRule(OSQLParseNode::column_ref);
3155 				if ( pColumnRef )
3156 				{
3157 					OTableFieldDescRef aField = new OTableFieldDesc();
3158 					if ( eOk == FillDragInfo(this,pColumnRef,aField) )
3159 					{
3160 						nType = aField->GetDataType();
3161 					}
3162 				}
3163 			}
3164 		}
3165 
3166 		Reference<XDatabaseMetaData> xMeta = xConnection->getMetaData();
3167 		parse::OParseColumn* pColumn = new parse::OParseColumn(	pEntry->GetField(),
3168 																::rtl::OUString(),
3169 																::rtl::OUString(),
3170                                                                 ::rtl::OUString(),
3171 																ColumnValue::NULLABLE_UNKNOWN,
3172 																0,
3173 																0,
3174 																nType,
3175 																sal_False,
3176 																sal_False,
3177 																xMeta.is() && xMeta->supportsMixedCaseQuotedIdentifiers());
3178 		_rxColumn = pColumn;
3179 		pColumn->setFunction(sal_True);
3180 		pColumn->setRealName(pEntry->GetField());
3181 	}
3182 	else
3183 	{
3184 		if (pWin)
3185 		{
3186 			Reference<XNameAccess> xColumns = pWin->GetOriginalColumns();
3187 			if (xColumns.is() && xColumns->hasByName(pEntry->GetField()))
3188 				xColumns->getByName(pEntry->GetField()) >>= _rxColumn;
3189 		}
3190 	}
3191 
3192 	OSQLParseNode* pParseNode = rParser.predicateTree(	_rsErrorMessage,
3193 														sTest,
3194 														static_cast<OQueryController&>(getController()).getNumberFormatter(),
3195 														_rxColumn);
3196 	return pParseNode;
3197 }
3198 // -----------------------------------------------------------------------------
3199 void OQueryDesignView::GetFocus()
3200 {
3201 	OQueryView::GetFocus();
3202 	if ( m_pSelectionBox && !m_pSelectionBox->HasChildPathFocus() )
3203 	{
3204 		// first we have to deactivate the current cell to refill when necessary
3205 		m_pSelectionBox->DeactivateCell();
3206 		m_pSelectionBox->ActivateCell(m_pSelectionBox->GetCurRow(), m_pSelectionBox->GetCurColumnId());
3207 		m_pSelectionBox->GrabFocus();
3208 	}
3209 }
3210 // -----------------------------------------------------------------------------
3211 void OQueryDesignView::reset()
3212 {
3213 	m_pTableView->ClearAll();
3214 	m_pTableView->ReSync();
3215 }
3216 // -----------------------------------------------------------------------------
3217 void OQueryDesignView::setNoneVisbleRow(sal_Int32 _nRows)
3218 {
3219 	m_pSelectionBox->SetNoneVisbleRow(_nRows);
3220 }
3221 
3222 // -----------------------------------------------------------------------------
3223 void OQueryDesignView::initByFieldDescriptions( const Sequence< PropertyValue >& i_rFieldDescriptions )
3224 {
3225 	OQueryController& rController = static_cast< OQueryController& >( getController() );
3226 
3227 	m_pSelectionBox->PreFill();
3228 	m_pSelectionBox->SetReadOnly( rController.isReadOnly() );
3229 	m_pSelectionBox->Fill();
3230 
3231     for (   const PropertyValue* field = i_rFieldDescriptions.getConstArray();
3232             field != i_rFieldDescriptions.getConstArray() + i_rFieldDescriptions.getLength();
3233             ++field
3234         )
3235     {
3236         ::vos::ORef< OTableFieldDesc > pField( new OTableFieldDesc() );
3237         pField->Load( *field, true );
3238         InsertField( pField, sal_True, sal_False );
3239     }
3240 
3241     rController.ClearUndoManager();
3242 	m_pSelectionBox->Invalidate();
3243 }
3244 
3245 // -----------------------------------------------------------------------------
3246 bool OQueryDesignView::initByParseIterator( ::dbtools::SQLExceptionInfo* _pErrorInfo )
3247 {
3248 	SqlParseError eErrorCode = eNativeMode;
3249 	m_rController.clearError();
3250 
3251     try
3252     {
3253         eErrorCode = InitFromParseNodeImpl( this, m_pSelectionBox );
3254 
3255         if ( eErrorCode != eOk )
3256 	    {
3257 	        if ( !m_rController.hasError() )
3258                 m_rController.appendError( getParseErrorMessage( eErrorCode ) );
3259 
3260             if ( _pErrorInfo )
3261             {
3262                 *_pErrorInfo = m_rController.getError();
3263             }
3264             else
3265             {
3266                 m_rController.displayError();
3267             }
3268 	    }
3269     }
3270     catch ( const Exception& )
3271     {
3272         DBG_UNHANDLED_EXCEPTION();
3273     }
3274 	return eErrorCode == eOk;
3275 }
3276 //------------------------------------------------------------------------------
3277 void OQueryDesignView::fillFunctionInfo(  const ::connectivity::OSQLParseNode* pNode
3278                                         ,const ::rtl::OUString& sFunctionTerm
3279                                         ,OTableFieldDescRef& aInfo)
3280 {
3281     // get the type out of the function name
3282     OQueryController& rController = static_cast<OQueryController&>(getController());
3283 	sal_Int32 nDataType = DataType::DOUBLE;
3284 	::rtl::OUString sFieldName = sFunctionTerm;
3285 	OSQLParseNode* pFunctionName = pNode->getChild(0);
3286 	if ( !SQL_ISPUNCTUATION(pFunctionName,"{") )
3287 	{
3288 		if ( SQL_ISRULEOR2(pNode,length_exp,char_value_fct) )
3289 			pFunctionName = pFunctionName->getChild(0);
3290 
3291 		::rtl::OUString sFunctionName = pFunctionName->getTokenValue();
3292 		if ( !sFunctionName.getLength() )
3293 			sFunctionName = ::rtl::OStringToOUString(OSQLParser::TokenIDToStr(pFunctionName->getTokenID()),RTL_TEXTENCODING_UTF8);
3294 
3295 		nDataType = OSQLParser::getFunctionReturnType(
3296 							sFunctionName
3297 							,&rController.getParser().getContext());
3298 	}
3299 	aInfo->SetDataType(nDataType);
3300 	aInfo->SetFieldType(TAB_NORMAL_FIELD);
3301 	aInfo->SetField(sFieldName);
3302 	aInfo->SetTabWindow(NULL);
3303 }
3304 // -----------------------------------------------------------------------------
3305