1 /////////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     Implementation of the wxDbTable class. 
   5 // Modified by: George Tasker 
  10 // Copyright:   (c) 1996 Remstar International, Inc. 
  11 // Licence:     wxWindows licence, plus: 
  12 // Notice:      This class library and its intellectual design are free of charge for use, 
  13 //              modification, enhancement, debugging under the following conditions: 
  14 //              1) These classes may only be used as part of the implementation of a 
  15 //                 wxWindows-based application 
  16 //              2) All enhancements and bug fixes are to be submitted back to the wxWindows 
  17 //                 user groups free of all charges for use with the wxWindows library. 
  18 //              3) These classes may not be distributed as part of any other class library, 
  19 //                 DLL, text (written or electronic), other than a complete distribution of 
  20 //                 the wxWindows GUI development toolkit. 
  21 /////////////////////////////////////////////////////////////////////////////// 
  28 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  29     #pragma implementation "dbtable.h" 
  32 #include  "wx/wxprec.h" 
  38 #ifdef DBDEBUG_CONSOLE 
  44     #include "wx/ioswrap.h" 
  48     #include "wx/string.h" 
  49     #include "wx/object.h" 
  54 #include "wx/filefn.h" 
  62 #include "wx/dbtable.h" 
  65 // The HPUX preprocessor lines below were commented out on 8/20/97 
  66 // because macros.h currently redefines DEBUG and is unneeded. 
  68 // #    include <macros.h> 
  71 #    include <sys/minmax.h> 
  75 ULONG lastTableID 
= 0; 
  83 void csstrncpyt(wxChar 
*target
, const wxChar 
*source
, int n
) 
  85     while ( (*target
++ = *source
++) != '\0' && --n 
) 
  93 /********** wxDbColDef::wxDbColDef() Constructor **********/ 
  94 wxDbColDef::wxDbColDef() 
 100 bool wxDbColDef::Initialize() 
 103     DbDataType      
= DB_DATA_TYPE_INTEGER
; 
 104     SqlCtype        
= SQL_C_LONG
; 
 109     InsertAllowed   
= FALSE
; 
 115 }  // wxDbColDef::Initialize() 
 118 /********** wxDbTable::wxDbTable() Constructor **********/ 
 119 wxDbTable::wxDbTable(wxDb 
*pwxDb
, const wxString 
&tblName
, const UWORD numColumns
, 
 120                     const wxString 
&qryTblName
, bool qryOnly
, const wxString 
&tblPath
) 
 122     if (!initialize(pwxDb
, tblName
, numColumns
, qryTblName
, qryOnly
, tblPath
)) 
 124 }  // wxDbTable::wxDbTable() 
 127 /***** DEPRECATED: use wxDbTable::wxDbTable() format above *****/ 
 128 wxDbTable::wxDbTable(wxDb 
*pwxDb
, const wxString 
&tblName
, const UWORD numColumns
, 
 129                     const wxChar 
*qryTblName
, bool qryOnly
, const wxString 
&tblPath
) 
 131     wxString tempQryTblName
; 
 132     tempQryTblName 
= qryTblName
; 
 133     if (!initialize(pwxDb
, tblName
, numColumns
, tempQryTblName
, qryOnly
, tblPath
)) 
 135 }  // wxDbTable::wxDbTable() 
 138 /********** wxDbTable::~wxDbTable() **********/ 
 139 wxDbTable::~wxDbTable() 
 142 }  // wxDbTable::~wxDbTable() 
 145 bool wxDbTable::initialize(wxDb 
*pwxDb
, const wxString 
&tblName
, const UWORD numColumns
, 
 146                     const wxString 
&qryTblName
, bool qryOnly
, const wxString 
&tblPath
) 
 148     // Initializing member variables 
 149     pDb                 
= pwxDb
;                    // Pointer to the wxDb object 
 153     m_hstmtGridQuery               
= 0; 
 154     hstmtDefault        
= 0;                        // Initialized below 
 155     hstmtCount          
= 0;                        // Initialized first time it is needed 
 162     noCols              
= numColumns
;               // Number of cols in the table 
 163     where
.Empty();                                  // Where clause 
 164     orderBy
.Empty();                                // Order By clause 
 165     from
.Empty();                                   // From clause 
 166     selectForUpdate     
= FALSE
;                    // SELECT ... FOR UPDATE; Indicates whether to include the FOR UPDATE phrase 
 171     queryTableName
.Empty(); 
 173     wxASSERT(tblName
.Length()); 
 179     tableName 
= tblName
;                        // Table Name 
 180     if (tblPath
.Length()) 
 181         tablePath 
= tblPath
;                    // Table Path - used for dBase files 
 185     if (qryTblName
.Length())                    // Name of the table/view to query 
 186         queryTableName 
= qryTblName
; 
 188         queryTableName 
= tblName
; 
 190     pDb
->incrementTableCount(); 
 193     tableID 
= ++lastTableID
; 
 194     s
.Printf(wxT("wxDbTable constructor (%-20s) tableID:[%6lu] pDb:[%p]"), tblName
.c_str(), tableID
, pDb
); 
 197     wxTablesInUse 
*tableInUse
; 
 198     tableInUse            
= new wxTablesInUse(); 
 199     tableInUse
->tableName 
= tblName
; 
 200     tableInUse
->tableID   
= tableID
; 
 201     tableInUse
->pDb       
= pDb
; 
 202     TablesInUse
.Append(tableInUse
); 
 207     // Grab the HENV and HDBC from the wxDb object 
 208     henv 
= pDb
->GetHENV(); 
 209     hdbc 
= pDb
->GetHDBC(); 
 211     // Allocate space for column definitions 
 213         colDefs 
= new wxDbColDef
[noCols
];  // Points to the first column definition 
 215     // Allocate statement handles for the table 
 218         // Allocate a separate statement handle for performing inserts 
 219         if (SQLAllocStmt(hdbc
, &hstmtInsert
) != SQL_SUCCESS
) 
 220             pDb
->DispAllErrors(henv
, hdbc
); 
 221         // Allocate a separate statement handle for performing deletes 
 222         if (SQLAllocStmt(hdbc
, &hstmtDelete
) != SQL_SUCCESS
) 
 223             pDb
->DispAllErrors(henv
, hdbc
); 
 224         // Allocate a separate statement handle for performing updates 
 225         if (SQLAllocStmt(hdbc
, &hstmtUpdate
) != SQL_SUCCESS
) 
 226             pDb
->DispAllErrors(henv
, hdbc
); 
 228     // Allocate a separate statement handle for internal use 
 229     if (SQLAllocStmt(hdbc
, &hstmtInternal
) != SQL_SUCCESS
) 
 230         pDb
->DispAllErrors(henv
, hdbc
); 
 232     // Set the cursor type for the statement handles 
 233     cursorType 
= SQL_CURSOR_STATIC
; 
 235     if (SQLSetStmtOption(hstmtInternal
, SQL_CURSOR_TYPE
, cursorType
) != SQL_SUCCESS
) 
 237         // Check to see if cursor type is supported 
 238         pDb
->GetNextError(henv
, hdbc
, hstmtInternal
); 
 239         if (! wxStrcmp(pDb
->sqlState
, wxT("01S02")))  // Option Value Changed 
 241             // Datasource does not support static cursors.  Driver 
 242             // will substitute a cursor type.  Call SQLGetStmtOption() 
 243             // to determine which cursor type was selected. 
 244             if (SQLGetStmtOption(hstmtInternal
, SQL_CURSOR_TYPE
, &cursorType
) != SQL_SUCCESS
) 
 245                 pDb
->DispAllErrors(henv
, hdbc
, hstmtInternal
); 
 246 #ifdef DBDEBUG_CONSOLE 
 247             cout 
<< wxT("Static cursor changed to: "); 
 250             case SQL_CURSOR_FORWARD_ONLY
: 
 251                 cout 
<< wxT("Forward Only"); 
 253             case SQL_CURSOR_STATIC
: 
 254                 cout 
<< wxT("Static"); 
 256             case SQL_CURSOR_KEYSET_DRIVEN
: 
 257                 cout 
<< wxT("Keyset Driven"); 
 259             case SQL_CURSOR_DYNAMIC
: 
 260                 cout 
<< wxT("Dynamic"); 
 263             cout 
<< endl 
<< endl
; 
 266             if (pDb
->FwdOnlyCursors() && cursorType 
!= SQL_CURSOR_FORWARD_ONLY
) 
 268                 // Force the use of a forward only cursor... 
 269                 cursorType 
= SQL_CURSOR_FORWARD_ONLY
; 
 270                 if (SQLSetStmtOption(hstmtInternal
, SQL_CURSOR_TYPE
, cursorType
) != SQL_SUCCESS
) 
 272                     // Should never happen 
 273                     pDb
->GetNextError(henv
, hdbc
, hstmtInternal
); 
 280             pDb
->DispNextError(); 
 281             pDb
->DispAllErrors(henv
, hdbc
, hstmtInternal
); 
 284 #ifdef DBDEBUG_CONSOLE 
 286         cout 
<< wxT("Cursor Type set to STATIC") << endl 
<< endl
; 
 291         // Set the cursor type for the INSERT statement handle 
 292         if (SQLSetStmtOption(hstmtInsert
, SQL_CURSOR_TYPE
, SQL_CURSOR_FORWARD_ONLY
) != SQL_SUCCESS
) 
 293             pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
); 
 294         // Set the cursor type for the DELETE statement handle 
 295         if (SQLSetStmtOption(hstmtDelete
, SQL_CURSOR_TYPE
, SQL_CURSOR_FORWARD_ONLY
) != SQL_SUCCESS
) 
 296             pDb
->DispAllErrors(henv
, hdbc
, hstmtDelete
); 
 297         // Set the cursor type for the UPDATE statement handle 
 298         if (SQLSetStmtOption(hstmtUpdate
, SQL_CURSOR_TYPE
, SQL_CURSOR_FORWARD_ONLY
) != SQL_SUCCESS
) 
 299             pDb
->DispAllErrors(henv
, hdbc
, hstmtUpdate
); 
 302     // Make the default cursor the active cursor 
 303     hstmtDefault 
= GetNewCursor(FALSE
,FALSE
); 
 304     wxASSERT(hstmtDefault
); 
 305     hstmt 
= *hstmtDefault
; 
 309 }  // wxDbTable::initialize() 
 312 void wxDbTable::cleanup() 
 317         s
.Printf(wxT("wxDbTable destructor (%-20s) tableID:[%6lu] pDb:[%p]"), tableName
.c_str(), tableID
, pDb
); 
 324         TablesInUse
.DeleteContents(TRUE
); 
 328         pNode 
= TablesInUse
.First(); 
 329         while (pNode 
&& !found
) 
 331             if (((wxTablesInUse 
*)pNode
->Data())->tableID 
== tableID
) 
 334                 if (!TablesInUse
.DeleteNode(pNode
)) 
 335                     wxLogDebug (s
,wxT("Unable to delete node!")); 
 338                 pNode 
= pNode
->Next(); 
 343             msg
.Printf(wxT("Unable to find the tableID in the linked\nlist of tables in use.\n\n%s"),s
.c_str()); 
 344             wxLogDebug (msg
,wxT("NOTICE...")); 
 349     // Decrement the wxDb table count 
 351         pDb
->decrementTableCount(); 
 353     // Delete memory allocated for column definitions 
 357     // Free statement handles 
 363 ODBC 3.0 says to use this form 
 364             if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS) 
 366             if (SQLFreeStmt(hstmtInsert
, SQL_DROP
) != SQL_SUCCESS
) 
 367                 pDb
->DispAllErrors(henv
, hdbc
); 
 373 ODBC 3.0 says to use this form 
 374             if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS) 
 376             if (SQLFreeStmt(hstmtDelete
, SQL_DROP
) != SQL_SUCCESS
) 
 377                 pDb
->DispAllErrors(henv
, hdbc
); 
 383 ODBC 3.0 says to use this form 
 384             if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS) 
 386             if (SQLFreeStmt(hstmtUpdate
, SQL_DROP
) != SQL_SUCCESS
) 
 387                 pDb
