xref: /AOO41X/main/dbaccess/source/ui/control/tabletree.cxx (revision 96de54900b79e13b861fbc62cbf36018b54e21b7)
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 
27 #ifndef _DBAUI_TABLETREE_HXX_
28 #include "tabletree.hxx"
29 #endif
30 #ifndef _DBAUI_TABLETREE_HRC_
31 #include "tabletree.hrc"
32 #endif
33 #ifndef DBACCESS_IMAGEPROVIDER_HXX
34 #include "imageprovider.hxx"
35 #endif
36 #ifndef _DBAUI_MODULE_DBU_HXX_
37 #include "moduledbu.hxx"
38 #endif
39 #ifndef _DBU_CONTROL_HRC_
40 #include "dbu_control.hrc"
41 #endif
42 #ifndef _SV_MENU_HXX
43 #include <vcl/menu.hxx>
44 #endif
45 #ifndef _CONNECTIVITY_DBTOOLS_HXX_
46 #include <connectivity/dbtools.hxx>
47 #endif
48 #ifndef _COMPHELPER_TYPES_HXX_
49 #include <comphelper/types.hxx>
50 #endif
51 #ifndef DBACCESS_SHARED_DBUSTRINGS_HRC
52 #include "dbustrings.hrc"
53 #endif
54 #ifndef _COM_SUN_STAR_SDB_APPLICATION_DATABASEOBJECT_HPP_
55 #include <com/sun/star/sdb/application/DatabaseObject.hpp>
56 #endif
57 #ifndef _COM_SUN_STAR_SDB_APPLICATION_DATABASEOBJECTFOLDER_HPP_
58 #include <com/sun/star/sdb/application/DatabaseObjectContainer.hpp>
59 #endif
60 #ifndef _COM_SUN_STAR_SDBC_XDRIVERACCESS_HPP_
61 #include <com/sun/star/sdbc/XDriverAccess.hpp>
62 #endif
63 #ifndef _COM_SUN_STAR_SDBCX_XDATADEFINITIONSUPPLIER_HPP_
64 #include <com/sun/star/sdbcx/XDataDefinitionSupplier.hpp>
65 #endif
66 #ifndef _COM_SUN_STAR_SDBCX_XVIEWSSUPPLIER_HPP_
67 #include <com/sun/star/sdbcx/XViewsSupplier.hpp>
68 #endif
69 #ifndef _COM_SUN_STAR_SDBCX_XTABLESSUPPLIER_HPP_
70 #include <com/sun/star/sdbcx/XTablesSupplier.hpp>
71 #endif
72 #ifndef _COM_SUN_STAR_SDB_SQLCONTEXT_HPP_
73 #include <com/sun/star/sdb/SQLContext.hpp>
74 #endif
75 #ifndef _COM_SUN_STAR_SDBC_XROW_HPP_
76 #include <com/sun/star/sdbc/XRow.hpp>
77 #endif
78 #ifndef _COM_SUN_STAR_BEANS_XPROPERTYSET_HPP_
79 #include <com/sun/star/beans/XPropertySet.hpp>
80 #endif
81 #ifndef _DBAUI_COMMON_TYPES_HXX_
82 #include "commontypes.hxx"
83 #endif
84 #ifndef _DBAUI_LISTVIEWITEMS_HXX_
85 #include "listviewitems.hxx"
86 #endif
87 #ifndef TOOLS_DIAGNOSE_EX_H
88 #include <tools/diagnose_ex.h>
89 #endif
90 #ifndef _RTL_USTRBUF_HXX_
91 #include <rtl/ustrbuf.hxx>
92 #endif
93 #include <connectivity/dbmetadata.hxx>
94 
95 #include <algorithm>
96 
97 //.........................................................................
98 namespace dbaui
99 {
100 //.........................................................................
101 
102 using namespace ::com::sun::star::uno;
103 using namespace ::com::sun::star::sdb;
104 using namespace ::com::sun::star::lang;
105 using namespace ::com::sun::star::sdbc;
106 using namespace ::com::sun::star::sdbcx;
107 using namespace ::com::sun::star::beans;
108 using namespace ::com::sun::star::container;
109 using namespace ::com::sun::star::sdb::application;
110 
111 using namespace ::dbtools;
112 using namespace ::comphelper;
113 
114 namespace DatabaseObject = ::com::sun::star::sdb::application::DatabaseObject;
115 namespace DatabaseObjectContainer = ::com::sun::star::sdb::application::DatabaseObjectContainer;
116 
117 //========================================================================
118 //= OTableTreeListBox
119 //========================================================================
OTableTreeListBox(Window * pParent,const Reference<XMultiServiceFactory> & _rxORB,WinBits nWinStyle,sal_Bool _bVirtualRoot)120 OTableTreeListBox::OTableTreeListBox( Window* pParent, const Reference< XMultiServiceFactory >& _rxORB, WinBits nWinStyle,sal_Bool _bVirtualRoot )
121     :OMarkableTreeListBox(pParent,_rxORB,nWinStyle)
122     ,m_pImageProvider( new ImageProvider )
123     ,m_bVirtualRoot(_bVirtualRoot)
124     ,m_bNoEmptyFolders( false )
125 {
126     implSetDefaultImages();
127 }
128 //------------------------------------------------------------------------
OTableTreeListBox(Window * pParent,const Reference<XMultiServiceFactory> & _rxORB,const ResId & rResId,sal_Bool _bVirtualRoot)129 OTableTreeListBox::OTableTreeListBox( Window* pParent, const Reference< XMultiServiceFactory >& _rxORB, const ResId& rResId ,sal_Bool _bVirtualRoot)
130     :OMarkableTreeListBox(pParent,_rxORB,rResId)
131     ,m_pImageProvider( new ImageProvider )
132     ,m_bVirtualRoot(_bVirtualRoot)
133     ,m_bNoEmptyFolders( false )
134 {
135     implSetDefaultImages();
136 }
137 
138 // -----------------------------------------------------------------------------
~OTableTreeListBox()139 OTableTreeListBox::~OTableTreeListBox()
140 {
141 }
142 
143 // -----------------------------------------------------------------------------
implSetDefaultImages()144 void OTableTreeListBox::implSetDefaultImages()
145 {
146     ImageProvider aImageProvider;
147     SetDefaultExpandedEntryBmp( aImageProvider.getFolderImage( DatabaseObject::TABLE, false ), BMP_COLOR_NORMAL );
148     SetDefaultExpandedEntryBmp( aImageProvider.getFolderImage( DatabaseObject::TABLE, true ), BMP_COLOR_HIGHCONTRAST );
149     SetDefaultCollapsedEntryBmp( aImageProvider.getFolderImage( DatabaseObject::TABLE, false ), BMP_COLOR_NORMAL );
150     SetDefaultCollapsedEntryBmp( aImageProvider.getFolderImage( DatabaseObject::TABLE, true ), BMP_COLOR_HIGHCONTRAST );
151 }
152 
153 // -----------------------------------------------------------------------------
isFolderEntry(const SvLBoxEntry * _pEntry) const154 bool  OTableTreeListBox::isFolderEntry( const SvLBoxEntry* _pEntry ) const
155 {
156     sal_Int32 nEntryType = reinterpret_cast< sal_IntPtr >( _pEntry->GetUserData() );
157     if  (   ( nEntryType == DatabaseObjectContainer::TABLES )
158         ||  ( nEntryType == DatabaseObjectContainer::CATALOG )
159         ||  ( nEntryType == DatabaseObjectContainer::SCHEMA )
160         )
161         return true;
162     return false;
163 }
164 
165 // -----------------------------------------------------------------------------
notifyHiContrastChanged()166 void OTableTreeListBox::notifyHiContrastChanged()
167 {
168     implSetDefaultImages();
169 
170     SvLBoxEntry* pEntryLoop = First();
171     while (pEntryLoop)
172     {
173         sal_uInt16 nCount = pEntryLoop->ItemCount();
174         for (sal_uInt16 i=0;i<nCount;++i)
175         {
176             SvLBoxItem* pItem = pEntryLoop->GetItem(i);
177             if ( pItem && pItem->IsA() == SV_ITEM_ID_LBOXCONTEXTBMP)
178             {
179                 SvLBoxContextBmp* pContextBitmapItem = static_cast< SvLBoxContextBmp* >( pItem );
180 
181                 Image aImage, aImageHC;
182                 if ( isFolderEntry( pEntryLoop ) )
183                 {
184                     aImage = m_pImageProvider->getFolderImage( DatabaseObject::TABLE, false );
185                     aImageHC = m_pImageProvider->getFolderImage( DatabaseObject::TABLE, true );
186                 }
187                 else
188                 {
189                     String sCompleteName( getQualifiedTableName( pEntryLoop ) );
190                     m_pImageProvider->getImages( sCompleteName, DatabaseObject::TABLE, aImage, aImageHC );
191                 }
192 
193                 pContextBitmapItem->SetBitmap1( aImage, BMP_COLOR_NORMAL );
194                 pContextBitmapItem->SetBitmap2( aImage, BMP_COLOR_NORMAL );
195                 pContextBitmapItem->SetBitmap1( aImageHC, BMP_COLOR_HIGHCONTRAST );
196                 pContextBitmapItem->SetBitmap2( aImageHC, BMP_COLOR_HIGHCONTRAST );
197                 // TODO: Now that we give both images to the entry item, it is not necessary anymore
198                 // to do this anytime HC changes - the tree control will do this itself now.
199                 // We would only need to properly initialize newly inserted entries.
200                 break;
201             }
202         }
203         pEntryLoop = Next(pEntryLoop);
204     }
205 }
206 
207 //------------------------------------------------------------------------
implOnNewConnection(const Reference<XConnection> & _rxConnection)208 void OTableTreeListBox::implOnNewConnection( const Reference< XConnection >& _rxConnection )
209 {
210     m_xConnection = _rxConnection;
211     m_pImageProvider.reset( new ImageProvider( m_xConnection  ) );
212 }
213 
214 //------------------------------------------------------------------------
UpdateTableList(const Reference<XConnection> & _rxConnection)215 void OTableTreeListBox::UpdateTableList( const Reference< XConnection >& _rxConnection ) throw(SQLException)
216 {
217     Sequence< ::rtl::OUString > sTables, sViews;
218 
219     String sCurrentActionError;
220     try
221     {
222         Reference< XTablesSupplier > xTableSupp( _rxConnection, UNO_QUERY_THROW );
223         sCurrentActionError = String(ModuleRes(STR_NOTABLEINFO));
224 
225         Reference< XNameAccess > xTables,xViews;
226 
227         Reference< XViewsSupplier > xViewSupp( _rxConnection, UNO_QUERY );
228         if ( xViewSupp.is() )
229         {
230             xViews = xViewSupp->getViews();
231             if (xViews.is())
232                 sViews = xViews->getElementNames();
233         }
234 
235         xTables = xTableSupp->getTables();
236         if (xTables.is())
237             sTables = xTables->getElementNames();
238     }
239     catch(RuntimeException&)
240     {
241         DBG_ERROR("OTableTreeListBox::UpdateTableList : caught an RuntimeException!");
242     }
243     catch ( const SQLException& )
244     {
245         throw;
246     }
247     catch(Exception&)
248     {
249         // a non-SQLException exception occured ... simply throw an SQLException
250         SQLException aInfo;
251         aInfo.Message = sCurrentActionError;
252         throw aInfo;
253     }
254 
255     UpdateTableList( _rxConnection, sTables, sViews );
256 }
257 // -----------------------------------------------------------------------------
258 namespace
259 {
260     struct OViewSetter : public ::std::unary_function< OTableTreeListBox::TNames::value_type, bool>
261     {
262         const Sequence< ::rtl::OUString> m_aViews;
263         ::comphelper::TStringMixEqualFunctor m_aEqualFunctor;
264 
OViewSetterdbaui::__anon603e057b0111::OViewSetter265         OViewSetter(const Sequence< ::rtl::OUString>& _rViews,sal_Bool _bCase) : m_aViews(_rViews),m_aEqualFunctor(_bCase){}
operator ()dbaui::__anon603e057b0111::OViewSetter266         OTableTreeListBox::TNames::value_type operator() (const ::rtl::OUString& lhs)
267         {
268             OTableTreeListBox::TNames::value_type aRet;
269             aRet.first = lhs;
270             const ::rtl::OUString* pIter = m_aViews.getConstArray();
271             const ::rtl::OUString* pEnd = m_aViews.getConstArray() + m_aViews.getLength();
272             aRet.second = (::std::find_if(pIter,pEnd,::std::bind2nd(m_aEqualFunctor,lhs)) != pEnd);
273 
274             return aRet;
275         }
276     };
277 
278 }
279 // -----------------------------------------------------------------------------
UpdateTableList(const Reference<XConnection> & _rxConnection,const Sequence<::rtl::OUString> & _rTables,const Sequence<::rtl::OUString> & _rViews)280 void OTableTreeListBox::UpdateTableList(
281                 const Reference< XConnection >& _rxConnection,
282                 const Sequence< ::rtl::OUString>& _rTables,
283                 const Sequence< ::rtl::OUString>& _rViews
284             )
285 {
286     TNames aTables;
287     aTables.resize(_rTables.getLength());
288     const ::rtl::OUString* pIter = _rTables.getConstArray();
289     const ::rtl::OUString* pEnd = _rTables.getConstArray() + _rTables.getLength();
290     try
291     {
292         Reference< XDatabaseMetaData > xMeta( _rxConnection->getMetaData(), UNO_QUERY_THROW );
293         ::std::transform( pIter, pEnd,
294             aTables.begin(), OViewSetter( _rViews, xMeta->supportsMixedCaseQuotedIdentifiers() ) );
295     }
296     catch(Exception&)
297     {
298         DBG_UNHANDLED_EXCEPTION();
299     }
300     UpdateTableList( _rxConnection, aTables );
301 }
302 
303 //------------------------------------------------------------------------
304 namespace
305 {
lcl_getMetaDataStrings_throw(const Reference<XResultSet> & _rxMetaDataResult,sal_Int32 _nColumnIndex)306     ::std::vector< ::rtl::OUString > lcl_getMetaDataStrings_throw( const Reference< XResultSet >& _rxMetaDataResult, sal_Int32 _nColumnIndex )
307     {
308         ::std::vector< ::rtl::OUString > aStrings;
309         Reference< XRow > xRow( _rxMetaDataResult, UNO_QUERY_THROW );
310         while ( _rxMetaDataResult->next() )
311             aStrings.push_back( xRow->getString( _nColumnIndex ) );
312         return aStrings;
313     }
314 
lcl_shouldDisplayEmptySchemasAndCatalogs(const Reference<XConnection> & _rxConnection)315     bool lcl_shouldDisplayEmptySchemasAndCatalogs( const Reference< XConnection >& _rxConnection )
316     {
317         ::dbtools::DatabaseMetaData aMetaData( _rxConnection );
318         return aMetaData.displayEmptyTableFolders();
319     }
320 }
321 
322 //------------------------------------------------------------------------
UpdateTableList(const Reference<XConnection> & _rxConnection,const TNames & _rTables)323 void OTableTreeListBox::UpdateTableList( const Reference< XConnection >& _rxConnection, const TNames& _rTables )
324 {
325     implOnNewConnection( _rxConnection );
326 
327     // throw away all the old stuff
328     Clear();
329 
330     try
331     {
332         // the root entry saying "all objects"
333         SvLBoxEntry* pAllObjects = NULL;
334         if (haveVirtualRoot())
335         {
336             String sRootEntryText;
337             TNames::const_iterator aViews = ::std::find_if(_rTables.begin(),_rTables.end(),
338             ::std::compose1(::std::bind2nd(::std::equal_to<sal_Bool>(),sal_False),::std::select2nd<TNames::value_type>()));
339             TNames::const_iterator aTables = ::std::find_if(_rTables.begin(),_rTables.end(),
340             ::std::compose1(::std::bind2nd(::std::equal_to<sal_Bool>(),sal_True),::std::select2nd<TNames::value_type>()));
341 
342             if ( aViews == _rTables.end() )
343                 sRootEntryText  = String(ModuleRes(STR_ALL_TABLES));
344             else if ( aTables == _rTables.end() )
345                 sRootEntryText  = String(ModuleRes(STR_ALL_VIEWS));
346             else
347                 sRootEntryText  = String(ModuleRes(STR_ALL_TABLES_AND_VIEWS));
348             pAllObjects = InsertEntry( sRootEntryText, NULL, sal_False, LIST_APPEND, reinterpret_cast< void* >( DatabaseObjectContainer::TABLES ) );
349         }
350 
351         if ( _rTables.empty() )
352             // nothing to do (besides inserting the root entry)
353             return;
354 
355         // get the table/view names
356         TNames::const_iterator aIter = _rTables.begin();
357         TNames::const_iterator aEnd = _rTables.end();
358 
359         Reference< XDatabaseMetaData > xMeta( _rxConnection->getMetaData(), UNO_QUERY_THROW );
360         for ( ; aIter != aEnd; ++aIter )
361         {
362             // add the entry
363             implAddEntry(
364                 xMeta,
365                 aIter->first,
366                 sal_False
367             );
368         }
369 
370         if ( !m_bNoEmptyFolders && lcl_shouldDisplayEmptySchemasAndCatalogs( _rxConnection ) )
371         {
372             sal_Bool bSupportsCatalogs = xMeta->supportsCatalogsInDataManipulation();
373             sal_Bool bSupportsSchemas = xMeta->supportsSchemasInDataManipulation();
374 
375             if ( bSupportsCatalogs || bSupportsSchemas )
376             {
377                 // we display empty catalogs if the DB supports catalogs, and they're noted at the beginning of a
378                 // composed name. Otherwise, we display empty schematas. (also see the tree structure explained in
379                 // implAddEntry)
380                 bool bCatalogs = bSupportsCatalogs && xMeta->isCatalogAtStart();
381 
382                 ::std::vector< ::rtl::OUString > aFolderNames( lcl_getMetaDataStrings_throw(
383                     bCatalogs ? xMeta->getCatalogs() : xMeta->getSchemas(), 1 ) );
384                 sal_Int32 nFolderType = bCatalogs ? DatabaseObjectContainer::CATALOG : DatabaseObjectContainer::SCHEMA;
385 
386                 SvLBoxEntry* pRootEntry = getAllObjectsEntry();
387                 for (   ::std::vector< ::rtl::OUString >::const_iterator folder = aFolderNames.begin();
388                         folder != aFolderNames.end();
389                         ++folder
390                     )
391                 {
392                     SvLBoxEntry* pFolder = GetEntryPosByName( *folder, pRootEntry );
393                     if ( !pFolder )
394                         pFolder = InsertEntry( *folder, pRootEntry, sal_False, LIST_APPEND, reinterpret_cast< void* >( nFolderType ) );
395                 }
396             }
397         }
398     }
399     catch ( const Exception& )
400     {
401         DBG_UNHANDLED_EXCEPTION();
402     }
403 }
404 //------------------------------------------------------------------------
isWildcardChecked(SvLBoxEntry * _pEntry) const405 sal_Bool OTableTreeListBox::isWildcardChecked(SvLBoxEntry* _pEntry) const
406 {
407     if (_pEntry)
408     {
409         OBoldListboxString* pTextItem = static_cast<OBoldListboxString*>(_pEntry->GetFirstItem(SV_ITEM_ID_BOLDLBSTRING));
410         if (pTextItem)
411             return pTextItem->isEmphasized();
412     }
413     return sal_False;
414 }
415 
416 //------------------------------------------------------------------------
checkWildcard(SvLBoxEntry * _pEntry)417 void OTableTreeListBox::checkWildcard(SvLBoxEntry* _pEntry)
418 {
419     SetCheckButtonState(_pEntry, SV_BUTTON_CHECKED);
420     checkedButton_noBroadcast(_pEntry);
421 }
422 
423 //------------------------------------------------------------------------
getAllObjectsEntry() const424 SvLBoxEntry* OTableTreeListBox::getAllObjectsEntry() const
425 {
426     return haveVirtualRoot() ? First() : NULL;
427 }
428 
429 //------------------------------------------------------------------------
checkedButton_noBroadcast(SvLBoxEntry * _pEntry)430 void OTableTreeListBox::checkedButton_noBroadcast(SvLBoxEntry* _pEntry)
431 {
432     OMarkableTreeListBox::checkedButton_noBroadcast(_pEntry);
433 
434     // if an entry has children, it makes a difference if the entry is checked because alls children are checked
435     // or if the user checked it explicitly.
436     // So we track explicit (un)checking
437 
438     SvButtonState eState = GetCheckButtonState(_pEntry);
439     DBG_ASSERT(SV_BUTTON_TRISTATE != eState, "OTableTreeListBox::CheckButtonHdl: user action which lead to TRISTATE?");
440     implEmphasize(_pEntry, SV_BUTTON_CHECKED == eState);
441 }
442 
443 //------------------------------------------------------------------------
implEmphasize(SvLBoxEntry * _pEntry,sal_Bool _bChecked,sal_Bool _bUpdateDescendants,sal_Bool _bUpdateAncestors)444 void OTableTreeListBox::implEmphasize(SvLBoxEntry* _pEntry, sal_Bool _bChecked, sal_Bool _bUpdateDescendants, sal_Bool _bUpdateAncestors)
445 {
446     DBG_ASSERT(_pEntry, "OTableTreeListBox::implEmphasize: invalid entry (NULL)!");
447 
448     // special emphasizing handling for the "all objects" entry
449     // 89709 - 16.07.2001 - frank.schoenheit@sun.com
450     sal_Bool bAllObjectsEntryAffected = haveVirtualRoot() && (getAllObjectsEntry() == _pEntry);
451     if  (   GetModel()->HasChilds(_pEntry)              // the entry has children
452         ||  bAllObjectsEntryAffected                    // or it is the "all objects" entry
453         )
454     {
455         OBoldListboxString* pTextItem = static_cast<OBoldListboxString*>(_pEntry->GetFirstItem(SV_ITEM_ID_BOLDLBSTRING));
456         if (pTextItem)
457             pTextItem->emphasize(_bChecked);
458 
459         if (bAllObjectsEntryAffected)
460             InvalidateEntry(_pEntry);
461     }
462 
463     if (_bUpdateDescendants)
464     {
465         // remove the mark for all children of the checked entry
466         SvLBoxEntry* pChildLoop = FirstChild(_pEntry);
467         while (pChildLoop)
468         {
469             if (GetModel()->HasChilds(pChildLoop))
470                 implEmphasize(pChildLoop, sal_False, sal_True, sal_False);
471             pChildLoop = NextSibling(pChildLoop);
472         }
473     }
474 
475     if (_bUpdateAncestors)
476     {
477         // remove the mark for all ancestors of the entry
478         if (GetModel()->HasParent(_pEntry))
479             implEmphasize(GetParent(_pEntry), sal_False, sal_False, sal_True);
480     }
481 }
482 
483 //------------------------------------------------------------------------
InitEntry(SvLBoxEntry * _pEntry,const XubString & _rString,const Image & _rCollapsedBitmap,const Image & _rExpandedBitmap,SvLBoxButtonKind _eButtonKind)484 void OTableTreeListBox::InitEntry(SvLBoxEntry* _pEntry, const XubString& _rString, const Image& _rCollapsedBitmap, const Image& _rExpandedBitmap, SvLBoxButtonKind _eButtonKind)
485 {
486     OMarkableTreeListBox::InitEntry(_pEntry, _rString, _rCollapsedBitmap, _rExpandedBitmap, _eButtonKind);
487 
488     // replace the text item with our own one
489     SvLBoxItem* pTextItem = _pEntry->GetFirstItem(SV_ITEM_ID_LBOXSTRING);
490     DBG_ASSERT(pTextItem, "OTableTreeListBox::InitEntry: no text item!?");
491     sal_uInt16 nTextPos = _pEntry->GetPos(pTextItem);
492     DBG_ASSERT(((sal_uInt16)-1) != nTextPos, "OTableTreeListBox::InitEntry: no text item pos!");
493 
494     _pEntry->ReplaceItem(new OBoldListboxString(_pEntry, 0, _rString), nTextPos);
495 }
496 
497 //------------------------------------------------------------------------
implAddEntry(const Reference<XDatabaseMetaData> & _rxMeta,const::rtl::OUString & _rTableName,sal_Bool _bCheckName)498 SvLBoxEntry* OTableTreeListBox::implAddEntry(
499         const Reference< XDatabaseMetaData >& _rxMeta,
500         const ::rtl::OUString& _rTableName,
501         sal_Bool _bCheckName
502     )
503 {
504     OSL_PRECOND( _rxMeta.is(), "OTableTreeListBox::implAddEntry: invalid meta data!" );
505     if ( !_rxMeta.is() )
506         return NULL;
507 
508     // split the complete name into it's components
509     ::rtl::OUString sCatalog, sSchema, sName;
510     qualifiedNameComponents( _rxMeta, _rTableName, sCatalog, sSchema, sName, ::dbtools::eInDataManipulation );
511 
512     SvLBoxEntry* pParentEntry = getAllObjectsEntry();
513 
514     // if the DB uses catalog at the start of identifiers, then our hierarchy is
515     //   catalog
516     //   +- schema
517     //      +- table
518     // else it is
519     //   schema
520     //   +- catalog
521     //      +- table
522     sal_Bool bCatalogAtStart = _rxMeta->isCatalogAtStart();
523     const ::rtl::OUString& rFirstName  = bCatalogAtStart ? sCatalog : sSchema;
524     const sal_Int32 nFirstFolderType   = bCatalogAtStart ? DatabaseObjectContainer::CATALOG : DatabaseObjectContainer::SCHEMA;
525     const ::rtl::OUString& rSecondName = bCatalogAtStart ? sSchema : sCatalog;
526     const sal_Int32 nSecondFolderType  = bCatalogAtStart ? DatabaseObjectContainer::SCHEMA : DatabaseObjectContainer::CATALOG;
527 
528     if ( rFirstName.getLength() )
529     {
530         SvLBoxEntry* pFolder = GetEntryPosByName( rFirstName, pParentEntry );
531         if ( !pFolder )
532             pFolder = InsertEntry( rFirstName, pParentEntry, sal_False, LIST_APPEND, reinterpret_cast< void* >( nFirstFolderType ) );
533         pParentEntry = pFolder;
534     }
535 
536     if ( rSecondName.getLength() )
537     {
538         SvLBoxEntry* pFolder = GetEntryPosByName( rSecondName, pParentEntry );
539         if ( !pFolder )
540             pFolder = InsertEntry( rSecondName, pParentEntry, sal_False, LIST_APPEND, reinterpret_cast< void* >( nSecondFolderType ) );
541         pParentEntry = pFolder;
542     }
543 
544     SvLBoxEntry* pRet = NULL;
545     if ( !_bCheckName || !GetEntryPosByName( sName, pParentEntry ) )
546     {
547         pRet = InsertEntry( sName, pParentEntry, sal_False, LIST_APPEND );
548 
549         Image aImage, aImageHC;
550         m_pImageProvider->getImages( _rTableName, DatabaseObject::TABLE, aImage, aImageHC );
551 
552         SetExpandedEntryBmp( pRet, aImage, BMP_COLOR_NORMAL );
553         SetCollapsedEntryBmp( pRet, aImage, BMP_COLOR_NORMAL );
554         SetExpandedEntryBmp( pRet, aImageHC, BMP_COLOR_HIGHCONTRAST );
555         SetCollapsedEntryBmp( pRet, aImageHC, BMP_COLOR_HIGHCONTRAST );
556     }
557     return pRet;
558 }
559 
560 //------------------------------------------------------------------------
describeObject(SvLBoxEntry * _pEntry)561 NamedDatabaseObject OTableTreeListBox::describeObject( SvLBoxEntry* _pEntry )
562 {
563     NamedDatabaseObject aObject;
564 
565     sal_Int32 nEntryType = reinterpret_cast< sal_IntPtr >( _pEntry->GetUserData() );
566 
567     if  ( nEntryType == DatabaseObjectContainer::TABLES )
568     {
569         aObject.Type = DatabaseObjectContainer::TABLES;
570     }
571     else if (   ( nEntryType == DatabaseObjectContainer::CATALOG )
572             ||  ( nEntryType == DatabaseObjectContainer::SCHEMA )
573             )
574     {
575         SvLBoxEntry* pParent = GetParent( _pEntry );
576         sal_Int32 nParentEntryType = pParent ? reinterpret_cast< sal_IntPtr >( pParent->GetUserData() ) : -1;
577 
578         ::rtl::OUStringBuffer buffer;
579         if  ( nEntryType == DatabaseObjectContainer::CATALOG )
580         {
581             if ( nParentEntryType == DatabaseObjectContainer::SCHEMA )
582             {
583                 buffer.append( GetEntryText( pParent ) );
584                 buffer.append( sal_Unicode( '.' ) );
585             }
586             buffer.append( GetEntryText( _pEntry ) );
587         }
588         else if ( nEntryType == DatabaseObjectContainer::SCHEMA )
589         {
590             if ( nParentEntryType == DatabaseObjectContainer::CATALOG )
591             {
592                 buffer.append( GetEntryText( pParent ) );
593                 buffer.append( sal_Unicode( '.' ) );
594             }
595             buffer.append( GetEntryText( _pEntry ) );
596         }
597     }
598     else
599     {
600         aObject.Type = DatabaseObject::TABLE;
601         aObject.Name = getQualifiedTableName( _pEntry );
602     }
603 
604     return aObject;
605 }
606 
607 //------------------------------------------------------------------------
addedTable(const::rtl::OUString & _rName)608 SvLBoxEntry* OTableTreeListBox::addedTable( const ::rtl::OUString& _rName )
609 {
610     try
611     {
612         Reference< XDatabaseMetaData > xMeta;
613         if ( impl_getAndAssertMetaData( xMeta ) )
614             return implAddEntry( xMeta, _rName );
615     }
616     catch( const Exception& )
617     {
618         DBG_UNHANDLED_EXCEPTION();
619     }
620     return NULL;
621 }
622 
623 //------------------------------------------------------------------------
impl_getAndAssertMetaData(Reference<XDatabaseMetaData> & _out_rMetaData) const624 bool OTableTreeListBox::impl_getAndAssertMetaData( Reference< XDatabaseMetaData >& _out_rMetaData ) const
625 {
626     if ( m_xConnection.is() )
627         _out_rMetaData = m_xConnection->getMetaData();
628     OSL_PRECOND( _out_rMetaData.is(), "OTableTreeListBox::impl_getAndAssertMetaData: invalid current connection!" );
629     return _out_rMetaData.is();
630 }
631 
632 //------------------------------------------------------------------------
getQualifiedTableName(SvLBoxEntry * _pEntry) const633 String OTableTreeListBox::getQualifiedTableName( SvLBoxEntry* _pEntry ) const
634 {
635     OSL_PRECOND( !isFolderEntry( _pEntry ), "OTableTreeListBox::getQualifiedTableName: folder entries not allowed here!" );
636 
637     try
638     {
639         Reference< XDatabaseMetaData > xMeta;
640         if ( !impl_getAndAssertMetaData( xMeta ) )
641             return String();
642 
643         ::rtl::OUString sCatalog;
644         ::rtl::OUString sSchema;
645         ::rtl::OUString sTable;
646 
647         SvLBoxEntry* pSchema = GetParent( _pEntry );
648         if ( pSchema )
649         {
650             SvLBoxEntry* pCatalog = GetParent( pSchema );
651             if  (   pCatalog
652                 ||  (   xMeta->supportsCatalogsInDataManipulation()
653                     &&  !xMeta->supportsSchemasInDataManipulation()
654                     )   // here we support catalog but no schema
655                 )
656             {
657                 if ( pCatalog == NULL )
658                 {
659                     pCatalog = pSchema;
660                     pSchema = NULL;
661                 }
662                 sCatalog = GetEntryText( pCatalog );
663             }
664             if ( pSchema )
665                 sSchema = GetEntryText(pSchema);
666         }
667         sTable = GetEntryText( _pEntry );
668 
669         return ::dbtools::composeTableName( xMeta, sCatalog, sSchema, sTable, sal_False, ::dbtools::eInDataManipulation );
670     }
671     catch( const Exception& )
672     {
673         DBG_UNHANDLED_EXCEPTION();
674     }
675     return String();
676 }
677 
678 //------------------------------------------------------------------------
getEntryByQualifiedName(const::rtl::OUString & _rName)679 SvLBoxEntry* OTableTreeListBox::getEntryByQualifiedName( const ::rtl::OUString& _rName )
680 {
681     try
682     {
683         Reference< XDatabaseMetaData > xMeta;
684         if ( !impl_getAndAssertMetaData( xMeta ) )
685             return NULL;
686 
687         // split the complete name into it's components
688         ::rtl::OUString sCatalog, sSchema, sName;
689         qualifiedNameComponents(xMeta, _rName, sCatalog, sSchema, sName,::dbtools::eInDataManipulation);
690 
691         SvLBoxEntry* pParent = getAllObjectsEntry();
692         SvLBoxEntry* pCat = NULL;
693         SvLBoxEntry* pSchema = NULL;
694         if ( sCatalog.getLength() )
695         {
696             pCat = GetEntryPosByName(sCatalog, pParent);
697             if ( pCat )
698                 pParent = pCat;
699         }
700 
701         if ( sSchema.getLength() )
702         {
703             pSchema = GetEntryPosByName(sSchema, pParent);
704             if ( pSchema )
705                 pParent = pSchema;
706         }
707 
708         return GetEntryPosByName(sName, pParent);
709     }
710     catch( const Exception& )
711     {
712         DBG_UNHANDLED_EXCEPTION();
713     }
714     return NULL;
715 }
716 //------------------------------------------------------------------------
removedTable(const::rtl::OUString & _rName)717 void OTableTreeListBox::removedTable( const ::rtl::OUString& _rName )
718 {
719     try
720     {
721         SvLBoxEntry* pEntry = getEntryByQualifiedName( _rName );
722         if ( pEntry )
723             GetModel()->Remove( pEntry );
724     }
725     catch( const Exception& )
726     {
727         DBG_UNHANDLED_EXCEPTION();
728     }
729 }
730 
731 //.........................................................................
732 }   // namespace dbaui
733 //.........................................................................
734 
735