->DispAllErrors(henv
, hdbc
); 
 393         if (SQLFreeStmt(hstmtInternal
, SQL_DROP
) != SQL_SUCCESS
) 
 394             pDb
->DispAllErrors(henv
, hdbc
); 
 397     // Delete dynamically allocated cursors 
 399         DeleteCursor(hstmtDefault
); 
 402         DeleteCursor(hstmtCount
); 
 404     if (m_hstmtGridQuery
) 
 405         DeleteCursor(m_hstmtGridQuery
); 
 407 }  // wxDbTable::cleanup() 
 410 /***************************** PRIVATE FUNCTIONS *****************************/ 
 413 /********** wxDbTable::bindParams() **********/ 
 414 bool wxDbTable::bindParams(bool forUpdate
) 
 416     wxASSERT(!queryOnly
); 
 421     SDWORD  precision   
= 0; 
 424     // Bind each column of the table that should be bound 
 425     // to a parameter marker 
 429     for (i
=0, colNo
=1; i 
< noCols
; i
++) 
 433             if (!colDefs
[i
].Updateable
) 
 438             if (!colDefs
[i
].InsertAllowed
) 
 442         switch(colDefs
[i
].DbDataType
) 
 444             case DB_DATA_TYPE_VARCHAR
: 
 445                 fSqlType 
= pDb
->GetTypeInfVarchar().FsqlType
; 
 446                 precision 
= colDefs
[i
].SzDataObj
; 
 449                     colDefs
[i
].CbValue 
= SQL_NULL_DATA
; 
 451                     colDefs
[i
].CbValue 
= SQL_NTS
; 
 453             case DB_DATA_TYPE_INTEGER
: 
 454                 fSqlType 
= pDb
->GetTypeInfInteger().FsqlType
; 
 455                 precision 
= pDb
->GetTypeInfInteger().Precision
; 
 458                     colDefs
[i
].CbValue 
= SQL_NULL_DATA
; 
 460                     colDefs
[i
].CbValue 
= 0; 
 462             case DB_DATA_TYPE_FLOAT
: 
 463                 fSqlType 
= pDb
->GetTypeInfFloat().FsqlType
; 
 464                 precision 
= pDb
->GetTypeInfFloat().Precision
; 
 465                 scale 
= pDb
->GetTypeInfFloat().MaximumScale
; 
 466                 // SQL Sybase Anywhere v5.5 returned a negative number for the 
 467                 // MaxScale.  This caused ODBC to kick out an error on ibscale. 
 468                 // I check for this here and set the scale = precision. 
 470                 // scale = (short) precision; 
 472                     colDefs
[i
].CbValue 
= SQL_NULL_DATA
; 
 474                     colDefs
[i
].CbValue 
= 0; 
 476             case DB_DATA_TYPE_DATE
: 
 477                 fSqlType 
= pDb
->GetTypeInfDate().FsqlType
; 
 478                 precision 
= pDb
->GetTypeInfDate().Precision
; 
 481                     colDefs
[i
].CbValue 
= SQL_NULL_DATA
; 
 483                     colDefs
[i
].CbValue 
= 0; 
 485             case DB_DATA_TYPE_BLOB
: 
 486                 fSqlType 
= pDb
->GetTypeInfBlob().FsqlType
; 
 490                     colDefs
[i
].CbValue 
= SQL_NULL_DATA
; 
 492                     colDefs
[i
].CbValue 
= SQL_LEN_DATA_AT_EXEC(colDefs
[i
].SzDataObj
); 
 497             if (SQLBindParameter(hstmtUpdate
, colNo
++, SQL_PARAM_INPUT
, colDefs
[i
].SqlCtype
, 
 498                                  fSqlType
, precision
, scale
, (UCHAR
*) colDefs
[i
].PtrDataObj
, 
 499                                  precision
+1, &colDefs
[i
].CbValue
) != SQL_SUCCESS
) 
 501                 return(pDb
->DispAllErrors(henv
, hdbc
, hstmtUpdate
)); 
 506             if (SQLBindParameter(hstmtInsert
, colNo
++, SQL_PARAM_INPUT
, colDefs
[i
].SqlCtype
, 
 507                                  fSqlType
, precision
, scale
, (UCHAR
*) colDefs
[i
].PtrDataObj
, 
 508                                  precision
+1,&colDefs
[i
].CbValue
) != SQL_SUCCESS
) 
 510                 return(pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
)); 
 515     // Completed successfully 
 518 }  // wxDbTable::bindParams() 
 521 /********** wxDbTable::bindInsertParams() **********/ 
 522 bool wxDbTable::bindInsertParams(void) 
 524     return bindParams(FALSE
); 
 525 }  // wxDbTable::bindInsertParams() 
 528 /********** wxDbTable::bindUpdateParams() **********/ 
 529 bool wxDbTable::bindUpdateParams(void) 
 531     return bindParams(TRUE
); 
 532 }  // wxDbTable::bindUpdateParams() 
 535 /********** wxDbTable::bindCols() **********/ 
 536 bool wxDbTable::bindCols(HSTMT cursor
) 
 540     // Bind each column of the table to a memory address for fetching data 
 542     for (i 
= 0; i 
< noCols
; i
++) 
 544         cb 
= colDefs
[i
].CbValue
; 
 545         if (SQLBindCol(cursor
, (UWORD
)(i
+1), colDefs
[i
].SqlCtype
, (UCHAR
*) colDefs
[i
].PtrDataObj
, 
 546                        colDefs
[i
].SzDataObj
, &cb 
) != SQL_SUCCESS
) 
 547           return (pDb
->DispAllErrors(henv
, hdbc
, cursor
)); 
 550     // Completed successfully 
 553 }  // wxDbTable::bindCols() 
 556 /********** wxDbTable::getRec() **********/ 
 557 bool wxDbTable::getRec(UWORD fetchType
) 
 561     if (!pDb
->FwdOnlyCursors()) 
 563         // Fetch the NEXT, PREV, FIRST or LAST record, depending on fetchType 
 567         retcode 
= SQLExtendedFetch(hstmt
, fetchType
, 0, &cRowsFetched
, &rowStatus
); 
 568         if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
 570             if (retcode 
== SQL_NO_DATA_FOUND
) 
 573                 return(pDb
->DispAllErrors(henv
, hdbc
, hstmt
)); 
 577             // Set the Null member variable to indicate the Null state 
 578             // of each column just read in. 
 580             for (i 
= 0; i 
< noCols
; i
++) 
 581                 colDefs
[i
].Null 
= (colDefs
[i
].CbValue 
== SQL_NULL_DATA
); 
 586         // Fetch the next record from the record set 
 587         retcode 
= SQLFetch(hstmt
); 
 588         if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
 590             if (retcode 
== SQL_NO_DATA_FOUND
) 
 593                 return(pDb
->DispAllErrors(henv
, hdbc
, hstmt
)); 
 597             // Set the Null member variable to indicate the Null state 
 598             // of each column just read in. 
 600             for (i 
= 0; i 
< noCols
; i
++) 
 601                 colDefs
[i
].Null 
= (colDefs
[i
].CbValue 
== SQL_NULL_DATA
); 
 605     // Completed successfully 
 608 }  // wxDbTable::getRec() 
 611 /********** wxDbTable::execDelete() **********/ 
 612 bool wxDbTable::execDelete(const wxString 
&pSqlStmt
) 
 616     // Execute the DELETE statement 
 617     retcode 
= SQLExecDirect(hstmtDelete
, (SQLTCHAR FAR 
*) pSqlStmt
.c_str(), SQL_NTS
); 
 619     if (retcode 
== SQL_SUCCESS 
|| 
 620         retcode 
== SQL_NO_DATA_FOUND 
|| 
 621         retcode 
== SQL_SUCCESS_WITH_INFO
) 
 623         // Record deleted successfully 
 627     // Problem deleting record 
 628     return(pDb
->DispAllErrors(henv
, hdbc
, hstmtDelete
)); 
 630 }  // wxDbTable::execDelete() 
 633 /********** wxDbTable::execUpdate() **********/ 
 634 bool wxDbTable::execUpdate(const wxString 
&pSqlStmt
) 
 638     // Execute the UPDATE statement 
 639     retcode 
= SQLExecDirect(hstmtUpdate
, (SQLTCHAR FAR 
*) pSqlStmt
.c_str(), SQL_NTS
); 
 641     if (retcode 
== SQL_SUCCESS 
|| 
 642         retcode 
== SQL_NO_DATA_FOUND 
|| 
 643         retcode 
== SQL_SUCCESS_WITH_INFO
) 
 645         // Record updated successfully 
 648     else if (retcode 
== SQL_NEED_DATA
) 
 651         retcode 
= SQLParamData(hstmtUpdate
, &pParmID
); 
 652         while (retcode 
== SQL_NEED_DATA
) 
 654             // Find the parameter 
 656             for (i
=0; i 
< noCols
; i
++) 
 658                 if (colDefs
[i
].PtrDataObj 
== pParmID
) 
 660                     // We found it.  Store the parameter. 
 661                     retcode 
= SQLPutData(hstmtUpdate
, pParmID
, colDefs
[i
].SzDataObj
); 
 662                     if (retcode 
!= SQL_SUCCESS
) 
 664                         pDb
->DispNextError(); 
 665                         return pDb
->DispAllErrors(henv
, hdbc
, hstmtUpdate
); 
 671         if (retcode 
== SQL_SUCCESS 
|| 
 672             retcode 
== SQL_NO_DATA_FOUND 
|| 
 673             retcode 
== SQL_SUCCESS_WITH_INFO
) 
 675             // Record updated successfully 
 680     // Problem updating record 
 681     return(pDb
->DispAllErrors(henv
, hdbc
, hstmtUpdate
)); 
 683 }  // wxDbTable::execUpdate() 
 686 /********** wxDbTable::query() **********/ 
 687 bool wxDbTable::query(int queryType
, bool forUpdate
, bool distinct
, const wxString 
&pSqlStmt
) 
 692         // The user may wish to select for update, but the DBMS may not be capable 
 693         selectForUpdate 
= CanSelectForUpdate(); 
 695         selectForUpdate 
= FALSE
; 
 697     // Set the SQL SELECT string 
 698     if (queryType 
!= DB_SELECT_STATEMENT
)               // A select statement was not passed in, 
 699     {                                                   // so generate a select statement. 
 700         BuildSelectStmt(sqlStmt
, queryType
, distinct
); 
 701         pDb
->WriteSqlLog(sqlStmt
); 
 704     // Make sure the cursor is closed first 
 705     if (!CloseCursor(hstmt
)) 
 708     // Execute the SQL SELECT statement 
 710     retcode 
= SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) (queryType 
== DB_SELECT_STATEMENT 
? pSqlStmt
.c_str() : sqlStmt
.c_str()), SQL_NTS
); 
 711     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
 712         return(pDb
->DispAllErrors(henv
, hdbc
, hstmt
)); 
 714     // Completed successfully 
 717 }  // wxDbTable::query() 
 720 /***************************** PUBLIC FUNCTIONS *****************************/ 
 723 /********** wxDbTable::Open() **********/ 
 724 bool wxDbTable::Open(bool checkPrivileges
, bool checkTableExists
) 
 734     // Calculate the maximum size of the concatenated 
 735     // keys for use with wxDbGrid 
 737     for (i
=0; i 
< noCols
; i
++) 
 739         if (colDefs
[i
].KeyField
) 
 742             m_keysize 
+= colDefs
[i
].SzDataObj
; 
 747     // Verify that the table exists in the database 
 748     if (checkTableExists 
&& !pDb
->TableExists(tableName
, pDb
->GetUsername(), tablePath
)) 
 750         s 
= wxT("Table/view does not exist in the database"); 
 751         if ( *(pDb
->dbInf
.accessibleTables
) == wxT('Y')) 
 752             s 
+= wxT(", or you have no permissions.\n"); 
 756     else if (checkPrivileges
) 
 758         // Verify the user has rights to access the table. 
 759         // Shortcut boolean evaluation to optimize out call to 
 762         // Unfortunately this optimization doesn't seem to be 
 764         if (// *(pDb->dbInf.accessibleTables) == 'N' && 
 765             !pDb
->TablePrivileges(tableName
,wxT("SELECT"), pDb
->GetUsername(), pDb
->GetUsername(), tablePath
)) 
 766             s 
= wxT("Current logged in user does not have sufficient privileges to access this table.\n"); 
 773         if (!tablePath
.IsEmpty()) 
 774             p
.Printf(wxT("Error opening '%s/%s'.\n"),tablePath
.c_str(),tableName
.c_str()); 
 776             p
.Printf(wxT("Error opening '%s'.\n"), tableName
.c_str()); 
 779         pDb
->LogError(p
.GetData()); 
 784     // Bind the member variables for field exchange between 
 785     // the wxDbTable object and the ODBC record. 
 788         if (!bindInsertParams())                    // Inserts 
 791         if (!bindUpdateParams())                    // Updates 
 795     if (!bindCols(*hstmtDefault
))                   // Selects 
 798     if (!bindCols(hstmtInternal
))                   // Internal use only 
 802      * Do NOT bind the hstmtCount cursor!!! 
 805     // Build an insert statement using parameter markers 
 806     if (!queryOnly 
&& noCols 
> 0) 
 808         bool needComma 
= FALSE
; 
 809         sqlStmt
.Printf(wxT("INSERT INTO %s ("), 
 810                        pDb
->SQLTableName(tableName
.c_str()).c_str()); 
 811         for (i 
= 0; i 
< noCols
; i
++) 
 813             if (! colDefs
[i
].InsertAllowed
) 
 817             sqlStmt 
+= pDb
->SQLColumnName(colDefs
[i
].ColName
); 
 818 //            sqlStmt += colDefs[i].ColName; 
 822         sqlStmt 
+= wxT(") VALUES ("); 
 824         int insertableCount 
= 0; 
 826         for (i 
= 0; i 
< noCols
; i
++) 
 828             if (! colDefs
[i
].InsertAllowed
) 
 838         // Prepare the insert statement for execution 
 841             if (SQLPrepare(hstmtInsert
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
) != SQL_SUCCESS
) 
 842                 return(pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
)); 
 848     // Completed successfully 
 851 }  // wxDbTable::Open() 
 854 /********** wxDbTable::Query() **********/ 
 855 bool wxDbTable::Query(bool forUpdate
, bool distinct
) 
 858     return(query(DB_SELECT_WHERE
, forUpdate
, distinct
)); 
 860 }  // wxDbTable::Query() 
 863 /********** wxDbTable::QueryBySqlStmt() **********/ 
 864 bool wxDbTable::QueryBySqlStmt(const wxString 
&pSqlStmt
) 
 866     pDb
->WriteSqlLog(pSqlStmt
); 
 868     return(query(DB_SELECT_STATEMENT
, FALSE
, FALSE
, pSqlStmt
)); 
 870 }  // wxDbTable::QueryBySqlStmt() 
 873 /********** wxDbTable::QueryMatching() **********/ 
 874 bool wxDbTable::QueryMatching(bool forUpdate
, bool distinct
) 
 877     return(query(DB_SELECT_MATCHING
, forUpdate
, distinct
)); 
 879 }  // wxDbTable::QueryMatching() 
 882 /********** wxDbTable::QueryOnKeyFields() **********/ 
 883 bool wxDbTable::QueryOnKeyFields(bool forUpdate
, bool distinct
) 
 886     return(query(DB_SELECT_KEYFIELDS
, forUpdate
, distinct
)); 
 888 }  // wxDbTable::QueryOnKeyFields() 
 891 /********** wxDbTable::GetPrev() **********/ 
 892 bool wxDbTable::GetPrev(void) 
 894     if (pDb
->FwdOnlyCursors()) 
 896         wxFAIL_MSG(wxT("GetPrev()::Backward scrolling cursors are not enabled for this instance of wxDbTable")); 
 900         return(getRec(SQL_FETCH_PRIOR
)); 
 902 }  // wxDbTable::GetPrev() 
 905 /********** wxDbTable::operator-- **********/ 
 906 bool wxDbTable::operator--(int) 
 908     if (pDb
->FwdOnlyCursors()) 
 910         wxFAIL_MSG(wxT("operator--:Backward scrolling cursors are not enabled for this instance of wxDbTable")); 
 914         return(getRec(SQL_FETCH_PRIOR
)); 
 916 }  // wxDbTable::operator-- 
 919 /********** wxDbTable::GetFirst() **********/ 
 920 bool wxDbTable::GetFirst(void) 
 922     if (pDb
->FwdOnlyCursors()) 
 924         wxFAIL_MSG(wxT("GetFirst():Backward scrolling cursors are not enabled for this instance of wxDbTable")); 
 928         return(getRec(SQL_FETCH_FIRST
)); 
 930 }  // wxDbTable::GetFirst() 
 933 /********** wxDbTable::GetLast() **********/ 
 934 bool wxDbTable::GetLast(void) 
 936     if (pDb
->FwdOnlyCursors()) 
 938         wxFAIL_MSG(wxT("GetLast()::Backward scrolling cursors are not enabled for this instance of wxDbTable")); 
 942         return(getRec(SQL_FETCH_LAST
)); 
 944 }  // wxDbTable::GetLast() 
 947 /********** wxDbTable::BuildDeleteStmt() **********/ 
 948 void wxDbTable::BuildDeleteStmt(wxString 
&pSqlStmt
, int typeOfDel
, const wxString 
&pWhereClause
) 
 950     wxASSERT(!queryOnly
); 
 954     wxString whereClause
; 
 958     // Handle the case of DeleteWhere() and the where clause is blank.  It should 
 959     // delete all records from the database in this case. 
 960     if (typeOfDel 
== DB_DEL_WHERE 
&& (pWhereClause
.Length() == 0)) 
 962         pSqlStmt
.Printf(wxT("DELETE FROM %s"), 
 963                         pDb
->SQLTableName(tableName
.c_str()).c_str()); 
 967     pSqlStmt
.Printf(wxT("DELETE FROM %s WHERE "), 
 968                     pDb
->SQLTableName(tableName
.c_str()).c_str()); 
 970     // Append the WHERE clause to the SQL DELETE statement 
 973         case DB_DEL_KEYFIELDS
: 
 974             // If the datasource supports the ROWID column, build 
 975             // the where on ROWID for efficiency purposes. 
 976             // e.g. DELETE FROM PARTS WHERE ROWID = '111.222.333' 
 980                 wxChar   rowid
[wxDB_ROWID_LEN
+1]; 
 982                 // Get the ROWID value.  If not successful retreiving the ROWID, 
 983                 // simply fall down through the code and build the WHERE clause 
 984                 // based on the key fields. 
 985                 if (SQLGetData(hstmt
, (UWORD
)(noCols
+1), SQL_C_CHAR
, (UCHAR
*) rowid
, wxDB_ROWID_LEN
, &cb
) == SQL_SUCCESS
) 
 987                     pSqlStmt 
+= wxT("ROWID = '"); 
 989                     pSqlStmt 
+= wxT("'"); 
 993             // Unable to delete by ROWID, so build a WHERE 
 994             // clause based on the keyfields. 
 995             BuildWhereClause(whereClause
, DB_WHERE_KEYFIELDS
); 
 996             pSqlStmt 
+= whereClause
; 
 999             pSqlStmt 
+= pWhereClause
; 
1001         case DB_DEL_MATCHING
: 
1002             BuildWhereClause(whereClause
, DB_WHERE_MATCHING
); 
1003             pSqlStmt 
+= whereClause
; 
1007 }  // BuildDeleteStmt() 
1010 /***** DEPRECATED: use wxDbTable::BuildDeleteStmt(wxString &....) form *****/ 
1011 void wxDbTable::BuildDeleteStmt(wxChar 
*pSqlStmt
, int typeOfDel
, const wxString 
&pWhereClause
) 
1013     wxString tempSqlStmt
; 
1014     BuildDeleteStmt(tempSqlStmt
, typeOfDel
, pWhereClause
); 
1015     wxStrcpy(pSqlStmt
, tempSqlStmt
); 
1016 }  // wxDbTable::BuildDeleteStmt() 
1019 /********** wxDbTable::BuildSelectStmt() **********/ 
1020 void wxDbTable::BuildSelectStmt(wxString 
&pSqlStmt
, int typeOfSelect
, bool distinct
) 
1022     wxString whereClause
; 
1023     whereClause
.Empty(); 
1025     // Build a select statement to query the database 
1026     pSqlStmt 
= wxT("SELECT "); 
1028     // SELECT DISTINCT values only? 
1030         pSqlStmt 
+= wxT("DISTINCT "); 
1032     // Was a FROM clause specified to join tables to the base table? 
1033     // Available for ::Query() only!!! 
1034     bool appendFromClause 
= FALSE
; 
1035 #if wxODBC_BACKWARD_COMPATABILITY 
1036     if (typeOfSelect 
== DB_SELECT_WHERE 
&& from 
&& wxStrlen(from
)) 
1037         appendFromClause 
= TRUE
; 
1039     if (typeOfSelect 
== DB_SELECT_WHERE 
&& from
.Length()) 
1040         appendFromClause 
= TRUE
; 
1043     // Add the column list 
1046     for (i 
= 0; i 
< noCols
; i
++) 
1048                   tStr 
= colDefs
[i
].ColName
; 
1049         // If joining tables, the base table column names must be qualified to avoid ambiguity 
1050         if ((appendFromClause 
|| pDb
->Dbms() == dbmsACCESS
) && !tStr
.Find(wxT('.'))) 
1052             pSqlStmt 
+= pDb
->SQLTableName(queryTableName
.c_str()); 
1053             pSqlStmt 
+= wxT("."); 
1055         pSqlStmt 
+= pDb
->SQLColumnName(colDefs
[i
].ColName
); 
1057             pSqlStmt 
+= wxT(","); 
1060     // If the datasource supports ROWID, get this column as well.  Exception: Don't retrieve 
1061     // the ROWID if querying distinct records.  The rowid will always be unique. 
1062     if (!distinct 
&& CanUpdByROWID()) 
1064         // If joining tables, the base table column names must be qualified to avoid ambiguity 
1065         if (appendFromClause 
|| pDb
->Dbms() == dbmsACCESS
) 
1067             pSqlStmt 
+= wxT(","); 
1068             pSqlStmt 
+= pDb
->SQLTableName(queryTableName
); 
1069 //            pSqlStmt += queryTableName; 
1070             pSqlStmt 
+= wxT(".ROWID"); 
1073             pSqlStmt 
+= wxT(",ROWID"); 
1076     // Append the FROM tablename portion 
1077     pSqlStmt 
+= wxT(" FROM "); 
1078     pSqlStmt 
+= pDb
->SQLTableName(queryTableName
); 
1079 //    pSqlStmt += queryTableName; 
1081     // Sybase uses the HOLDLOCK keyword to lock a record during query. 
1082     // The HOLDLOCK keyword follows the table name in the from clause. 
1083     // Each table in the from clause must specify HOLDLOCK or 
1084     // NOHOLDLOCK (the default).  Note: The "FOR UPDATE" clause 
1085     // is parsed but ignored in SYBASE Transact-SQL. 
1086     if (selectForUpdate 
&& (pDb
->Dbms() == dbmsSYBASE_ASA 
|| pDb
->Dbms() == dbmsSYBASE_ASE
)) 
1087         pSqlStmt 
+= wxT(" HOLDLOCK"); 
1089     if (appendFromClause
) 
1092     // Append the WHERE clause.  Either append the where clause for the class 
1093     // or build a where clause.  The typeOfSelect determines this. 
1094     switch(typeOfSelect
) 
1096         case DB_SELECT_WHERE
: 
1097 #if wxODBC_BACKWARD_COMPATABILITY 
1098             if (where 
&& wxStrlen(where
))   // May not want a where clause!!! 
1100             if (where
.Length())   // May not want a where clause!!! 
1103                 pSqlStmt 
+= wxT(" WHERE "); 
1107         case DB_SELECT_KEYFIELDS
: 
1108             BuildWhereClause(whereClause
, DB_WHERE_KEYFIELDS
); 
1109             if (whereClause
.Length()) 
1111                 pSqlStmt 
+= wxT(" WHERE "); 
1112                 pSqlStmt 
+= whereClause
; 
1115         case DB_SELECT_MATCHING
: 
1116             BuildWhereClause(whereClause
, DB_WHERE_MATCHING
); 
1117             if (whereClause
.Length()) 
1119                 pSqlStmt 
+= wxT(" WHERE "); 
1120                 pSqlStmt 
+= whereClause
; 
1125     // Append the ORDER BY clause 
1126 #if wxODBC_BACKWARD_COMPATABILITY 
1127     if (orderBy 
&& wxStrlen(orderBy
)) 
1129     if (orderBy
.Length()) 
1132         pSqlStmt 
+= wxT(" ORDER BY "); 
1133         pSqlStmt 
+= orderBy
; 
1136     // SELECT FOR UPDATE if told to do so and the datasource is capable.  Sybase 
1137     // parses the FOR UPDATE clause but ignores it.  See the comment above on the 
1138     // HOLDLOCK for Sybase. 
1139     if (selectForUpdate 
&& CanSelectForUpdate()) 
1140         pSqlStmt 
+= wxT(" FOR UPDATE"); 
1142 }  // wxDbTable::BuildSelectStmt() 
1145 /***** DEPRECATED: use wxDbTable::BuildSelectStmt(wxString &....) form *****/ 
1146 void wxDbTable::BuildSelectStmt(wxChar 
*pSqlStmt
, int typeOfSelect
, bool distinct
) 
1148     wxString tempSqlStmt
; 
1149     BuildSelectStmt(tempSqlStmt
, typeOfSelect
, distinct
); 
1150     wxStrcpy(pSqlStmt
, tempSqlStmt
); 
1151 }  // wxDbTable::BuildSelectStmt() 
1154 /********** wxDbTable::BuildUpdateStmt() **********/ 
1155 void wxDbTable::BuildUpdateStmt(wxString 
&pSqlStmt
, int typeOfUpd
, const wxString 
&pWhereClause
) 
1157     wxASSERT(!queryOnly
); 
1161     wxString whereClause
; 
1162     whereClause
.Empty(); 
1164     bool firstColumn 
= TRUE
; 
1166     pSqlStmt
.Printf(wxT("UPDATE %s SET "), 
1167                     pDb
->SQLTableName(tableName
.c_str()).c_str()); 
1169     // Append a list of columns to be updated 
1171     for (i 
= 0; i 
< noCols
; i
++) 
1173         // Only append Updateable columns 
1174         if (colDefs
[i
].Updateable
) 
1177                 pSqlStmt 
+= wxT(","); 
1179                 firstColumn 
= FALSE
; 
1181             pSqlStmt 
+= pDb
->SQLColumnName(colDefs
[i
].ColName
); 
1182 //            pSqlStmt += colDefs[i].ColName; 
1183             pSqlStmt 
+= wxT(" = ?"); 
1187     // Append the WHERE clause to the SQL UPDATE statement 
1188     pSqlStmt 
+= wxT(" WHERE "); 
1191         case DB_UPD_KEYFIELDS
: 
1192             // If the datasource supports the ROWID column, build 
1193             // the where on ROWID for efficiency purposes. 
1194             // e.g. UPDATE PARTS SET Col1 = ?, Col2 = ? WHERE ROWID = '111.222.333' 
1195             if (CanUpdByROWID()) 
1198                 wxChar rowid
[wxDB_ROWID_LEN
+1]; 
1200                 // Get the ROWID value.  If not successful retreiving the ROWID, 
1201                 // simply fall down through the code and build the WHERE clause 
1202                 // based on the key fields. 
1203                 if (SQLGetData(hstmt
, (UWORD
)(noCols
+1), SQL_C_CHAR
, (UCHAR
*) rowid
, wxDB_ROWID_LEN
, &cb
) == SQL_SUCCESS
) 
1205                     pSqlStmt 
+= wxT("ROWID = '"); 
1207                     pSqlStmt 
+= wxT("'"); 
1211             // Unable to delete by ROWID, so build a WHERE 
1212             // clause based on the keyfields. 
1213             BuildWhereClause(whereClause
, DB_WHERE_KEYFIELDS
); 
1214             pSqlStmt 
+= whereClause
; 
1217             pSqlStmt 
+= pWhereClause
; 
1220 }  // BuildUpdateStmt() 
1223 /***** DEPRECATED: use wxDbTable::BuildUpdateStmt(wxString &....) form *****/ 
1224 void wxDbTable::BuildUpdateStmt(wxChar 
*pSqlStmt
, int typeOfUpd
, const wxString 
&pWhereClause
) 
1226     wxString tempSqlStmt
; 
1227     BuildUpdateStmt(tempSqlStmt
, typeOfUpd
, pWhereClause
); 
1228     wxStrcpy(pSqlStmt
, tempSqlStmt
); 
1229 }  // BuildUpdateStmt() 
1232 /********** wxDbTable::BuildWhereClause() **********/ 
1233 void wxDbTable::BuildWhereClause(wxString 
&pWhereClause
, int typeOfWhere
, 
1234                                  const wxString 
&qualTableName
, bool useLikeComparison
) 
1236  * Note: BuildWhereClause() currently ignores timestamp columns. 
1237  *       They are not included as part of the where clause. 
1240     bool moreThanOneColumn 
= FALSE
; 
1243     // Loop through the columns building a where clause as you go 
1245     for (colNo 
= 0; colNo 
< noCols
; colNo
++) 
1247         // Determine if this column should be included in the WHERE clause 
1248         if ((typeOfWhere 
== DB_WHERE_KEYFIELDS 
&& colDefs
[colNo
].KeyField
) || 
1249              (typeOfWhere 
== DB_WHERE_MATCHING  
&& (!IsColNull(colNo
)))) 
1251             // Skip over timestamp columns 
1252             if (colDefs
[colNo
].SqlCtype 
== SQL_C_TIMESTAMP
) 
1254             // If there is more than 1 column, join them with the keyword "AND" 
1255             if (moreThanOneColumn
) 
1256                 pWhereClause 
+= wxT(" AND "); 
1258                 moreThanOneColumn 
= TRUE
; 
1260             // Concatenate where phrase for the column 
1261             wxString tStr 
= colDefs
[colNo
].ColName
; 
1263             if (qualTableName
.Length() && !tStr
.Find(wxT('.'))) 
1265                 pWhereClause 
+= pDb
->SQLTableName(qualTableName
); 
1266                 pWhereClause 
+= wxT("."); 
1268             pWhereClause 
+= pDb
->SQLColumnName(colDefs
[colNo
].ColName
); 
1270             if (useLikeComparison 
&& (colDefs
[colNo
].SqlCtype 
== SQL_C_CHAR
)) 
1271                 pWhereClause 
+= wxT(" LIKE "); 
1273                 pWhereClause 
+= wxT(" = "); 
1275             switch(colDefs
[colNo
].SqlCtype
) 
1278                     colValue
.Printf(wxT("'%s'"), (UCHAR FAR 
*) colDefs
[colNo
].PtrDataObj
); 
1281                     colValue
.Printf(wxT("%hi"), *((SWORD 
*) colDefs
[colNo
].PtrDataObj
)); 
1284                     colValue
.Printf(wxT("%hu"), *((UWORD 
*) colDefs
[colNo
].PtrDataObj
)); 
1287                     colValue
.Printf(wxT("%li"), *((SDWORD 
*) colDefs
[colNo
].PtrDataObj
)); 
1290                     colValue
.Printf(wxT("%lu"), *((UDWORD 
*) colDefs
[colNo
].PtrDataObj
)); 
1293                     colValue
.Printf(wxT("%.6f"), *((SFLOAT 
*) colDefs
[colNo
].PtrDataObj
)); 
1296                     colValue
.Printf(wxT("%.6f"), *((SDOUBLE 
*) colDefs
[colNo
].PtrDataObj
)); 
1299             pWhereClause 
+= colValue
; 
1302 }  // wxDbTable::BuildWhereClause() 
1305 /***** DEPRECATED: use wxDbTable::BuildWhereClause(wxString &....) form *****/ 
1306 void wxDbTable::BuildWhereClause(wxChar 
*pWhereClause
, int typeOfWhere
, 
1307                                  const wxString 
&qualTableName
, bool useLikeComparison
) 
1309     wxString tempSqlStmt
; 
1310     BuildWhereClause(tempSqlStmt
, typeOfWhere
, qualTableName
, useLikeComparison
); 
1311     wxStrcpy(pWhereClause
, tempSqlStmt
); 
1312 }  // wxDbTable::BuildWhereClause() 
1315 /********** wxDbTable::GetRowNum() **********/ 
1316 UWORD 
wxDbTable::GetRowNum(void) 
1320     if (SQLGetStmtOption(hstmt
, SQL_ROW_NUMBER
, (UCHAR
*) &rowNum
) != SQL_SUCCESS
) 
1322         pDb
->DispAllErrors(henv
, hdbc
, hstmt
); 
1326     // Completed successfully 
1327     return((UWORD
) rowNum
); 
1329 }  // wxDbTable::GetRowNum() 
1332 /********** wxDbTable::CloseCursor() **********/ 
1333 bool wxDbTable::CloseCursor(HSTMT cursor
) 
1335     if (SQLFreeStmt(cursor
, SQL_CLOSE
) != SQL_SUCCESS
) 
1336         return(pDb
->DispAllErrors(henv
, hdbc
, cursor
)); 
1338     // Completed successfully 
1341 }  // wxDbTable::CloseCursor() 
1344 /********** wxDbTable::CreateTable() **********/ 
1345 bool wxDbTable::CreateTable(bool attemptDrop
) 
1353 #ifdef DBDEBUG_CONSOLE 
1354     cout 
<< wxT("Creating Table ") << tableName 
<< wxT("...") << endl
; 
1358     if (attemptDrop 
&& !DropTable()) 
1362 #ifdef DBDEBUG_CONSOLE 
1363     for (i 
= 0; i 
< noCols
; i
++) 
1365         // Exclude derived columns since they are NOT part of the base table 
1366         if (colDefs
[i
].DerivedCol
) 
1368         cout 
<< i 
+ 1 << wxT(": ") << colDefs
[i
].ColName 
<< wxT("; "); 
1369         switch(colDefs
[i
].DbDataType
) 
1371             case DB_DATA_TYPE_VARCHAR
: 
1372                 cout 
<< pDb
->GetTypeInfVarchar().TypeName 
<< wxT("(") << colDefs
[i
].SzDataObj 
<< wxT(")"); 
1374             case DB_DATA_TYPE_INTEGER
: 
1375                 cout 
<< pDb
->GetTypeInfInteger().TypeName
; 
1377             case DB_DATA_TYPE_FLOAT
: 
1378                 cout 
<< pDb
->GetTypeInfFloat().TypeName
; 
1380             case DB_DATA_TYPE_DATE
: 
1381                 cout 
<< pDb
->GetTypeInfDate().TypeName
; 
1383             case DB_DATA_TYPE_BLOB
: 
1384                 cout 
<< pDb
->GetTypeInfBlob().TypeName
; 
1391     // Build a CREATE TABLE string from the colDefs structure. 
1392     bool needComma 
= FALSE
; 
1394     sqlStmt
.Printf(wxT("CREATE TABLE %s ("), 
1395                    pDb
->SQLTableName(tableName
.c_str()).c_str()); 
1397     for (i 
= 0; i 
< noCols
; i
++) 
1399         // Exclude derived columns since they are NOT part of the base table 
1400         if (colDefs
[i
].DerivedCol
) 
1404             sqlStmt 
+= wxT(","); 
1406         sqlStmt 
+= pDb
->SQLColumnName(colDefs
[i
].ColName
); 
1407 //        sqlStmt += colDefs[i].ColName; 
1408         sqlStmt 
+= wxT(" "); 
1410         switch(colDefs
[i
].DbDataType
) 
1412             case DB_DATA_TYPE_VARCHAR
: 
1413                 sqlStmt 
+= pDb
->GetTypeInfVarchar().TypeName
; 
1415             case DB_DATA_TYPE_INTEGER
: 
1416                 sqlStmt 
+= pDb
->GetTypeInfInteger().TypeName
; 
1418             case DB_DATA_TYPE_FLOAT
: 
1419                 sqlStmt 
+= pDb
->GetTypeInfFloat().TypeName
; 
1421             case DB_DATA_TYPE_DATE
: 
1422                 sqlStmt 
+= pDb
->GetTypeInfDate().TypeName
; 
1424             case DB_DATA_TYPE_BLOB
: 
1425                 sqlStmt 
+= pDb
->GetTypeInfBlob().TypeName
; 
1428         // For varchars, append the size of the string 
1429         if (colDefs
[i
].DbDataType 
== DB_DATA_TYPE_VARCHAR 
&& 
1430             (pDb
->Dbms() != dbmsMY_SQL 
|| pDb
->GetTypeInfVarchar().TypeName 
!= _T("text")))// || 
1431 //            colDefs[i].DbDataType == DB_DATA_TYPE_BLOB) 
1434             s
.Printf(wxT("(%d)"), colDefs
[i
].SzDataObj
); 
1438         if (pDb
->Dbms() == dbmsDB2 
|| 
1439             pDb
->Dbms() == dbmsMY_SQL 
|| 
1440             pDb
->Dbms() == dbmsSYBASE_ASE  
|| 
1441             pDb
->Dbms() == dbmsINTERBASE  
|| 
1442             pDb
->Dbms() == dbmsMS_SQL_SERVER
) 
1444             if (colDefs
[i
].KeyField
) 
1446                 sqlStmt 
+= wxT(" NOT NULL"); 
1452     // If there is a primary key defined, include it in the create statement 
1453     for (i 
= j 
= 0; i 
< noCols
; i
++) 
1455         if (colDefs
[i
].KeyField
) 
1461     if (j 
&& (pDb
->Dbms() != dbmsDBASE
)  
1462                   && (pDb
->Dbms() != dbmsXBASE_SEQUITER
) 
1463            )  // Found a keyfield 
1465         switch (pDb
->Dbms()) 
1469             case dbmsSYBASE_ASA
: 
1470             case dbmsSYBASE_ASE
: 
1473                 // MySQL goes out on this one. We also declare the relevant key NON NULL above 
1474                 sqlStmt 
+= wxT(",PRIMARY KEY ("); 
1479                 sqlStmt 
+= wxT(",CONSTRAINT "); 
1480                 //  DB2 is limited to 18 characters for index names 
1481                 if (pDb
->Dbms() == dbmsDB2
) 
1483                     wxASSERT_MSG((tableName 
&& wxStrlen(tableName
) <= 13), wxT("DB2 table/index names must be no longer than 13 characters in length.\n\nTruncating table name to 13 characters.")); 
1484                     sqlStmt 
+= pDb
->SQLTableName(tableName
.substr(0, 13).c_str()); 
1485 //                    sqlStmt += tableName.substr(0, 13); 
1488                     sqlStmt 
+= pDb
->SQLTableName(tableName
.c_str()); 
1489 //                    sqlStmt += tableName; 
1491                 sqlStmt 
+= wxT("_PIDX PRIMARY KEY ("); 
1496         // List column name(s) of column(s) comprising the primary key 
1497         for (i 
= j 
= 0; i 
< noCols
; i
++) 
1499             if (colDefs
[i
].KeyField
) 
1501                 if (j
++) // Multi part key, comma separate names 
1502                     sqlStmt 
+= wxT(","); 
1503                 sqlStmt 
+= pDb
->SQLColumnName(colDefs
[i
].ColName
); 
1505                 if (pDb
->Dbms() == dbmsMY_SQL 
&& 
1506                     colDefs
[i
].DbDataType 
==  DB_DATA_TYPE_VARCHAR
) 
1509                     s
.Printf(wxT("(%d)"), colDefs
[i
].SzDataObj
); 
1514         sqlStmt 
+= wxT(")"); 
1516         if (pDb
->Dbms() == dbmsINFORMIX 
|| 
1517             pDb
->Dbms() == dbmsSYBASE_ASA 
|| 
1518             pDb
->Dbms() == dbmsSYBASE_ASE
) 
1520             sqlStmt 
+= wxT(" CONSTRAINT "); 
1521             sqlStmt 
+= pDb
->SQLTableName(tableName
); 
1522 //            sqlStmt += tableName; 
1523             sqlStmt 
+= wxT("_PIDX"); 
1526     // Append the closing parentheses for the create table statement 
1527     sqlStmt 
+= wxT(")"); 
1529     pDb
->WriteSqlLog(sqlStmt
); 
1531 #ifdef DBDEBUG_CONSOLE 
1532     cout 
<< endl 
<< sqlStmt
.c_str() << endl
; 
1535     // Execute the CREATE TABLE statement 
1536     RETCODE retcode 
= SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
); 
1537     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
1539         pDb
->DispAllErrors(henv
, hdbc
, hstmt
); 
1540         pDb
->RollbackTrans(); 
1545     // Commit the transaction and close the cursor 
1546     if (!pDb
->CommitTrans()) 
1548     if (!CloseCursor(hstmt
)) 
1551     // Database table created successfully 
1554 } // wxDbTable::CreateTable() 
1557 /********** wxDbTable::DropTable() **********/ 
1558 bool wxDbTable::DropTable() 
1560     // NOTE: This function returns TRUE if the Table does not exist, but 
1561     //       only for identified databases.  Code will need to be added 
1562     //       below for any other databases when those databases are defined 
1563     //       to handle this situation consistently 
1567     sqlStmt
.Printf(wxT("DROP TABLE %s"), 
1568                    pDb
->SQLTableName(tableName
.c_str()).c_str()); 
1570     pDb
->WriteSqlLog(sqlStmt
); 
1572 #ifdef DBDEBUG_CONSOLE 
1573     cout 
<< endl 
<< sqlStmt
.c_str() << endl
; 
1576     RETCODE retcode 
= SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
); 
1577     if (retcode 
!= SQL_SUCCESS
) 
1579         // Check for "Base table not found" error and ignore 
1580         pDb
->GetNextError(henv
, hdbc
, hstmt
); 
1581         if (wxStrcmp(pDb
->sqlState
, wxT("S0002")) /*&& 
1582             wxStrcmp(pDb->sqlState, wxT("S1000"))*/)  // "Base table not found" 
1584             // Check for product specific error codes 
1585             if (!((pDb
->Dbms() == dbmsSYBASE_ASA    
&& !wxStrcmp(pDb
->sqlState
,wxT("42000")))   ||  // 5.x (and lower?) 
1586                   (pDb
->Dbms() == dbmsSYBASE_ASE    
&& !wxStrcmp(pDb
->sqlState
,wxT("37000")))   || 
1587                   (pDb
->Dbms() == dbmsPERVASIVE_SQL 
&& !wxStrcmp(pDb
->sqlState
,wxT("S1000")))   ||  // Returns an S1000 then an S0002 
1588                   (pDb
->Dbms() == dbmsPOSTGRES      
&& !wxStrcmp(pDb
->sqlState
,wxT("08S01"))))) 
1590                 pDb
->DispNextError(); 
1591                 pDb
->DispAllErrors(henv
, hdbc
, hstmt
); 
1592                 pDb
->RollbackTrans(); 
1593 //                CloseCursor(hstmt); 
1599     // Commit the transaction and close the cursor 
1600     if (! pDb
->CommitTrans()) 
1602     if (! CloseCursor(hstmt
)) 
1606 }  // wxDbTable::DropTable() 
1609 /********** wxDbTable::CreateIndex() **********/ 
1610 bool wxDbTable::CreateIndex(const wxString 
&idxName
, bool unique
, UWORD noIdxCols
, 
1611                                      wxDbIdxDef 
*pIdxDefs
, bool attemptDrop
) 
1615     // Drop the index first 
1616     if (attemptDrop 
&& !DropIndex(idxName
)) 
1619     // MySQL (and possibly Sybase ASE?? - gt) require that any columns which are used as portions 
1620     // of an index have the columns defined as "NOT NULL".  During initial table creation though, 
1621     // it may not be known which columns are necessarily going to be part of an index (e.g. the 
1622     // table was created, then months later you determine that an additional index while 
1623     // give better performance, so you want to add an index). 
1625     // The following block of code will modify the column definition to make the column be 
1626     // defined with the "NOT NULL" qualifier. 
1627     if (pDb
->Dbms() == dbmsMY_SQL
) 
1632         for (i 
= 0; i 
< noIdxCols 
&& ok
; i
++) 
1636             // Find the column definition that has the ColName that matches the 
1637             // index column name.  We need to do this to get the DB_DATA_TYPE of 
1638             // the index column, as MySQL's syntax for the ALTER column requires 
1640             while (!found 
&& (j 
< this->noCols
)) 
1642                 if (wxStrcmp(colDefs
[j
].ColName
,pIdxDefs
[i
].ColName
) == 0) 
1650                 ok 
= pDb
->ModifyColumn(tableName
, pIdxDefs
[i
].ColName
, 
1651                                         colDefs
[j
].DbDataType
, colDefs
[j
].SzDataObj
, 
1657                     // retcode is not used 
1658                     wxODBC_ERRORS retcode
; 
1659                     // Oracle returns a DB_ERR_GENERAL_ERROR if the column is already 
1660                     // defined to be NOT NULL, but reportedly MySQL doesn't mind. 
1661                     // This line is just here for debug checking of the value 
1662                     retcode 
= (wxODBC_ERRORS
)pDb
->DB_STATUS
; 
1673             pDb
->RollbackTrans(); 
1678     // Build a CREATE INDEX statement 
1679     sqlStmt 
= wxT("CREATE "); 
1681         sqlStmt 
+= wxT("UNIQUE "); 
1683     sqlStmt 
+= wxT("INDEX "); 
1684     sqlStmt 
+= pDb
->SQLTableName(idxName
); 
1685     sqlStmt 
+= wxT(" ON "); 
1687     sqlStmt 
+= pDb
->SQLTableName(tableName
); 
1688 //    sqlStmt += tableName; 
1689     sqlStmt 
+= wxT(" ("); 
1691     // Append list of columns making up index 
1693     for (i 
= 0; i 
< noIdxCols
; i
++) 
1695         sqlStmt 
+= pDb
->SQLColumnName(pIdxDefs
[i
].ColName
); 
1696 //        sqlStmt += pIdxDefs[i].ColName; 
1698         // MySQL requires a key length on VARCHAR keys 
1699         if ( pDb
->Dbms() == dbmsMY_SQL 
) 
1701             // Find the details on this column 
1703             for ( j 
= 0; j 
< noCols
; ++j 
) 
1705                 if ( wxStrcmp( pIdxDefs
[i
].ColName
, colDefs
[j
].ColName 
) == 0 ) 
1710             if ( colDefs
[j
].DbDataType 
==  DB_DATA_TYPE_VARCHAR
) 
1713                 s
.Printf(wxT("(%d)"), colDefs
[i
].SzDataObj
); 
1718         // Postgres and SQL Server 7 do not support the ASC/DESC keywords for index columns 
1719         if (!((pDb
->Dbms() == dbmsMS_SQL_SERVER
) && (wxStrncmp(pDb
->dbInf
.dbmsVer
,_T("07"),2)==0)) && 
1720             !(pDb
->Dbms() == dbmsPOSTGRES
)) 
1722             if (pIdxDefs
[i
].Ascending
) 
1723                 sqlStmt 
+= wxT(" ASC"); 
1725                 sqlStmt 
+= wxT(" DESC"); 
1728             wxASSERT_MSG(pIdxDefs
[i
].Ascending
, _T("Datasource does not support DESCending index columns")); 
1730         if ((i 
+ 1) < noIdxCols
) 
1731             sqlStmt 
+= wxT(","); 
1734     // Append closing parentheses 
1735     sqlStmt 
+= wxT(")"); 
1737     pDb
->WriteSqlLog(sqlStmt
); 
1739 #ifdef DBDEBUG_CONSOLE 
1740     cout 
<< endl 
<< sqlStmt
.c_str() << endl 
<< endl
; 
1743     // Execute the CREATE INDEX statement 
1744     if (SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
) != SQL_SUCCESS
) 
1746         pDb
->DispAllErrors(henv
, hdbc
, hstmt
); 
1747         pDb
->RollbackTrans(); 
1752     // Commit the transaction and close the cursor 
1753     if (! pDb
->CommitTrans()) 
1755     if (! CloseCursor(hstmt
)) 
1758     // Index Created Successfully 
1761 }  // wxDbTable::CreateIndex() 
1764 /********** wxDbTable::DropIndex() **********/ 
1765 bool wxDbTable::DropIndex(const wxString 
&idxName
) 
1767     // NOTE: This function returns TRUE if the Index does not exist, but 
1768     //       only for identified databases.  Code will need to be added 
1769     //       below for any other databases when those databases are defined 
1770     //       to handle this situation consistently 
1774     if (pDb
->Dbms() == dbmsACCESS 
|| pDb
->Dbms() == dbmsMY_SQL 
|| 
1775         pDb
->Dbms() == dbmsDBASE 
/*|| Paradox needs this syntax too when we add support*/) 
1776         sqlStmt
.Printf(wxT("DROP INDEX %s ON %s"), 
1777                        pDb
->SQLTableName(idxName
.c_str()).c_str(), 
1778                        pDb
->SQLTableName(tableName
.c_str()).c_str()); 
1779     else if ((pDb
->Dbms() == dbmsMS_SQL_SERVER
) || 
1780              (pDb
->Dbms() == dbmsSYBASE_ASE
) || 
1781                          (pDb
->Dbms() == dbmsXBASE_SEQUITER
)) 
1782         sqlStmt
.Printf(wxT("DROP INDEX %s.%s"), 
1783                        pDb
->SQLTableName(tableName
.c_str()).c_str(), 
1784                        pDb
->SQLTableName(idxName
.c_str()).c_str()); 
1786         sqlStmt
.Printf(wxT("DROP INDEX %s"), 
1787                        pDb
->SQLTableName(idxName
.c_str()).c_str()); 
1789     pDb
->WriteSqlLog(sqlStmt
); 
1791 #ifdef DBDEBUG_CONSOLE 
1792     cout 
<< endl 
<< sqlStmt
.c_str() << endl
; 
1795     if (SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
) != SQL_SUCCESS
) 
1797         // Check for "Index not found" error and ignore 
1798         pDb
->GetNextError(henv
, hdbc
, hstmt
); 
1799         if (wxStrcmp(pDb
->sqlState
,wxT("S0012")))  // "Index not found" 
1801             // Check for product specific error codes 
1802             if (!((pDb
->Dbms() == dbmsSYBASE_ASA    
&& !wxStrcmp(pDb
->sqlState
,wxT("42000"))) ||  // v5.x (and lower?) 
1803                   (pDb
->Dbms() == dbmsSYBASE_ASE    
&& !wxStrcmp(pDb
->sqlState
,wxT("37000"))) || 
1804                   (pDb
->Dbms() == dbmsMS_SQL_SERVER 
&& !wxStrcmp(pDb
->sqlState
,wxT("S1000"))) || 
1805                   (pDb
->Dbms() == dbmsINTERBASE      
&& !wxStrcmp(pDb
->sqlState
,wxT("S1000"))) || 
1806                   (pDb
->Dbms() == dbmsSYBASE_ASE    
&& !wxStrcmp(pDb
->sqlState
,wxT("S0002"))) ||  // Base table not found 
1807                   (pDb
->Dbms() == dbmsMY_SQL        
&& !wxStrcmp(pDb
->sqlState
,wxT("42S12"))) ||  // tested by Christopher Ludwik Marino-Cebulski using v3.23.21beta 
1808                   (pDb
->Dbms() == dbmsPOSTGRES      
&& !wxStrcmp(pDb
->sqlState
,wxT("08S01"))) 
1811                 pDb
->DispNextError(); 
1812                 pDb
->DispAllErrors(henv
, hdbc
, hstmt
); 
1813                 pDb
->RollbackTrans(); 
1820     // Commit the transaction and close the cursor 
1821     if (! pDb
->CommitTrans()) 
1823     if (! CloseCursor(hstmt
)) 
1827 }  // wxDbTable::DropIndex() 
1830 /********** wxDbTable::SetOrderByColNums() **********/ 
1831 bool wxDbTable::SetOrderByColNums(UWORD first
, ... ) 
1833     int        colNo 
= first
;  // using 'int' to be able to look for wxDB_NO_MORE_COLUN_NUMBERS 
1839     va_start(argptr
, first
);     /* Initialize variable arguments. */ 
1840     while (!abort 
&& (colNo 
!= wxDB_NO_MORE_COLUMN_NUMBERS
)) 
1842         // Make sure the passed in column number 
1843         // is within the valid range of columns 
1845         // Valid columns are 0 thru noCols-1 
1846         if (colNo 
>= noCols 
|| colNo 
< 0) 
1853             tempStr 
+= wxT(","); 
1855         tempStr 
+= colDefs
[colNo
].ColName
; 
1856         colNo 
= va_arg (argptr
, int); 
1858     va_end (argptr
);              /* Reset variable arguments.      */ 
1860     SetOrderByClause(tempStr
); 
1863 }  // wxDbTable::SetOrderByColNums() 
1866 /********** wxDbTable::Insert() **********/ 
1867 int wxDbTable::Insert(void) 
1869     wxASSERT(!queryOnly
); 
1870     if (queryOnly 
|| !insertable
) 
1875     // Insert the record by executing the already prepared insert statement 
1877     retcode
=SQLExecute(hstmtInsert
); 
1878     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
&& 
1879         retcode 
!= SQL_NEED_DATA
) 
1881         // Check to see if integrity constraint was violated 
1882         pDb
->GetNextError(henv
, hdbc
, hstmtInsert
); 
1883         if (! wxStrcmp(pDb
->sqlState
, wxT("23000")))  // Integrity constraint violated 
1884             return(DB_ERR_INTEGRITY_CONSTRAINT_VIOL
); 
1887             pDb
->DispNextError(); 
1888             pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
); 
1892     if (retcode 
== SQL_NEED_DATA
) 
1895         retcode 
= SQLParamData(hstmtInsert
, &pParmID
); 
1896         while (retcode 
== SQL_NEED_DATA
) 
1898             // Find the parameter 
1900             for (i
=0; i 
< noCols
; i
++) 
1902                 if (colDefs
[i
].PtrDataObj 
== pParmID
) 
1904                     // We found it.  Store the parameter. 
1905                     retcode 
= SQLPutData(hstmtInsert
, pParmID
, colDefs
[i
].SzDataObj
); 
1906                     if (retcode 
!= SQL_SUCCESS
) 
1908                         pDb
->DispNextError(); 
1909                         pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
); 
1918     // Record inserted into the datasource successfully 
1921 }  // wxDbTable::Insert() 
1924 /********** wxDbTable::Update() **********/ 
1925 bool wxDbTable::Update(void) 
1927     wxASSERT(!queryOnly
); 
1933     // Build the SQL UPDATE statement 
1934     BuildUpdateStmt(sqlStmt
, DB_UPD_KEYFIELDS
); 
1936     pDb
->WriteSqlLog(sqlStmt
); 
1938 #ifdef DBDEBUG_CONSOLE 
1939     cout 
<< endl 
<< sqlStmt
.c_str() << endl 
<< endl
; 
1942     // Execute the SQL UPDATE statement 
1943     return(execUpdate(sqlStmt
)); 
1945 }  // wxDbTable::Update() 
1948 /********** wxDbTable::Update(pSqlStmt) **********/ 
1949 bool wxDbTable::Update(const wxString 
&pSqlStmt
) 
1951     wxASSERT(!queryOnly
); 
1955     pDb
->WriteSqlLog(pSqlStmt
); 
1957     return(execUpdate(pSqlStmt
)); 
1959 }  // wxDbTable::Update(pSqlStmt) 
1962 /********** wxDbTable::UpdateWhere() **********/ 
1963 bool wxDbTable::UpdateWhere(const wxString 
&pWhereClause
) 
1965     wxASSERT(!queryOnly
); 
1971     // Build the SQL UPDATE statement 
1972     BuildUpdateStmt(sqlStmt
, DB_UPD_WHERE
, pWhereClause
); 
1974     pDb
->WriteSqlLog(sqlStmt
); 
1976 #ifdef DBDEBUG_CONSOLE 
1977     cout 
<< endl 
<< sqlStmt
.c_str() << endl 
<< endl
; 
1980     // Execute the SQL UPDATE statement 
1981     return(execUpdate(sqlStmt
)); 
1983 }  // wxDbTable::UpdateWhere() 
1986 /********** wxDbTable::Delete() **********/ 
1987 bool wxDbTable::Delete(void) 
1989     wxASSERT(!queryOnly
); 
1996     // Build the SQL DELETE statement 
1997     BuildDeleteStmt(sqlStmt
, DB_DEL_KEYFIELDS
); 
1999     pDb
->WriteSqlLog(sqlStmt
); 
2001     // Execute the SQL DELETE statement 
2002     return(execDelete(sqlStmt
)); 
2004 }  // wxDbTable::Delete() 
2007 /********** wxDbTable::DeleteWhere() **********/ 
2008 bool wxDbTable::DeleteWhere(const wxString 
&pWhereClause
) 
2010     wxASSERT(!queryOnly
); 
2017     // Build the SQL DELETE statement 
2018     BuildDeleteStmt(sqlStmt
, DB_DEL_WHERE
, pWhereClause
); 
2020     pDb
->WriteSqlLog(sqlStmt
); 
2022     // Execute the SQL DELETE statement 
2023     return(execDelete(sqlStmt
)); 
2025 }  // wxDbTable::DeleteWhere() 
2028 /********** wxDbTable::DeleteMatching() **********/ 
2029 bool wxDbTable::DeleteMatching(void) 
2031     wxASSERT(!queryOnly
); 
2038     // Build the SQL DELETE statement 
2039     BuildDeleteStmt(sqlStmt
, DB_DEL_MATCHING
); 
2041     pDb
->WriteSqlLog(sqlStmt
); 
2043     // Execute the SQL DELETE statement 
2044     return(execDelete(sqlStmt
)); 
2046 }  // wxDbTable::DeleteMatching() 
2049 /********** wxDbTable::IsColNull() **********/ 
2050 bool wxDbTable::IsColNull(UWORD colNo
) const 
2053     This logic is just not right.  It would indicate TRUE 
2054     if a numeric field were set to a value of 0. 
2056     switch(colDefs[colNo].SqlCtype) 
2059             return(((UCHAR FAR *) colDefs[colNo].PtrDataObj)[0] == 0); 
2061             return((  *((SWORD *) colDefs[colNo].PtrDataObj))   == 0); 
2063             return((   *((UWORD*) colDefs[colNo].PtrDataObj))   == 0); 
2065             return(( *((SDWORD *) colDefs[colNo].PtrDataObj))   == 0); 
2067             return(( *((UDWORD *) colDefs[colNo].PtrDataObj))   == 0); 
2069             return(( *((SFLOAT *) colDefs[colNo].PtrDataObj))   == 0); 
2071             return((*((SDOUBLE *) colDefs[colNo].PtrDataObj))   == 0); 
2072         case SQL_C_TIMESTAMP: 
2073             TIMESTAMP_STRUCT *pDt; 
2074             pDt = (TIMESTAMP_STRUCT *) colDefs[colNo].PtrDataObj; 
2075             if (pDt->year == 0 && pDt->month == 0 && pDt->day == 0) 
2083     return (colDefs
[colNo
].Null
); 
2084 }  // wxDbTable::IsColNull() 
2087 /********** wxDbTable::CanSelectForUpdate() **********/ 
2088 bool wxDbTable::CanSelectForUpdate(void) 
2093     if (pDb
->Dbms() == dbmsMY_SQL
) 
2096     if ((pDb
->Dbms() == dbmsORACLE
) || 
2097         (pDb
->dbInf
.posStmts 
& SQL_PS_SELECT_FOR_UPDATE
)) 
2102 }  // wxDbTable::CanSelectForUpdate() 
2105 /********** wxDbTable::CanUpdByROWID() **********/ 
2106 bool wxDbTable::CanUpdByROWID(void) 
2109  * NOTE: Returning FALSE for now until this can be debugged, 
2110  *        as the ROWID is not getting updated correctly 
2114     if (pDb->Dbms() == dbmsORACLE) 
2119 }  // wxDbTable::CanUpdByROWID() 
2122 /********** wxDbTable::IsCursorClosedOnCommit() **********/ 
2123 bool wxDbTable::IsCursorClosedOnCommit(void) 
2125     if (pDb
->dbInf
.cursorCommitBehavior 
== SQL_CB_PRESERVE
) 
2130 }  // wxDbTable::IsCursorClosedOnCommit() 
2134 /********** wxDbTable::ClearMemberVar() **********/ 
2135 void wxDbTable::ClearMemberVar(UWORD colNo
, bool setToNull
) 
2137     wxASSERT(colNo 
< noCols
); 
2139     switch(colDefs
[colNo
].SqlCtype
) 
2142             ((UCHAR FAR 
*) colDefs
[colNo
].PtrDataObj
)[0]    = 0; 
2145             *((SWORD 
*) colDefs
[colNo
].PtrDataObj
)          = 0; 
2148             *((UWORD
*) colDefs
[colNo
].PtrDataObj
)           = 0; 
2151             *((SDWORD 
*) colDefs
[colNo
].PtrDataObj
)         = 0; 
2154             *((UDWORD 
*) colDefs
[colNo
].PtrDataObj
)         = 0; 
2157             *((SFLOAT 
*) colDefs
[colNo
].PtrDataObj
)         = 0.0f
; 
2160             *((SDOUBLE 
*) colDefs
[colNo
].PtrDataObj
)        = 0.0f
; 
2162         case SQL_C_TIMESTAMP
: 
2163             TIMESTAMP_STRUCT 
*pDt
; 
2164             pDt 
= (TIMESTAMP_STRUCT 
*) colDefs
[colNo
].PtrDataObj
; 
2177 }  // wxDbTable::ClearMemberVar() 
2180 /********** wxDbTable::ClearMemberVars() **********/ 
2181 void wxDbTable::ClearMemberVars(bool setToNull
) 
2185     // Loop through the columns setting each member variable to zero 
2186     for (i
=0; i 
< noCols
; i
++) 
2187         ClearMemberVar(i
,setToNull
); 
2189 }  // wxDbTable::ClearMemberVars() 
2192 /********** wxDbTable::SetQueryTimeout() **********/ 
2193 bool wxDbTable::SetQueryTimeout(UDWORD nSeconds
) 
2195     if (SQLSetStmtOption(hstmtInsert
, SQL_QUERY_TIMEOUT
, nSeconds
) != SQL_SUCCESS
) 
2196         return(pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
)); 
2197     if (SQLSetStmtOption(hstmtUpdate
, SQL_QUERY_TIMEOUT
, nSeconds
) != SQL_SUCCESS
) 
2198         return(pDb
->DispAllErrors(henv
, hdbc
, hstmtUpdate
)); 
2199     if (SQLSetStmtOption(hstmtDelete
, SQL_QUERY_TIMEOUT
, nSeconds
) != SQL_SUCCESS
) 
2200         return(pDb
->DispAllErrors(henv
, hdbc
, hstmtDelete
)); 
2201     if (SQLSetStmtOption(hstmtInternal
, SQL_QUERY_TIMEOUT
, nSeconds
) != SQL_SUCCESS
) 
2202         return(pDb
->DispAllErrors(henv
, hdbc
, hstmtInternal
)); 
2204     // Completed Successfully 
2207 }  // wxDbTable::SetQueryTimeout() 
2210 /********** wxDbTable::SetColDefs() **********/ 
2211 void wxDbTable::SetColDefs(UWORD index
, const wxString 
&fieldName
, int dataType
, void *pData
, 
2212                            SWORD cType
, int size
, bool keyField
, bool upd
, 
2213                            bool insAllow
, bool derivedCol
) 
2215     wxASSERT_MSG( index 
< noCols
, 
2216                   _T("Specified column index exceeds the maximum number of columns for this table.") ); 
2218     if (!colDefs
)  // May happen if the database connection fails 
2221     if (fieldName
.Length() > (unsigned int) DB_MAX_COLUMN_NAME_LEN
) 
2223         wxStrncpy(colDefs
[index
].ColName
, fieldName
, DB_MAX_COLUMN_NAME_LEN
); 
2224         colDefs
[index
].ColName
[DB_MAX_COLUMN_NAME_LEN
] = 0; 
2228         tmpMsg
.Printf(_T("Column name '%s' is too long. Truncated to '%s'."), 
2229                       fieldName
.c_str(),colDefs
[index
].ColName
); 
2231 #endif // __WXDEBUG__ 
2234         wxStrcpy(colDefs
[index
].ColName
, fieldName
); 
2236     colDefs
[index
].DbDataType       
= dataType
; 
2237     colDefs
[index
].PtrDataObj       
= pData
; 
2238     colDefs
[index
].SqlCtype         
= cType
; 
2239     colDefs
[index
].SzDataObj        
= size
; 
2240     colDefs
[index
].KeyField         
= keyField
; 
2241     colDefs
[index
].DerivedCol       
= derivedCol
; 
2242     // Derived columns by definition would NOT be "Insertable" or "Updateable" 
2245         colDefs
[index
].Updateable       
= FALSE
; 
2246         colDefs
[index
].InsertAllowed    
= FALSE
; 
2250         colDefs
[index
].Updateable       
= upd
; 
2251         colDefs
[index
].InsertAllowed    
= insAllow
; 
2254     colDefs
[index
].Null                 
= FALSE
; 
2256 }  // wxDbTable::SetColDefs() 
2259 /********** wxDbTable::SetColDefs() **********/ 
2260 wxDbColDataPtr
* wxDbTable::SetColDefs(wxDbColInf 
*pColInfs
, UWORD numCols
) 
2263     wxDbColDataPtr 
*pColDataPtrs 
= NULL
; 
2269         pColDataPtrs 
= new wxDbColDataPtr
[numCols
+1]; 
2271         for (index 
= 0; index 
< numCols
; index
++) 
2273             // Process the fields 
2274             switch (pColInfs
[index
].dbDataType
) 
2276                 case DB_DATA_TYPE_VARCHAR
: 
2277                    pColDataPtrs
[index
].PtrDataObj 
= new wxChar
[pColInfs
[index
].bufferLength
+1]; 
2278                    pColDataPtrs
[index
].SzDataObj  
= pColInfs
[index
].columnSize
; 
2279                    pColDataPtrs
[index
].SqlCtype   
= SQL_C_CHAR
; 
2281                 case DB_DATA_TYPE_INTEGER
: 
2282                     // Can be long or short 
2283                     if (pColInfs
[index
].bufferLength 
== sizeof(long)) 
2285                       pColDataPtrs
[index
].PtrDataObj 
= new long; 
2286                       pColDataPtrs
[index
].SzDataObj  
= sizeof(long); 
2287                       pColDataPtrs
[index
].SqlCtype   
= SQL_C_SLONG
; 
2291                         pColDataPtrs
[index
].PtrDataObj 
= new short; 
2292                         pColDataPtrs
[index
].SzDataObj  
= sizeof(short); 
2293                         pColDataPtrs
[index
].SqlCtype   
= SQL_C_SSHORT
; 
2296                 case DB_DATA_TYPE_FLOAT
: 
2297                     // Can be float or double 
2298                     if (pColInfs
[index
].bufferLength 
== sizeof(float)) 
2300                         pColDataPtrs
[index
].PtrDataObj 
= new float; 
2301                         pColDataPtrs
[index
].SzDataObj  
= sizeof(float); 
2302                         pColDataPtrs
[index
].SqlCtype   
= SQL_C_FLOAT
; 
2306                         pColDataPtrs
[index
].PtrDataObj 
= new double; 
2307                         pColDataPtrs
[index
].SzDataObj  
= sizeof(double); 
2308                         pColDataPtrs
[index
].SqlCtype   
= SQL_C_DOUBLE
; 
2311                 case DB_DATA_TYPE_DATE
: 
2312                     pColDataPtrs
[index
].PtrDataObj 
= new TIMESTAMP_STRUCT
; 
2313                     pColDataPtrs
[index
].SzDataObj  
= sizeof(TIMESTAMP_STRUCT
); 
2314                     pColDataPtrs
[index
].SqlCtype   
= SQL_C_TIMESTAMP
; 
2316                 case DB_DATA_TYPE_BLOB
: 
2317                     wxFAIL_MSG(wxT("This form of ::SetColDefs() cannot be used with BLOB columns")); 
2318                     pColDataPtrs
[index
].PtrDataObj 
= /*BLOB ADDITION NEEDED*/NULL
; 
2319                     pColDataPtrs
[index
].SzDataObj  
= /*BLOB ADDITION NEEDED*/sizeof(void *); 
2320                     pColDataPtrs
[index
].SqlCtype   
= SQL_VARBINARY
; 
2323             if (pColDataPtrs
[index
].PtrDataObj 
!= NULL
) 
2324                 SetColDefs (index
,pColInfs
[index
].colName
,pColInfs
[index
].dbDataType
, pColDataPtrs
[index
].PtrDataObj
, pColDataPtrs
[index
].SqlCtype
, pColDataPtrs
[index
].SzDataObj
); 
2327                 // Unable to build all the column definitions, as either one of 
2328                 // the calls to "new" failed above, or there was a BLOB field 
2329                 // to have a column definition for.  If BLOBs are to be used, 
2330                 // the other form of ::SetColDefs() must be used, as it is impossible 
2331                 // to know the maximum size to create the PtrDataObj to be. 
2332                 delete [] pColDataPtrs
; 
2338     return (pColDataPtrs
); 
2340 } // wxDbTable::SetColDefs() 
2343 /********** wxDbTable::SetCursor() **********/ 
2344 void wxDbTable::SetCursor(HSTMT 
*hstmtActivate
) 
2346     if (hstmtActivate 
== wxDB_DEFAULT_CURSOR
) 
2347         hstmt 
= *hstmtDefault
; 
2349         hstmt 
= *hstmtActivate
; 
2351 }  // wxDbTable::SetCursor() 
2354 /********** wxDbTable::Count(const wxString &) **********/ 
2355 ULONG 
wxDbTable::Count(const wxString 
&args
) 
2361     // Build a "SELECT COUNT(*) FROM queryTableName [WHERE whereClause]" SQL Statement 
2362     sqlStmt  
= wxT("SELECT COUNT("); 
2364     sqlStmt 
+= wxT(") FROM "); 
2365     sqlStmt 
+= pDb
->SQLTableName(queryTableName
); 
2366 //    sqlStmt += queryTableName; 
2367 #if wxODBC_BACKWARD_COMPATABILITY 
2368     if (from 
&& wxStrlen(from
)) 
2374     // Add the where clause if one is provided 
2375 #if wxODBC_BACKWARD_COMPATABILITY 
2376     if (where 
&& wxStrlen(where
)) 
2381         sqlStmt 
+= wxT(" WHERE "); 
2385     pDb
->WriteSqlLog(sqlStmt
); 
2387     // Initialize the Count cursor if it's not already initialized 
2390         hstmtCount 
= GetNewCursor(FALSE
,FALSE
); 
2391         wxASSERT(hstmtCount
); 
2396     // Execute the SQL statement 
2397     if (SQLExecDirect(*hstmtCount
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
) != SQL_SUCCESS
) 
2399         pDb
->DispAllErrors(henv
, hdbc
, *hstmtCount
); 
2404     if (SQLFetch(*hstmtCount
) != SQL_SUCCESS
) 
2406         pDb
->DispAllErrors(henv
, hdbc
, *hstmtCount
); 
2410     // Obtain the result 
2411     if (SQLGetData(*hstmtCount
, (UWORD
)1, SQL_C_ULONG
, &count
, sizeof(count
), &cb
) != SQL_SUCCESS
) 
2413         pDb
->DispAllErrors(henv
, hdbc
, *hstmtCount
); 
2418     if (SQLFreeStmt(*hstmtCount
, SQL_CLOSE
) != SQL_SUCCESS
) 
2419         pDb
->DispAllErrors(henv
, hdbc
, *hstmtCount
); 
2421     // Return the record count 
2424 }  // wxDbTable::Count() 
2427 /********** wxDbTable::Refresh() **********/ 
2428 bool wxDbTable::Refresh(void) 
2432     // Switch to the internal cursor so any active cursors are not corrupted 
2433     HSTMT currCursor 
= GetCursor(); 
2434     hstmt 
= hstmtInternal
; 
2435 #if wxODBC_BACKWARD_COMPATABILITY 
2436     // Save the where and order by clauses 
2437     wxChar 
*saveWhere 
= where
; 
2438     wxChar 
*saveOrderBy 
= orderBy
; 
2440     wxString saveWhere 
= where
; 
2441     wxString saveOrderBy 
= orderBy
; 
2443     // Build a where clause to refetch the record with.  Try and use the 
2444     // ROWID if it's available, ow use the key fields. 
2445     wxString whereClause
; 
2446     whereClause
.Empty(); 
2448     if (CanUpdByROWID()) 
2451         wxChar   rowid
[wxDB_ROWID_LEN
+1]; 
2453         // Get the ROWID value.  If not successful retreiving the ROWID, 
2454         // simply fall down through the code and build the WHERE clause 
2455         // based on the key fields. 
2456         if (SQLGetData(hstmt
, (UWORD
)(noCols
+1), SQL_C_CHAR
, (UCHAR
*) rowid
, wxDB_ROWID_LEN
, &cb
) == SQL_SUCCESS
) 
2458             whereClause 
+= pDb
->SQLTableName(queryTableName
); 
2459 //            whereClause += queryTableName; 
2460             whereClause 
+= wxT(".ROWID = '"); 
2461             whereClause 
+= rowid
; 
2462             whereClause 
+= wxT("'"); 
2466     // If unable to use the ROWID, build a where clause from the keyfields 
2467     if (wxStrlen(whereClause
) == 0) 
2468         BuildWhereClause(whereClause
, DB_WHERE_KEYFIELDS
, queryTableName
); 
2470     // Requery the record 
2471     where 
= whereClause
; 
2476     if (result 
&& !GetNext()) 
2479     // Switch back to original cursor 
2480     SetCursor(&currCursor
); 
2482     // Free the internal cursor 
2483     if (SQLFreeStmt(hstmtInternal
, SQL_CLOSE
) != SQL_SUCCESS
) 
2484         pDb
->DispAllErrors(henv
, hdbc
, hstmtInternal
); 
2486     // Restore the original where and order by clauses 
2488     orderBy 
= saveOrderBy
; 
2492 }  // wxDbTable::Refresh() 
2495 /********** wxDbTable::SetColNull() **********/ 
2496 bool wxDbTable::SetColNull(UWORD colNo
, bool set
) 
2500         colDefs
[colNo
].Null 
= set
; 
2501         if (set
)  // Blank out the values in the member variable 
2503            colDefs
[colNo
].CbValue 
= SQL_NULL_DATA
; // SF PATCH#766404 
2504            ClearMemberVar(colNo
,FALSE
);  // Must call with FALSE, or infinite recursion will happen 
2511 }  // wxDbTable::SetColNull() 
2514 /********** wxDbTable::SetColNull() **********/ 
2515 bool wxDbTable::SetColNull(const wxString 
&colName
, bool set
) 
2518     for (colNo 
= 0; colNo 
< noCols
; colNo
++) 
2520         if (!wxStricmp(colName
, colDefs
[colNo
].ColName
)) 
2526         colDefs
[colNo
].Null 
= set
; 
2527         if (set
)  // Blank out the values in the member variable 
2529            colDefs
[colNo
].CbValue 
= SQL_NULL_DATA
;  // SF PATCH#766404 
2530            ClearMemberVar(colNo
,FALSE
);  // Must call with FALSE, or infinite recursion will happen 
2537 }  // wxDbTable::SetColNull() 
2540 /********** wxDbTable::GetNewCursor() **********/ 
2541 HSTMT 
*wxDbTable::GetNewCursor(bool setCursor
, bool bindColumns
) 
2543     HSTMT 
*newHSTMT 
= new HSTMT
; 
2548     if (SQLAllocStmt(hdbc
, newHSTMT
) != SQL_SUCCESS
) 
2550         pDb
->DispAllErrors(henv
, hdbc
); 
2555     if (SQLSetStmtOption(*newHSTMT
, SQL_CURSOR_TYPE
, cursorType
) != SQL_SUCCESS
) 
2557         pDb
->DispAllErrors(henv
, hdbc
, *newHSTMT
); 
2564         if (!bindCols(*newHSTMT
)) 
2572         SetCursor(newHSTMT
); 
2576 }   // wxDbTable::GetNewCursor() 
2579 /********** wxDbTable::DeleteCursor() **********/ 
2580 bool wxDbTable::DeleteCursor(HSTMT 
*hstmtDel
) 
2584     if (!hstmtDel
)  // Cursor already deleted 
2588 ODBC 3.0 says to use this form 
2589     if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS) 
2592     if (SQLFreeStmt(*hstmtDel
, SQL_DROP
) != SQL_SUCCESS
) 
2594         pDb
->DispAllErrors(henv
, hdbc
); 
2602 }  // wxDbTable::DeleteCursor() 
2604 ////////////////////////////////////////////////////////////// 
2605 // wxDbGrid support functions 
2606 ////////////////////////////////////////////////////////////// 
2608 void wxDbTable::SetRowMode(const rowmode_t rowmode
) 
2610     if (!m_hstmtGridQuery
) 
2612         m_hstmtGridQuery 
= GetNewCursor(FALSE
,FALSE
); 
2613         if (!bindCols(*m_hstmtGridQuery
)) 
2617     m_rowmode 
= rowmode
; 
2620         case WX_ROW_MODE_QUERY
: 
2621             SetCursor(m_hstmtGridQuery
); 
2623         case WX_ROW_MODE_INDIVIDUAL
: 
2624             SetCursor(hstmtDefault
); 
2629 }  // wxDbTable::SetRowMode() 
2632 wxVariant 
wxDbTable::GetCol(const int colNo
) const 
2635     if ((colNo 
< noCols
) && (!IsColNull(colNo
))) 
2637         switch (colDefs
[colNo
].SqlCtype
) 
2641                 val 
= (wxChar 
*)(colDefs
[colNo
].PtrDataObj
); 
2645                 val 
= *(long *)(colDefs
[colNo
].PtrDataObj
); 
2649                 val 
= (long int )(*(short *)(colDefs
[colNo
].PtrDataObj
)); 
2652                 val 
= (long)(*(unsigned long *)(colDefs
[colNo
].PtrDataObj
)); 
2655                 val 
= (long)(*(wxChar 
*)(colDefs
[colNo
].PtrDataObj
)); 
2657             case SQL_C_UTINYINT
: 
2658                 val 
= (long)(*(wxChar 
*)(colDefs
[colNo
].PtrDataObj
)); 
2661                 val 
= (long)(*(UWORD 
*)(colDefs
[colNo
].PtrDataObj
)); 
2664                 val 
= (DATE_STRUCT 
*)(colDefs
[colNo
].PtrDataObj
); 
2667                 val 
= (TIME_STRUCT 
*)(colDefs
[colNo
].PtrDataObj
); 
2669             case SQL_C_TIMESTAMP
: 
2670                 val 
= (TIMESTAMP_STRUCT 
*)(colDefs
[colNo
].PtrDataObj
); 
2673                 val 
= *(double *)(colDefs
[colNo
].PtrDataObj
); 
2680 }  // wxDbTable::GetCol() 
2683 void wxDbTable::SetCol(const int colNo
, const wxVariant val
) 
2685     //FIXME: Add proper wxDateTime support to wxVariant.. 
2688     SetColNull(colNo
, val
.IsNull()); 
2692         if ((colDefs
[colNo
].SqlCtype 
== SQL_C_DATE
) 
2693             || (colDefs
[colNo
].SqlCtype 
== SQL_C_TIME
) 
2694             || (colDefs
[colNo
].SqlCtype 
== SQL_C_TIMESTAMP
)) 
2696             //Returns null if invalid! 
2697             if (!dateval
.ParseDate(val
.GetString())) 
2698                 SetColNull(colNo
, TRUE
); 
2701         switch (colDefs
[colNo
].SqlCtype
) 
2705                 csstrncpyt((wxChar 
*)(colDefs
[colNo
].PtrDataObj
), 
2706                            val
.GetString().c_str(), 
2707                            colDefs
[colNo
].SzDataObj
-1); 
2711                 *(long *)(colDefs
[colNo
].PtrDataObj
) = val
; 
2715                 *(short *)(colDefs
[colNo
].PtrDataObj
) = val
.GetLong(); 
2718                 *(unsigned long *)(colDefs
[colNo
].PtrDataObj
) = val
.GetLong(); 
2721                 *(wxChar 
*)(colDefs
[colNo
].PtrDataObj
) = val
.GetChar(); 
2723             case SQL_C_UTINYINT
: 
2724                 *(wxChar 
*)(colDefs
[colNo
].PtrDataObj
) = val
.GetChar(); 
2727                 *(unsigned short *)(colDefs
[colNo
].PtrDataObj
) = val
.GetLong(); 
2729             //FIXME: Add proper wxDateTime support to wxVariant.. 
2732                     DATE_STRUCT 
*dataptr 
= 
2733                         (DATE_STRUCT 
*)colDefs
[colNo
].PtrDataObj
; 
2735                     dataptr
->year   
= dateval
.GetYear(); 
2736                     dataptr
->month  
= dateval
.GetMonth()+1; 
2737                     dataptr
->day    
= dateval
.GetDay(); 
2742                     TIME_STRUCT 
*dataptr 
= 
2743                         (TIME_STRUCT 
*)colDefs
[colNo
].PtrDataObj
; 
2745                     dataptr
->hour   
= dateval
.GetHour(); 
2746                     dataptr
->minute 
= dateval
.GetMinute(); 
2747                     dataptr
->second 
= dateval
.GetSecond(); 
2750             case SQL_C_TIMESTAMP
: 
2752                     TIMESTAMP_STRUCT 
*dataptr 
= 
2753                         (TIMESTAMP_STRUCT 
*)colDefs
[colNo
].PtrDataObj
; 
2754                     dataptr
->year   
= dateval
.GetYear(); 
2755                     dataptr
->month  
= dateval
.GetMonth()+1; 
2756                     dataptr
->day    
= dateval
.GetDay(); 
2758                     dataptr
->hour   
= dateval
.GetHour(); 
2759                     dataptr
->minute 
= dateval
.GetMinute(); 
2760                     dataptr
->second 
= dateval
.GetSecond(); 
2764                 *(double *)(colDefs
[colNo
].PtrDataObj
) = val
; 
2769     }  // if (!val.IsNull()) 
2770 }  // wxDbTable::SetCol() 
2773 GenericKey 
wxDbTable::GetKey() 
2778     blk 
= malloc(m_keysize
); 
2779     blkptr 
= (wxChar 
*) blk
; 
2782     for (i
=0; i 
< noCols
; i
++) 
2784         if (colDefs
[i
].KeyField
) 
2786             memcpy(blkptr
,colDefs
[i
].PtrDataObj
, colDefs
[i
].SzDataObj
); 
2787             blkptr 
+= colDefs
[i
].SzDataObj
; 
2791     GenericKey k 
= GenericKey(blk
, m_keysize
); 
2795 }  // wxDbTable::GetKey() 
2798 void wxDbTable::SetKey(const GenericKey
& k
) 
2804     blkptr 
= (wxChar 
*)blk
; 
2807     for (i
=0; i 
< noCols
; i
++) 
2809         if (colDefs
[i
].KeyField
) 
2811             SetColNull(i
, FALSE
); 
2812             memcpy(colDefs
[i
].PtrDataObj
, blkptr
, colDefs
[i
].SzDataObj
); 
2813             blkptr 
+= colDefs
[i
].SzDataObj
; 
2816 }  // wxDbTable::SetKey() 
2819 #endif  // wxUSE_ODBC