1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/common/dbtable.cpp 
   3 // Purpose:     Implementation of the wxDbTable class. 
   5 // Modified by: George Tasker 
  10 // Copyright:   (c) 1996 Remstar International, Inc. 
  11 // Licence:     wxWindows licence 
  12 /////////////////////////////////////////////////////////////////////////////// 
  14 #include  "wx/wxprec.h" 
  23     #include "wx/object.h" 
  25     #include "wx/string.h" 
  30 #ifdef DBDEBUG_CONSOLE 
  36     #include "wx/ioswrap.h" 
  39 #include "wx/filefn.h" 
  45 #include "wx/dbtable.h" 
  48 // The HPUX preprocessor lines below were commented out on 8/20/97 
  49 // because macros.h currently redefines DEBUG and is unneeded. 
  51 // #    include <macros.h> 
  54 #    include <sys/minmax.h> 
  58 ULONG lastTableID 
= 0; 
  66 void csstrncpyt(wxChar 
*target
, const wxChar 
*source
, int n
) 
  68     while ( (*target
++ = *source
++) != '\0' && --n 
!= 0 ) 
  76 /********** wxDbColDef::wxDbColDef() Constructor **********/ 
  77 wxDbColDef::wxDbColDef() 
  83 bool wxDbColDef::Initialize() 
  86     DbDataType      
= DB_DATA_TYPE_INTEGER
; 
  87     SqlCtype        
= SQL_C_LONG
; 
  92     InsertAllowed   
= false; 
  98 }  // wxDbColDef::Initialize() 
 101 /********** wxDbTable::wxDbTable() Constructor **********/ 
 102 wxDbTable::wxDbTable(wxDb 
*pwxDb
, const wxString 
&tblName
, const UWORD numColumns
, 
 103                     const wxString 
&qryTblName
, bool qryOnly
, const wxString 
&tblPath
) 
 105     if (!initialize(pwxDb
, tblName
, numColumns
, qryTblName
, qryOnly
, tblPath
)) 
 107 }  // wxDbTable::wxDbTable() 
 110 /***** DEPRECATED: use wxDbTable::wxDbTable() format above *****/ 
 111 #if WXWIN_COMPATIBILITY_2_4 
 112 wxDbTable::wxDbTable(wxDb 
*pwxDb
, const wxString 
&tblName
, const UWORD numColumns
, 
 113                     const wxChar 
*qryTblName
, bool qryOnly
, const wxString 
&tblPath
) 
 115     wxString tempQryTblName
; 
 116     tempQryTblName 
= qryTblName
; 
 117     if (!initialize(pwxDb
, tblName
, numColumns
, tempQryTblName
, qryOnly
, tblPath
)) 
 119 }  // wxDbTable::wxDbTable() 
 120 #endif // WXWIN_COMPATIBILITY_2_4 
 123 /********** wxDbTable::~wxDbTable() **********/ 
 124 wxDbTable::~wxDbTable() 
 127 }  // wxDbTable::~wxDbTable() 
 130 bool wxDbTable::initialize(wxDb 
*pwxDb
, const wxString 
&tblName
, const UWORD numColumns
, 
 131                     const wxString 
&qryTblName
, bool qryOnly
, const wxString 
&tblPath
) 
 133     // Initializing member variables 
 134     pDb                 
= pwxDb
;                    // Pointer to the wxDb object 
 138     m_hstmtGridQuery               
= 0; 
 139     hstmtDefault        
= 0;                        // Initialized below 
 140     hstmtCount          
= 0;                        // Initialized first time it is needed 
 147     m_numCols           
= numColumns
;               // Number of columns in the table 
 148     where
.Empty();                                  // Where clause 
 149     orderBy
.Empty();                                // Order By clause 
 150     from
.Empty();                                   // From clause 
 151     selectForUpdate     
= false;                    // SELECT ... FOR UPDATE; Indicates whether to include the FOR UPDATE phrase 
 156     queryTableName
.Empty(); 
 158     wxASSERT(tblName
.length()); 
 164     tableName 
= tblName
;                        // Table Name 
 165     if ((pDb
->Dbms() == dbmsORACLE
) || 
 166         (pDb
->Dbms() == dbmsFIREBIRD
) || 
 167         (pDb
->Dbms() == dbmsINTERBASE
)) 
 168         tableName 
= tableName
.Upper(); 
 170     if (tblPath
.length()) 
 171         tablePath 
= tblPath
;                    // Table Path - used for dBase files 
 175     if (qryTblName
.length())                    // Name of the table/view to query 
 176         queryTableName 
= qryTblName
; 
 178         queryTableName 
= tblName
; 
 180     if ((pDb
->Dbms() == dbmsORACLE
) || 
 181         (pDb
->Dbms() == dbmsFIREBIRD
) || 
 182         (pDb
->Dbms() == dbmsINTERBASE
)) 
 183         queryTableName 
= queryTableName
.Upper(); 
 185     pDb
->incrementTableCount(); 
 188     tableID 
= ++lastTableID
; 
 189     s
.Printf(wxT("wxDbTable constructor (%-20s) tableID:[%6lu] pDb:[%p]"), 
 190              tblName
.c_str(), tableID
, wx_static_cast(void*, pDb
)); 
 193     wxTablesInUse 
*tableInUse
; 
 194     tableInUse            
= new wxTablesInUse(); 
 195     tableInUse
->tableName 
= tblName
; 
 196     tableInUse
->tableID   
= tableID
; 
 197     tableInUse
->pDb       
= pDb
; 
 198     TablesInUse
.Append(tableInUse
); 
 203     // Grab the HENV and HDBC from the wxDb object 
 204     henv 
= pDb
->GetHENV(); 
 205     hdbc 
= pDb
->GetHDBC(); 
 207     // Allocate space for column definitions 
 209         colDefs 
= new wxDbColDef
[m_numCols
];  // Points to the first column definition 
 211     // Allocate statement handles for the table 
 214         // Allocate a separate statement handle for performing inserts 
 215         if (SQLAllocStmt(hdbc
, &hstmtInsert
) != SQL_SUCCESS
) 
 216             pDb
->DispAllErrors(henv
, hdbc
); 
 217         // Allocate a separate statement handle for performing deletes 
 218         if (SQLAllocStmt(hdbc
, &hstmtDelete
) != SQL_SUCCESS
) 
 219             pDb
->DispAllErrors(henv
, hdbc
); 
 220         // Allocate a separate statement handle for performing updates 
 221         if (SQLAllocStmt(hdbc
, &hstmtUpdate
) != SQL_SUCCESS
) 
 222             pDb
->DispAllErrors(henv
, hdbc
); 
 224     // Allocate a separate statement handle for internal use 
 225     if (SQLAllocStmt(hdbc
, &hstmtInternal
) != SQL_SUCCESS
) 
 226         pDb
->DispAllErrors(henv
, hdbc
); 
 228     // Set the cursor type for the statement handles 
 229     cursorType 
= SQL_CURSOR_STATIC
; 
 231     if (SQLSetStmtOption(hstmtInternal
, SQL_CURSOR_TYPE
, cursorType
) != SQL_SUCCESS
) 
 233         // Check to see if cursor type is supported 
 234         pDb
->GetNextError(henv
, hdbc
, hstmtInternal
); 
 235         if (! wxStrcmp(pDb
->sqlState
, wxT("01S02")))  // Option Value Changed 
 237             // Datasource does not support static cursors.  Driver 
 238             // will substitute a cursor type.  Call SQLGetStmtOption() 
 239             // to determine which cursor type was selected. 
 240             if (SQLGetStmtOption(hstmtInternal
, SQL_CURSOR_TYPE
, &cursorType
) != SQL_SUCCESS
) 
 241                 pDb
->DispAllErrors(henv
, hdbc
, hstmtInternal
); 
 242 #ifdef DBDEBUG_CONSOLE 
 243             cout 
<< wxT("Static cursor changed to: "); 
 246             case SQL_CURSOR_FORWARD_ONLY
: 
 247                 cout 
<< wxT("Forward Only"); 
 249             case SQL_CURSOR_STATIC
: 
 250                 cout 
<< wxT("Static"); 
 252             case SQL_CURSOR_KEYSET_DRIVEN
: 
 253                 cout 
<< wxT("Keyset Driven"); 
 255             case SQL_CURSOR_DYNAMIC
: 
 256                 cout 
<< wxT("Dynamic"); 
 259             cout 
<< endl 
<< endl
; 
 262             if (pDb
->FwdOnlyCursors() && cursorType 
!= SQL_CURSOR_FORWARD_ONLY
) 
 264                 // Force the use of a forward only cursor... 
 265                 cursorType 
= SQL_CURSOR_FORWARD_ONLY
; 
 266                 if (SQLSetStmtOption(hstmtInternal
, SQL_CURSOR_TYPE
, cursorType
) != SQL_SUCCESS
) 
 268                     // Should never happen 
 269                     pDb
->GetNextError(henv
, hdbc
, hstmtInternal
); 
 276             pDb
->DispNextError(); 
 277             pDb
->DispAllErrors(henv
, hdbc
, hstmtInternal
); 
 280 #ifdef DBDEBUG_CONSOLE 
 282         cout 
<< wxT("Cursor Type set to STATIC") << endl 
<< endl
; 
 287         // Set the cursor type for the INSERT statement handle 
 288         if (SQLSetStmtOption(hstmtInsert
, SQL_CURSOR_TYPE
, SQL_CURSOR_FORWARD_ONLY
) != SQL_SUCCESS
) 
 289             pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
); 
 290         // Set the cursor type for the DELETE statement handle 
 291         if (SQLSetStmtOption(hstmtDelete
, SQL_CURSOR_TYPE
, SQL_CURSOR_FORWARD_ONLY
) != SQL_SUCCESS
) 
 292             pDb
->DispAllErrors(henv
, hdbc
, hstmtDelete
); 
 293         // Set the cursor type for the UPDATE statement handle 
 294         if (SQLSetStmtOption(hstmtUpdate
, SQL_CURSOR_TYPE
, SQL_CURSOR_FORWARD_ONLY
) != SQL_SUCCESS
) 
 295             pDb
->DispAllErrors(henv
, hdbc
, hstmtUpdate
); 
 298     // Make the default cursor the active cursor 
 299     hstmtDefault 
= GetNewCursor(false,false); 
 300     wxASSERT(hstmtDefault
); 
 301     hstmt 
= *hstmtDefault
; 
 305 }  // wxDbTable::initialize() 
 308 void wxDbTable::cleanup() 
 313         s
.Printf(wxT("wxDbTable destructor (%-20s) tableID:[%6lu] pDb:[%p]"), 
 314                  tableName
.c_str(), tableID
, wx_static_cast(void*, pDb
)); 
 323         wxList::compatibility_iterator pNode
; 
 324         pNode 
= TablesInUse
.GetFirst(); 
 325         while (pNode 
&& !found
) 
 327             if (((wxTablesInUse 
*)pNode
->GetData())->tableID 
== tableID
) 
 330                 delete (wxTablesInUse 
*)pNode
->GetData(); 
 331                 TablesInUse
.Erase(pNode
); 
 334                 pNode 
= pNode
->GetNext(); 
 339             msg
.Printf(wxT("Unable to find the tableID in the linked\nlist of tables in use.\n\n%s"),s
.c_str()); 
 340             wxLogDebug (msg
,wxT("NOTICE...")); 
 345     // Decrement the wxDb table count 
 347         pDb
->decrementTableCount(); 
 349     // Delete memory allocated for column definitions 
 353     // Free statement handles 
 359 ODBC 3.0 says to use this form 
 360             if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS) 
 362             if (SQLFreeStmt(hstmtInsert
, SQL_DROP
) != SQL_SUCCESS
) 
 363                 pDb
->DispAllErrors(henv
, hdbc
); 
 369 ODBC 3.0 says to use this form 
 370             if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS) 
 372             if (SQLFreeStmt(hstmtDelete
, SQL_DROP
) != SQL_SUCCESS
) 
 373                 pDb
->DispAllErrors(henv
, hdbc
); 
 379 ODBC 3.0 says to use this form 
 380             if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS) 
 382             if (SQLFreeStmt(hstmtUpdate
, SQL_DROP
) != SQL_SUCCESS
) 
 383                 pDb
->DispAllErrors(henv
, hdbc
); 
 389         if (SQLFreeStmt(hstmtInternal
, SQL_DROP
) != SQL_SUCCESS
) 
 390             pDb
->DispAllErrors(henv
, hdbc
); 
 393     // Delete dynamically allocated cursors 
 395         DeleteCursor(hstmtDefault
); 
 398         DeleteCursor(hstmtCount
); 
 400     if (m_hstmtGridQuery
) 
 401         DeleteCursor(m_hstmtGridQuery
); 
 403 }  // wxDbTable::cleanup() 
 406 /***************************** PRIVATE FUNCTIONS *****************************/ 
 409 void wxDbTable::setCbValueForColumn(int columnIndex
) 
 411     switch(colDefs
[columnIndex
].DbDataType
) 
 413         case DB_DATA_TYPE_VARCHAR
: 
 414         case DB_DATA_TYPE_MEMO
: 
 415             if (colDefs
[columnIndex
].Null
) 
 416                 colDefs
[columnIndex
].CbValue 
= SQL_NULL_DATA
; 
 418                 colDefs
[columnIndex
].CbValue 
= SQL_NTS
; 
 420         case DB_DATA_TYPE_INTEGER
: 
 421             if (colDefs
[columnIndex
].Null
) 
 422                 colDefs
[columnIndex
].CbValue 
= SQL_NULL_DATA
; 
 424                 colDefs
[columnIndex
].CbValue 
= 0; 
 426         case DB_DATA_TYPE_FLOAT
: 
 427             if (colDefs
[columnIndex
].Null
) 
 428                 colDefs
[columnIndex
].CbValue 
= SQL_NULL_DATA
; 
 430                 colDefs
[columnIndex
].CbValue 
= 0; 
 432         case DB_DATA_TYPE_DATE
: 
 433             if (colDefs
[columnIndex
].Null
) 
 434                 colDefs
[columnIndex
].CbValue 
= SQL_NULL_DATA
; 
 436                 colDefs
[columnIndex
].CbValue 
= 0; 
 438         case DB_DATA_TYPE_BLOB
: 
 439             if (colDefs
[columnIndex
].Null
) 
 440                 colDefs
[columnIndex
].CbValue 
= SQL_NULL_DATA
; 
 442                 if (colDefs
[columnIndex
].SqlCtype 
== SQL_C_WXCHAR
) 
 443                     colDefs
[columnIndex
].CbValue 
= SQL_NTS
; 
 445                     colDefs
[columnIndex
].CbValue 
= SQL_LEN_DATA_AT_EXEC(colDefs
[columnIndex
].SzDataObj
); 
 450 /********** wxDbTable::bindParams() **********/ 
 451 bool wxDbTable::bindParams(bool forUpdate
) 
 453     wxASSERT(!queryOnly
); 
 458     SDWORD  precision   
= 0; 
 461     // Bind each column of the table that should be bound 
 462     // to a parameter marker 
 466     for (i
=0, colNumber
=1; i 
< m_numCols
; i
++) 
 470             if (!colDefs
[i
].Updateable
) 
 475             if (!colDefs
[i
].InsertAllowed
) 
 479         switch(colDefs
[i
].DbDataType
) 
 481             case DB_DATA_TYPE_VARCHAR
: 
 482                 fSqlType 
= pDb
->GetTypeInfVarchar().FsqlType
; 
 483                 precision 
= colDefs
[i
].SzDataObj
; 
 486             case DB_DATA_TYPE_MEMO
: 
 487                 fSqlType 
= pDb
->GetTypeInfMemo().FsqlType
; 
 488                 precision 
= colDefs
[i
].SzDataObj
; 
 491             case DB_DATA_TYPE_INTEGER
: 
 492                 fSqlType 
= pDb
->GetTypeInfInteger().FsqlType
; 
 493                 precision 
= pDb
->GetTypeInfInteger().Precision
; 
 496             case DB_DATA_TYPE_FLOAT
: 
 497                 fSqlType 
= pDb
->GetTypeInfFloat().FsqlType
; 
 498                 precision 
= pDb
->GetTypeInfFloat().Precision
; 
 499                 scale 
= pDb
->GetTypeInfFloat().MaximumScale
; 
 500                 // SQL Sybase Anywhere v5.5 returned a negative number for the 
 501                 // MaxScale.  This caused ODBC to kick out an error on ibscale. 
 502                 // I check for this here and set the scale = precision. 
 504                 // scale = (short) precision; 
 506             case DB_DATA_TYPE_DATE
: 
 507                 fSqlType 
= pDb
->GetTypeInfDate().FsqlType
; 
 508                 precision 
= pDb
->GetTypeInfDate().Precision
; 
 511             case DB_DATA_TYPE_BLOB
: 
 512                 fSqlType 
= pDb
->GetTypeInfBlob().FsqlType
; 
 513                 precision 
= colDefs
[i
].SzDataObj
; 
 518         setCbValueForColumn(i
); 
 522             if (SQLBindParameter(hstmtUpdate
, colNumber
++, SQL_PARAM_INPUT
, colDefs
[i
].SqlCtype
, 
 523                                  fSqlType
, precision
, scale
, (UCHAR
*) colDefs
[i
].PtrDataObj
, 
 524                                  precision
+1, &colDefs
[i
].CbValue
) != SQL_SUCCESS
) 
 526                 return(pDb
->DispAllErrors(henv
, hdbc
, hstmtUpdate
)); 
 531             if (SQLBindParameter(hstmtInsert
, colNumber
++, SQL_PARAM_INPUT
, colDefs
[i
].SqlCtype
, 
 532                                  fSqlType
, precision
, scale
, (UCHAR
*) colDefs
[i
].PtrDataObj
, 
 533                                  precision
+1, &colDefs
[i
].CbValue
) != SQL_SUCCESS
) 
 535                 return(pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
)); 
 540     // Completed successfully 
 543 }  // wxDbTable::bindParams() 
 546 /********** wxDbTable::bindInsertParams() **********/ 
 547 bool wxDbTable::bindInsertParams(void) 
 549     return bindParams(false); 
 550 }  // wxDbTable::bindInsertParams() 
 553 /********** wxDbTable::bindUpdateParams() **********/ 
 554 bool wxDbTable::bindUpdateParams(void) 
 556     return bindParams(true); 
 557 }  // wxDbTable::bindUpdateParams() 
 560 /********** wxDbTable::bindCols() **********/ 
 561 bool wxDbTable::bindCols(HSTMT cursor
) 
 565     // Bind each column of the table to a memory address for fetching data 
 567     for (i 
= 0; i 
< m_numCols
; i
++) 
 569         cb 
= colDefs
[i
].CbValue
; 
 570         if (SQLBindCol(cursor
, (UWORD
)(i
+1), colDefs
[i
].SqlCtype
, (UCHAR
*) colDefs
[i
].PtrDataObj
, 
 571                        colDefs
[i
].SzDataObj
, &cb 
) != SQL_SUCCESS
) 
 572           return (pDb
->DispAllErrors(henv
, hdbc
, cursor
)); 
 575     // Completed successfully 
 578 }  // wxDbTable::bindCols() 
 581 /********** wxDbTable::getRec() **********/ 
 582 bool wxDbTable::getRec(UWORD fetchType
) 
 586     if (!pDb
->FwdOnlyCursors()) 
 588         // Fetch the NEXT, PREV, FIRST or LAST record, depending on fetchType 
 589         SQLULEN cRowsFetched
; 
 592         retcode 
= SQLExtendedFetch(hstmt
, fetchType
, 0, &cRowsFetched
, &rowStatus
); 
 593         if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
 595             if (retcode 
== SQL_NO_DATA_FOUND
) 
 598                 return(pDb
->DispAllErrors(henv
, hdbc
, hstmt
)); 
 602             // Set the Null member variable to indicate the Null state 
 603             // of each column just read in. 
 605             for (i 
= 0; i 
< m_numCols
; i
++) 
 606                 colDefs
[i
].Null 
= (colDefs
[i
].CbValue 
== SQL_NULL_DATA
); 
 611         // Fetch the next record from the record set 
 612         retcode 
= SQLFetch(hstmt
); 
 613         if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
 615             if (retcode 
== SQL_NO_DATA_FOUND
) 
 618                 return(pDb
->DispAllErrors(henv
, hdbc
, hstmt
)); 
 622             // Set the Null member variable to indicate the Null state 
 623             // of each column just read in. 
 625             for (i 
= 0; i 
< m_numCols
; i
++) 
 626                 colDefs
[i
].Null 
= (colDefs
[i
].CbValue 
== SQL_NULL_DATA
); 
 630     // Completed successfully 
 633 }  // wxDbTable::getRec() 
 636 /********** wxDbTable::execDelete() **********/ 
 637 bool wxDbTable::execDelete(const wxString 
&pSqlStmt
) 
 641     // Execute the DELETE statement 
 642     retcode 
= SQLExecDirect(hstmtDelete
, (SQLTCHAR FAR 
*) pSqlStmt
.c_str(), SQL_NTS
); 
 644     if (retcode 
== SQL_SUCCESS 
|| 
 645         retcode 
== SQL_NO_DATA_FOUND 
|| 
 646         retcode 
== SQL_SUCCESS_WITH_INFO
) 
 648         // Record deleted successfully 
 652     // Problem deleting record 
 653     return(pDb
->DispAllErrors(henv
, hdbc
, hstmtDelete
)); 
 655 }  // wxDbTable::execDelete() 
 658 /********** wxDbTable::execUpdate() **********/ 
 659 bool wxDbTable::execUpdate(const wxString 
&pSqlStmt
) 
 663     // Execute the UPDATE statement 
 664     retcode 
= SQLExecDirect(hstmtUpdate
, (SQLTCHAR FAR 
*) pSqlStmt
.c_str(), SQL_NTS
); 
 666     if (retcode 
== SQL_SUCCESS 
|| 
 667         retcode 
== SQL_NO_DATA_FOUND 
|| 
 668         retcode 
== SQL_SUCCESS_WITH_INFO
) 
 670         // Record updated successfully 
 673     else if (retcode 
== SQL_NEED_DATA
) 
 676         retcode 
= SQLParamData(hstmtUpdate
, &pParmID
); 
 677         while (retcode 
== SQL_NEED_DATA
) 
 679             // Find the parameter 
 681             for (i
=0; i 
< m_numCols
; i
++) 
 683                 if (colDefs
[i
].PtrDataObj 
== pParmID
) 
 685                     // We found it.  Store the parameter. 
 686                     retcode 
= SQLPutData(hstmtUpdate
, pParmID
, colDefs
[i
].SzDataObj
); 
 687                     if (retcode 
!= SQL_SUCCESS
) 
 689                         pDb
->DispNextError(); 
 690                         return pDb
->DispAllErrors(henv
, hdbc
, hstmtUpdate
); 
 695             retcode 
= SQLParamData(hstmtUpdate
, &pParmID
); 
 697         if (retcode 
== SQL_SUCCESS 
|| 
 698             retcode 
== SQL_NO_DATA_FOUND 
|| 
 699             retcode 
== SQL_SUCCESS_WITH_INFO
) 
 701             // Record updated successfully 
 706     // Problem updating record 
 707     return(pDb
->DispAllErrors(henv
, hdbc
, hstmtUpdate
)); 
 709 }  // wxDbTable::execUpdate() 
 712 /********** wxDbTable::query() **********/ 
 713 bool wxDbTable::query(int queryType
, bool forUpdate
, bool distinct
, const wxString 
&pSqlStmt
) 
 718         // The user may wish to select for update, but the DBMS may not be capable 
 719         selectForUpdate 
= CanSelectForUpdate(); 
 721         selectForUpdate 
= false; 
 723     // Set the SQL SELECT string 
 724     if (queryType 
!= DB_SELECT_STATEMENT
)               // A select statement was not passed in, 
 725     {                                                   // so generate a select statement. 
 726         BuildSelectStmt(sqlStmt
, queryType
, distinct
); 
 727         pDb
->WriteSqlLog(sqlStmt
); 
 730     // Make sure the cursor is closed first 
 731     if (!CloseCursor(hstmt
)) 
 734     // Execute the SQL SELECT statement 
 736     retcode 
= SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) (queryType 
== DB_SELECT_STATEMENT 
? pSqlStmt
.c_str() : sqlStmt
.c_str()), SQL_NTS
); 
 737     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
 738         return(pDb
->DispAllErrors(henv
, hdbc
, hstmt
)); 
 740     // Completed successfully 
 743 }  // wxDbTable::query() 
 746 /***************************** PUBLIC FUNCTIONS *****************************/ 
 749 /********** wxDbTable::Open() **********/ 
 750 bool wxDbTable::Open(bool checkPrivileges
, bool checkTableExists
) 
 759     // Calculate the maximum size of the concatenated 
 760     // keys for use with wxDbGrid 
 762     for (i
=0; i 
< m_numCols
; i
++) 
 764         if (colDefs
[i
].KeyField
) 
 766             m_keysize 
+= colDefs
[i
].SzDataObj
; 
 773     if (checkTableExists
) 
 775         if (pDb
->Dbms() == dbmsPOSTGRES
) 
 776             exists 
= pDb
->TableExists(tableName
, NULL
, tablePath
); 
 778             exists 
= pDb
->TableExists(tableName
, pDb
->GetUsername(), tablePath
); 
 781     // Verify that the table exists in the database 
 784         s 
= wxT("Table/view does not exist in the database"); 
 785         if ( *(pDb
->dbInf
.accessibleTables
) == wxT('Y')) 
 786             s 
+= wxT(", or you have no permissions.\n"); 
 790     else if (checkPrivileges
) 
 792         // Verify the user has rights to access the table. 
 793         bool hasPrivs 
wxDUMMY_INITIALIZE(true); 
 795         if (pDb
->Dbms() == dbmsPOSTGRES
) 
 796             hasPrivs 
= pDb
->TablePrivileges(tableName
, wxT("SELECT"), pDb
->GetUsername(), NULL
, tablePath
); 
 798             hasPrivs 
= pDb
->TablePrivileges(tableName
, wxT("SELECT"), pDb
->GetUsername(), pDb
->GetUsername(), tablePath
); 
 801             s 
= wxT("Connecting user does not have sufficient privileges to access this table.\n"); 
 808         if (!tablePath
.empty()) 
 809             p
.Printf(wxT("Error opening '%s/%s'.\n"),tablePath
.c_str(),tableName
.c_str()); 
 811             p
.Printf(wxT("Error opening '%s'.\n"), tableName
.c_str()); 
 814         pDb
->LogError(p
.GetData()); 
 819     // Bind the member variables for field exchange between 
 820     // the wxDbTable object and the ODBC record. 
 823         if (!bindInsertParams())                    // Inserts 
 826         if (!bindUpdateParams())                    // Updates 
 830     if (!bindCols(*hstmtDefault
))                   // Selects 
 833     if (!bindCols(hstmtInternal
))                   // Internal use only 
 837      * Do NOT bind the hstmtCount cursor!!! 
 840     // Build an insert statement using parameter markers 
 841     if (!queryOnly 
&& m_numCols 
> 0) 
 843         bool needComma 
= false; 
 844         sqlStmt
.Printf(wxT("INSERT INTO %s ("), 
 845                        pDb
->SQLTableName(tableName
.c_str()).c_str()); 
 846         for (i 
= 0; i 
< m_numCols
; i
++) 
 848             if (! colDefs
[i
].InsertAllowed
) 
 852             sqlStmt 
+= pDb
->SQLColumnName(colDefs
[i
].ColName
); 
 856         sqlStmt 
+= wxT(") VALUES ("); 
 858         int insertableCount 
= 0; 
 860         for (i 
= 0; i 
< m_numCols
; i
++) 
 862             if (! colDefs
[i
].InsertAllowed
) 
 872         // Prepare the insert statement for execution 
 875             if (SQLPrepare(hstmtInsert
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
) != SQL_SUCCESS
) 
 876                 return(pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
)); 
 882     // Completed successfully 
 885 }  // wxDbTable::Open() 
 888 /********** wxDbTable::Query() **********/ 
 889 bool wxDbTable::Query(bool forUpdate
, bool distinct
) 
 892     return(query(DB_SELECT_WHERE
, forUpdate
, distinct
)); 
 894 }  // wxDbTable::Query() 
 897 /********** wxDbTable::QueryBySqlStmt() **********/ 
 898 bool wxDbTable::QueryBySqlStmt(const wxString 
&pSqlStmt
) 
 900     pDb
->WriteSqlLog(pSqlStmt
); 
 902     return(query(DB_SELECT_STATEMENT
, false, false, pSqlStmt
)); 
 904 }  // wxDbTable::QueryBySqlStmt() 
 907 /********** wxDbTable::QueryMatching() **********/ 
 908 bool wxDbTable::QueryMatching(bool forUpdate
, bool distinct
) 
 911     return(query(DB_SELECT_MATCHING
, forUpdate
, distinct
)); 
 913 }  // wxDbTable::QueryMatching() 
 916 /********** wxDbTable::QueryOnKeyFields() **********/ 
 917 bool wxDbTable::QueryOnKeyFields(bool forUpdate
, bool distinct
) 
 920     return(query(DB_SELECT_KEYFIELDS
, forUpdate
, distinct
)); 
 922 }  // wxDbTable::QueryOnKeyFields() 
 925 /********** wxDbTable::GetPrev() **********/ 
 926 bool wxDbTable::GetPrev(void) 
 928     if (pDb
->FwdOnlyCursors()) 
 930         wxFAIL_MSG(wxT("GetPrev()::Backward scrolling cursors are not enabled for this instance of wxDbTable")); 
 934         return(getRec(SQL_FETCH_PRIOR
)); 
 936 }  // wxDbTable::GetPrev() 
 939 /********** wxDbTable::operator-- **********/ 
 940 bool wxDbTable::operator--(int) 
 942     if (pDb
->FwdOnlyCursors()) 
 944         wxFAIL_MSG(wxT("operator--:Backward scrolling cursors are not enabled for this instance of wxDbTable")); 
 948         return(getRec(SQL_FETCH_PRIOR
)); 
 950 }  // wxDbTable::operator-- 
 953 /********** wxDbTable::GetFirst() **********/ 
 954 bool wxDbTable::GetFirst(void) 
 956     if (pDb
->FwdOnlyCursors()) 
 958         wxFAIL_MSG(wxT("GetFirst():Backward scrolling cursors are not enabled for this instance of wxDbTable")); 
 962         return(getRec(SQL_FETCH_FIRST
)); 
 964 }  // wxDbTable::GetFirst() 
 967 /********** wxDbTable::GetLast() **********/ 
 968 bool wxDbTable::GetLast(void) 
 970     if (pDb
->FwdOnlyCursors()) 
 972         wxFAIL_MSG(wxT("GetLast()::Backward scrolling cursors are not enabled for this instance of wxDbTable")); 
 976         return(getRec(SQL_FETCH_LAST
)); 
 978 }  // wxDbTable::GetLast() 
 981 /********** wxDbTable::BuildDeleteStmt() **********/ 
 982 void wxDbTable::BuildDeleteStmt(wxString 
&pSqlStmt
, int typeOfDel
, const wxString 
&pWhereClause
) 
 984     wxASSERT(!queryOnly
); 
 988     wxString whereClause
; 
 992     // Handle the case of DeleteWhere() and the where clause is blank.  It should 
 993     // delete all records from the database in this case. 
 994     if (typeOfDel 
== DB_DEL_WHERE 
&& (pWhereClause
.length() == 0)) 
 996         pSqlStmt
.Printf(wxT("DELETE FROM %s"), 
 997                         pDb
->SQLTableName(tableName
.c_str()).c_str()); 
1001     pSqlStmt
.Printf(wxT("DELETE FROM %s WHERE "), 
1002                     pDb
->SQLTableName(tableName
.c_str()).c_str()); 
1004     // Append the WHERE clause to the SQL DELETE statement 
1007         case DB_DEL_KEYFIELDS
: 
1008             // If the datasource supports the ROWID column, build 
1009             // the where on ROWID for efficiency purposes. 
1010             // e.g. DELETE FROM PARTS WHERE ROWID = '111.222.333' 
1011             if (CanUpdateByROWID()) 
1014                 wxChar   rowid
[wxDB_ROWID_LEN
+1]; 
1016                 // Get the ROWID value.  If not successful retreiving the ROWID, 
1017                 // simply fall down through the code and build the WHERE clause 
1018                 // based on the key fields. 
1019                 if (SQLGetData(hstmt
, (UWORD
)(m_numCols
+1), SQL_C_WXCHAR
, (UCHAR
*) rowid
, sizeof(rowid
), &cb
) == SQL_SUCCESS
) 
1021                     pSqlStmt 
+= wxT("ROWID = '"); 
1023                     pSqlStmt 
+= wxT("'"); 
1027             // Unable to delete by ROWID, so build a WHERE 
1028             // clause based on the keyfields. 
1029             BuildWhereClause(whereClause
, DB_WHERE_KEYFIELDS
); 
1030             pSqlStmt 
+= whereClause
; 
1033             pSqlStmt 
+= pWhereClause
; 
1035         case DB_DEL_MATCHING
: 
1036             BuildWhereClause(whereClause
, DB_WHERE_MATCHING
); 
1037             pSqlStmt 
+= whereClause
; 
1041 }  // BuildDeleteStmt() 
1044 /***** DEPRECATED: use wxDbTable::BuildDeleteStmt(wxString &....) form *****/ 
1045 void wxDbTable::BuildDeleteStmt(wxChar 
*pSqlStmt
, int typeOfDel
, const wxString 
&pWhereClause
) 
1047     wxString tempSqlStmt
; 
1048     BuildDeleteStmt(tempSqlStmt
, typeOfDel
, pWhereClause
); 
1049     wxStrcpy(pSqlStmt
, tempSqlStmt
); 
1050 }  // wxDbTable::BuildDeleteStmt() 
1053 /********** wxDbTable::BuildSelectStmt() **********/ 
1054 void wxDbTable::BuildSelectStmt(wxString 
&pSqlStmt
, int typeOfSelect
, bool distinct
) 
1056     wxString whereClause
; 
1057     whereClause
.Empty(); 
1059     // Build a select statement to query the database 
1060     pSqlStmt 
= wxT("SELECT "); 
1062     // SELECT DISTINCT values only? 
1064         pSqlStmt 
+= wxT("DISTINCT "); 
1066     // Was a FROM clause specified to join tables to the base table? 
1067     // Available for ::Query() only!!! 
1068     bool appendFromClause 
= false; 
1069 #if wxODBC_BACKWARD_COMPATABILITY 
1070     if (typeOfSelect 
== DB_SELECT_WHERE 
&& from 
&& wxStrlen(from
)) 
1071         appendFromClause 
= true; 
1073     if (typeOfSelect 
== DB_SELECT_WHERE 
&& from
.length()) 
1074         appendFromClause 
= true; 
1077     // Add the column list 
1080     for (i 
= 0; i 
< m_numCols
; i
++) 
1082         tStr 
= colDefs
[i
].ColName
; 
1083         // If joining tables, the base table column names must be qualified to avoid ambiguity 
1084         if ((appendFromClause 
|| pDb
->Dbms() == dbmsACCESS
) && tStr
.Find(wxT('.')) == wxNOT_FOUND
) 
1086             pSqlStmt 
+= pDb
->SQLTableName(queryTableName
.c_str()); 
1087             pSqlStmt 
+= wxT("."); 
1089         pSqlStmt 
+= pDb
->SQLColumnName(colDefs
[i
].ColName
); 
1090         if (i 
+ 1 < m_numCols
) 
1091             pSqlStmt 
+= wxT(","); 
1094     // If the datasource supports ROWID, get this column as well.  Exception: Don't retrieve 
1095     // the ROWID if querying distinct records.  The rowid will always be unique. 
1096     if (!distinct 
&& CanUpdateByROWID()) 
1098         // If joining tables, the base table column names must be qualified to avoid ambiguity 
1099         if (appendFromClause 
|| pDb
->Dbms() == dbmsACCESS
) 
1101             pSqlStmt 
+= wxT(","); 
1102             pSqlStmt 
+= pDb
->SQLTableName(queryTableName
); 
1103             pSqlStmt 
+= wxT(".ROWID"); 
1106             pSqlStmt 
+= wxT(",ROWID"); 
1109     // Append the FROM tablename portion 
1110     pSqlStmt 
+= wxT(" FROM "); 
1111     pSqlStmt 
+= pDb
->SQLTableName(queryTableName
); 
1112 //    pSqlStmt += queryTableName; 
1114     // Sybase uses the HOLDLOCK keyword to lock a record during query. 
1115     // The HOLDLOCK keyword follows the table name in the from clause. 
1116     // Each table in the from clause must specify HOLDLOCK or 
1117     // NOHOLDLOCK (the default).  Note: The "FOR UPDATE" clause 
1118     // is parsed but ignored in SYBASE Transact-SQL. 
1119     if (selectForUpdate 
&& (pDb
->Dbms() == dbmsSYBASE_ASA 
|| pDb
->Dbms() == dbmsSYBASE_ASE
)) 
1120         pSqlStmt 
+= wxT(" HOLDLOCK"); 
1122     if (appendFromClause
) 
1125     // Append the WHERE clause.  Either append the where clause for the class 
1126     // or build a where clause.  The typeOfSelect determines this. 
1127     switch(typeOfSelect
) 
1129         case DB_SELECT_WHERE
: 
1130 #if wxODBC_BACKWARD_COMPATABILITY 
1131             if (where 
&& wxStrlen(where
))   // May not want a where clause!!! 
1133             if (where
.length())   // May not want a where clause!!! 
1136                 pSqlStmt 
+= wxT(" WHERE "); 
1140         case DB_SELECT_KEYFIELDS
: 
1141             BuildWhereClause(whereClause
, DB_WHERE_KEYFIELDS
); 
1142             if (whereClause
.length()) 
1144                 pSqlStmt 
+= wxT(" WHERE "); 
1145                 pSqlStmt 
+= whereClause
; 
1148         case DB_SELECT_MATCHING
: 
1149             BuildWhereClause(whereClause
, DB_WHERE_MATCHING
); 
1150             if (whereClause
.length()) 
1152                 pSqlStmt 
+= wxT(" WHERE "); 
1153                 pSqlStmt 
+= whereClause
; 
1158     // Append the ORDER BY clause 
1159 #if wxODBC_BACKWARD_COMPATABILITY 
1160     if (orderBy 
&& wxStrlen(orderBy
)) 
1162     if (orderBy
.length()) 
1165         pSqlStmt 
+= wxT(" ORDER BY "); 
1166         pSqlStmt 
+= orderBy
; 
1169     // SELECT FOR UPDATE if told to do so and the datasource is capable.  Sybase 
1170     // parses the FOR UPDATE clause but ignores it.  See the comment above on the 
1171     // HOLDLOCK for Sybase. 
1172     if (selectForUpdate 
&& CanSelectForUpdate()) 
1173         pSqlStmt 
+= wxT(" FOR UPDATE"); 
1175 }  // wxDbTable::BuildSelectStmt() 
1178 /***** DEPRECATED: use wxDbTable::BuildSelectStmt(wxString &....) form *****/ 
1179 void wxDbTable::BuildSelectStmt(wxChar 
*pSqlStmt
, int typeOfSelect
, bool distinct
) 
1181     wxString tempSqlStmt
; 
1182     BuildSelectStmt(tempSqlStmt
, typeOfSelect
, distinct
); 
1183     wxStrcpy(pSqlStmt
, tempSqlStmt
); 
1184 }  // wxDbTable::BuildSelectStmt() 
1187 /********** wxDbTable::BuildUpdateStmt() **********/ 
1188 void wxDbTable::BuildUpdateStmt(wxString 
&pSqlStmt
, int typeOfUpdate
, const wxString 
&pWhereClause
) 
1190     wxASSERT(!queryOnly
); 
1194     wxString whereClause
; 
1195     whereClause
.Empty(); 
1197     bool firstColumn 
= true; 
1199     pSqlStmt
.Printf(wxT("UPDATE %s SET "), 
1200                     pDb
->SQLTableName(tableName
.c_str()).c_str()); 
1202     // Append a list of columns to be updated 
1204     for (i 
= 0; i 
< m_numCols
; i
++) 
1206         // Only append Updateable columns 
1207         if (colDefs
[i
].Updateable
) 
1210                 pSqlStmt 
+= wxT(","); 
1212                 firstColumn 
= false; 
1214             pSqlStmt 
+= pDb
->SQLColumnName(colDefs
[i
].ColName
); 
1215 //            pSqlStmt += colDefs[i].ColName; 
1216             pSqlStmt 
+= wxT(" = ?"); 
1220     // Append the WHERE clause to the SQL UPDATE statement 
1221     pSqlStmt 
+= wxT(" WHERE "); 
1222     switch(typeOfUpdate
) 
1224         case DB_UPD_KEYFIELDS
: 
1225             // If the datasource supports the ROWID column, build 
1226             // the where on ROWID for efficiency purposes. 
1227             // e.g. UPDATE PARTS SET Col1 = ?, Col2 = ? WHERE ROWID = '111.222.333' 
1228             if (CanUpdateByROWID()) 
1231                 wxChar rowid
[wxDB_ROWID_LEN
+1]; 
1233                 // Get the ROWID value.  If not successful retreiving the ROWID, 
1234                 // simply fall down through the code and build the WHERE clause 
1235                 // based on the key fields. 
1236                 if (SQLGetData(hstmt
, (UWORD
)(m_numCols
+1), SQL_C_WXCHAR
, (UCHAR
*) rowid
, sizeof(rowid
), &cb
) == SQL_SUCCESS
) 
1238                     pSqlStmt 
+= wxT("ROWID = '"); 
1240                     pSqlStmt 
+= wxT("'"); 
1244             // Unable to delete by ROWID, so build a WHERE 
1245             // clause based on the keyfields. 
1246             BuildWhereClause(whereClause
, DB_WHERE_KEYFIELDS
); 
1247             pSqlStmt 
+= whereClause
; 
1250             pSqlStmt 
+= pWhereClause
; 
1253 }  // BuildUpdateStmt() 
1256 /***** DEPRECATED: use wxDbTable::BuildUpdateStmt(wxString &....) form *****/ 
1257 void wxDbTable::BuildUpdateStmt(wxChar 
*pSqlStmt
, int typeOfUpdate
, const wxString 
&pWhereClause
) 
1259     wxString tempSqlStmt
; 
1260     BuildUpdateStmt(tempSqlStmt
, typeOfUpdate
, pWhereClause
); 
1261     wxStrcpy(pSqlStmt
, tempSqlStmt
); 
1262 }  // BuildUpdateStmt() 
1265 /********** wxDbTable::BuildWhereClause() **********/ 
1266 void wxDbTable::BuildWhereClause(wxString 
&pWhereClause
, int typeOfWhere
, 
1267                                  const wxString 
&qualTableName
, bool useLikeComparison
) 
1269  * Note: BuildWhereClause() currently ignores timestamp columns. 
1270  *       They are not included as part of the where clause. 
1273     bool moreThanOneColumn 
= false; 
1276     // Loop through the columns building a where clause as you go 
1278     for (colNumber 
= 0; colNumber 
< m_numCols
; colNumber
++) 
1280         // Determine if this column should be included in the WHERE clause 
1281         if ((typeOfWhere 
== DB_WHERE_KEYFIELDS 
&& colDefs
[colNumber
].KeyField
) || 
1282              (typeOfWhere 
== DB_WHERE_MATCHING  
&& (!IsColNull((UWORD
)colNumber
)))) 
1284             // Skip over timestamp columns 
1285             if (colDefs
[colNumber
].SqlCtype 
== SQL_C_TIMESTAMP
) 
1287             // If there is more than 1 column, join them with the keyword "AND" 
1288             if (moreThanOneColumn
) 
1289                 pWhereClause 
+= wxT(" AND "); 
1291                 moreThanOneColumn 
= true; 
1293             // Concatenate where phrase for the column 
1294             wxString tStr 
= colDefs
[colNumber
].ColName
; 
1296             if (qualTableName
.length() && tStr
.Find(wxT('.')) == wxNOT_FOUND
) 
1298                 pWhereClause 
+= pDb
->SQLTableName(qualTableName
); 
1299                 pWhereClause 
+= wxT("."); 
1301             pWhereClause 
+= pDb
->SQLColumnName(colDefs
[colNumber
].ColName
); 
1303             if (useLikeComparison 
&& (colDefs
[colNumber
].SqlCtype 
== SQL_C_WXCHAR
)) 
1304                 pWhereClause 
+= wxT(" LIKE "); 
1306                 pWhereClause 
+= wxT(" = "); 
1308             switch(colDefs
[colNumber
].SqlCtype
) 
1314                 //case SQL_C_WXCHAR:  SQL_C_WXCHAR is covered by either SQL_C_CHAR or SQL_C_WCHAR 
1315                     colValue
.Printf(wxT("'%s'"), (UCHAR FAR 
*) colDefs
[colNumber
].PtrDataObj
); 
1319                     colValue
.Printf(wxT("%hi"), *((SWORD 
*) colDefs
[colNumber
].PtrDataObj
)); 
1322                     colValue
.Printf(wxT("%hu"), *((UWORD 
*) colDefs
[colNumber
].PtrDataObj
)); 
1326                     colValue
.Printf(wxT("%li"), *((SDWORD 
*) colDefs
[colNumber
].PtrDataObj
)); 
1329                     colValue
.Printf(wxT("%lu"), *((UDWORD 
*) colDefs
[colNumber
].PtrDataObj
)); 
1332                     colValue
.Printf(wxT("%.6f"), *((SFLOAT 
*) colDefs
[colNumber
].PtrDataObj
)); 
1335                     colValue
.Printf(wxT("%.6f"), *((SDOUBLE 
*) colDefs
[colNumber
].PtrDataObj
)); 
1340                         strMsg
.Printf(wxT("wxDbTable::bindParams(): Unknown column type for colDefs %d colName %s"), 
1341                                     colNumber
,colDefs
[colNumber
].ColName
); 
1342                         wxFAIL_MSG(strMsg
.c_str()); 
1346             pWhereClause 
+= colValue
; 
1349 }  // wxDbTable::BuildWhereClause() 
1352 /***** DEPRECATED: use wxDbTable::BuildWhereClause(wxString &....) form *****/ 
1353 void wxDbTable::BuildWhereClause(wxChar 
*pWhereClause
, int typeOfWhere
, 
1354                                  const wxString 
&qualTableName
, bool useLikeComparison
) 
1356     wxString tempSqlStmt
; 
1357     BuildWhereClause(tempSqlStmt
, typeOfWhere
, qualTableName
, useLikeComparison
); 
1358     wxStrcpy(pWhereClause
, tempSqlStmt
); 
1359 }  // wxDbTable::BuildWhereClause() 
1362 /********** wxDbTable::GetRowNum() **********/ 
1363 UWORD 
wxDbTable::GetRowNum(void) 
1367     if (SQLGetStmtOption(hstmt
, SQL_ROW_NUMBER
, (UCHAR
*) &rowNum
) != SQL_SUCCESS
) 
1369         pDb
->DispAllErrors(henv
, hdbc
, hstmt
); 
1373     // Completed successfully 
1374     return((UWORD
) rowNum
); 
1376 }  // wxDbTable::GetRowNum() 
1379 /********** wxDbTable::CloseCursor() **********/ 
1380 bool wxDbTable::CloseCursor(HSTMT cursor
) 
1382     if (SQLFreeStmt(cursor
, SQL_CLOSE
) != SQL_SUCCESS
) 
1383         return(pDb
->DispAllErrors(henv
, hdbc
, cursor
)); 
1385     // Completed successfully 
1388 }  // wxDbTable::CloseCursor() 
1391 /********** wxDbTable::CreateTable() **********/ 
1392 bool wxDbTable::CreateTable(bool attemptDrop
) 
1400 #ifdef DBDEBUG_CONSOLE 
1401     cout 
<< wxT("Creating Table ") << tableName 
<< wxT("...") << endl
; 
1405     if (attemptDrop 
&& !DropTable()) 
1409 #ifdef DBDEBUG_CONSOLE 
1410     for (i 
= 0; i 
< m_numCols
; i
++) 
1412         // Exclude derived columns since they are NOT part of the base table 
1413         if (colDefs
[i
].DerivedCol
) 
1415         cout 
<< i 
+ 1 << wxT(": ") << colDefs
[i
].ColName 
<< wxT("; "); 
1416         switch(colDefs
[i
].DbDataType
) 
1418             case DB_DATA_TYPE_VARCHAR
: 
1419                 cout 
<< pDb
->GetTypeInfVarchar().TypeName 
<< wxT("(") << (int)(colDefs
[i
].SzDataObj 
/ sizeof(wxChar
)) << wxT(")"); 
1421             case DB_DATA_TYPE_MEMO
: 
1422                 cout 
<< pDb
->GetTypeInfMemo().TypeName
; 
1424             case DB_DATA_TYPE_INTEGER
: 
1425                 cout 
<< pDb
->GetTypeInfInteger().TypeName
; 
1427             case DB_DATA_TYPE_FLOAT
: 
1428                 cout 
<< pDb
->GetTypeInfFloat().TypeName
; 
1430             case DB_DATA_TYPE_DATE
: 
1431                 cout 
<< pDb
->GetTypeInfDate().TypeName
; 
1433             case DB_DATA_TYPE_BLOB
: 
1434                 cout 
<< pDb
->GetTypeInfBlob().TypeName
; 
1441     // Build a CREATE TABLE string from the colDefs structure. 
1442     bool needComma 
= false; 
1444     sqlStmt
.Printf(wxT("CREATE TABLE %s ("), 
1445                    pDb
->SQLTableName(tableName
.c_str()).c_str()); 
1447     for (i 
= 0; i 
< m_numCols
; i
++) 
1449         // Exclude derived columns since they are NOT part of the base table 
1450         if (colDefs
[i
].DerivedCol
) 
1454             sqlStmt 
+= wxT(","); 
1456         sqlStmt 
+= pDb
->SQLColumnName(colDefs
[i
].ColName
); 
1457 //        sqlStmt += colDefs[i].ColName; 
1458         sqlStmt 
+= wxT(" "); 
1460         switch(colDefs
[i
].DbDataType
) 
1462             case DB_DATA_TYPE_VARCHAR
: 
1463                 sqlStmt 
+= pDb
->GetTypeInfVarchar().TypeName
; 
1465             case DB_DATA_TYPE_MEMO
: 
1466                 sqlStmt 
+= pDb
->GetTypeInfMemo().TypeName
; 
1468             case DB_DATA_TYPE_INTEGER
: 
1469                 sqlStmt 
+= pDb
->GetTypeInfInteger().TypeName
; 
1471             case DB_DATA_TYPE_FLOAT
: 
1472                 sqlStmt 
+= pDb
->GetTypeInfFloat().TypeName
; 
1474             case DB_DATA_TYPE_DATE
: 
1475                 sqlStmt 
+= pDb
->GetTypeInfDate().TypeName
; 
1477             case DB_DATA_TYPE_BLOB
: 
1478                 sqlStmt 
+= pDb
->GetTypeInfBlob().TypeName
; 
1481         // For varchars, append the size of the string 
1482         if (colDefs
[i
].DbDataType 
== DB_DATA_TYPE_VARCHAR 
&& 
1483             (pDb
->Dbms() != dbmsMY_SQL 
|| pDb
->GetTypeInfVarchar().TypeName 
!= _T("text")))// || 
1484 //            colDefs[i].DbDataType == DB_DATA_TYPE_BLOB) 
1487             s
.Printf(wxT("(%d)"), (int)(colDefs
[i
].SzDataObj 
/ sizeof(wxChar
))); 
1491         if (pDb
->Dbms() == dbmsDB2 
|| 
1492             pDb
->Dbms() == dbmsMY_SQL 
|| 
1493             pDb
->Dbms() == dbmsSYBASE_ASE  
|| 
1494             pDb
->Dbms() == dbmsINTERBASE  
|| 
1495             pDb
->Dbms() == dbmsFIREBIRD  
|| 
1496             pDb
->Dbms() == dbmsMS_SQL_SERVER
) 
1498             if (colDefs
[i
].KeyField
) 
1500                 sqlStmt 
+= wxT(" NOT NULL"); 
1506     // If there is a primary key defined, include it in the create statement 
1507     for (i 
= j 
= 0; i 
< m_numCols
; i
++) 
1509         if (colDefs
[i
].KeyField
) 
1515     if ( j 
&& (pDb
->Dbms() != dbmsDBASE
) 
1516         && (pDb
->Dbms() != dbmsXBASE_SEQUITER
) )  // Found a keyfield 
1518         switch (pDb
->Dbms()) 
1522             case dbmsSYBASE_ASA
: 
1523             case dbmsSYBASE_ASE
: 
1527                 // MySQL goes out on this one. We also declare the relevant key NON NULL above 
1528                 sqlStmt 
+= wxT(",PRIMARY KEY ("); 
1533                 sqlStmt 
+= wxT(",CONSTRAINT "); 
1534                 //  DB2 is limited to 18 characters for index names 
1535                 if (pDb
->Dbms() == dbmsDB2
) 
1537                     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.")); 
1538                     sqlStmt 
+= pDb
->SQLTableName(tableName
.substr(0, 13).c_str()); 
1539 //                    sqlStmt += tableName.substr(0, 13); 
1542                     sqlStmt 
+= pDb
->SQLTableName(tableName
.c_str()); 
1543 //                    sqlStmt += tableName; 
1545                 sqlStmt 
+= wxT("_PIDX PRIMARY KEY ("); 
1550         // List column name(s) of column(s) comprising the primary key 
1551         for (i 
= j 
= 0; i 
< m_numCols
; i
++) 
1553             if (colDefs
[i
].KeyField
) 
1555                 if (j
++) // Multi part key, comma separate names 
1556                     sqlStmt 
+= wxT(","); 
1557                 sqlStmt 
+= pDb
->SQLColumnName(colDefs
[i
].ColName
); 
1559                 if (pDb
->Dbms() == dbmsMY_SQL 
&& 
1560                     colDefs
[i
].DbDataType 
==  DB_DATA_TYPE_VARCHAR
) 
1563                     s
.Printf(wxT("(%d)"), (int)(colDefs
[i
].SzDataObj 
/ sizeof(wxChar
))); 
1568         sqlStmt 
+= wxT(")"); 
1570         if (pDb
->Dbms() == dbmsINFORMIX 
|| 
1571             pDb
->Dbms() == dbmsSYBASE_ASA 
|| 
1572             pDb
->Dbms() == dbmsSYBASE_ASE
) 
1574             sqlStmt 
+= wxT(" CONSTRAINT "); 
1575             sqlStmt 
+= pDb
->SQLTableName(tableName
); 
1576 //            sqlStmt += tableName; 
1577             sqlStmt 
+= wxT("_PIDX"); 
1580     // Append the closing parentheses for the create table statement 
1581     sqlStmt 
+= wxT(")"); 
1583     pDb
->WriteSqlLog(sqlStmt
); 
1585 #ifdef DBDEBUG_CONSOLE 
1586     cout 
<< endl 
<< sqlStmt
.c_str() << endl
; 
1589     // Execute the CREATE TABLE statement 
1590     RETCODE retcode 
= SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
); 
1591     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
1593         pDb
->DispAllErrors(henv
, hdbc
, hstmt
); 
1594         pDb
->RollbackTrans(); 
1599     // Commit the transaction and close the cursor 
1600     if (!pDb
->CommitTrans()) 
1602     if (!CloseCursor(hstmt
)) 
1605     // Database table created successfully 
1608 } // wxDbTable::CreateTable() 
1611 /********** wxDbTable::DropTable() **********/ 
1612 bool wxDbTable::DropTable() 
1614     // NOTE: This function returns true if the Table does not exist, but 
1615     //       only for identified databases.  Code will need to be added 
1616     //       below for any other databases when those databases are defined 
1617     //       to handle this situation consistently 
1621     sqlStmt
.Printf(wxT("DROP TABLE %s"), 
1622                    pDb
->SQLTableName(tableName
.c_str()).c_str()); 
1624     pDb
->WriteSqlLog(sqlStmt
); 
1626 #ifdef DBDEBUG_CONSOLE 
1627     cout 
<< endl 
<< sqlStmt
.c_str() << endl
; 
1630     RETCODE retcode 
= SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
); 
1631     if (retcode 
!= SQL_SUCCESS
) 
1633         // Check for "Base table not found" error and ignore 
1634         pDb
->GetNextError(henv
, hdbc
, hstmt
); 
1635         if (wxStrcmp(pDb
->sqlState
, wxT("S0002")) /*&& 
1636             wxStrcmp(pDb->sqlState, wxT("S1000"))*/)  // "Base table not found" 
1638             // Check for product specific error codes 
1639             if (!((pDb
->Dbms() == dbmsSYBASE_ASA    
&& !wxStrcmp(pDb
->sqlState
,wxT("42000")))   ||  // 5.x (and lower?) 
1640                   (pDb
->Dbms() == dbmsSYBASE_ASE    
&& !wxStrcmp(pDb
->sqlState
,wxT("37000")))   || 
1641                   (pDb
->Dbms() == dbmsPERVASIVE_SQL 
&& !wxStrcmp(pDb
->sqlState
,wxT("S1000")))   ||  // Returns an S1000 then an S0002 
1642                   (pDb
->Dbms() == dbmsPOSTGRES      
&& !wxStrcmp(pDb
->sqlState
,wxT("08S01"))))) 
1644                 pDb
->DispNextError(); 
1645                 pDb
->DispAllErrors(henv
, hdbc
, hstmt
); 
1646                 pDb
->RollbackTrans(); 
1647 //                CloseCursor(hstmt); 
1653     // Commit the transaction and close the cursor 
1654     if (! pDb
->CommitTrans()) 
1656     if (! CloseCursor(hstmt
)) 
1660 }  // wxDbTable::DropTable() 
1663 /********** wxDbTable::CreateIndex() **********/ 
1664 bool wxDbTable::CreateIndex(const wxString 
&indexName
, bool unique
, UWORD numIndexColumns
, 
1665                                      wxDbIdxDef 
*pIndexDefs
, bool attemptDrop
) 
1669     // Drop the index first 
1670     if (attemptDrop 
&& !DropIndex(indexName
)) 
1673     // MySQL (and possibly Sybase ASE?? - gt) require that any columns which are used as portions 
1674     // of an index have the columns defined as "NOT NULL".  During initial table creation though, 
1675     // it may not be known which columns are necessarily going to be part of an index (e.g. the 
1676     // table was created, then months later you determine that an additional index while 
1677     // give better performance, so you want to add an index). 
1679     // The following block of code will modify the column definition to make the column be 
1680     // defined with the "NOT NULL" qualifier. 
1681     if (pDb
->Dbms() == dbmsMY_SQL
) 
1686         for (i 
= 0; i 
< numIndexColumns 
&& ok
; i
++) 
1690             // Find the column definition that has the ColName that matches the 
1691             // index column name.  We need to do this to get the DB_DATA_TYPE of 
1692             // the index column, as MySQL's syntax for the ALTER column requires 
1694             while (!found 
&& (j 
< this->m_numCols
)) 
1696                 if (wxStrcmp(colDefs
[j
].ColName
,pIndexDefs
[i
].ColName
) == 0) 
1704                 ok 
= pDb
->ModifyColumn(tableName
, pIndexDefs
[i
].ColName
, 
1705                                         colDefs
[j
].DbDataType
, (int)(colDefs
[j
].SzDataObj 
/ sizeof(wxChar
)), 
1711                     // retcode is not used 
1712                     wxODBC_ERRORS retcode
; 
1713                     // Oracle returns a DB_ERR_GENERAL_ERROR if the column is already 
1714                     // defined to be NOT NULL, but reportedly MySQL doesn't mind. 
1715                     // This line is just here for debug checking of the value 
1716                     retcode 
= (wxODBC_ERRORS
)pDb
->DB_STATUS
; 
1727             pDb
->RollbackTrans(); 
1732     // Build a CREATE INDEX statement 
1733     sqlStmt 
= wxT("CREATE "); 
1735         sqlStmt 
+= wxT("UNIQUE "); 
1737     sqlStmt 
+= wxT("INDEX "); 
1738     sqlStmt 
+= pDb
->SQLTableName(indexName
); 
1739     sqlStmt 
+= wxT(" ON "); 
1741     sqlStmt 
+= pDb
->SQLTableName(tableName
); 
1742 //    sqlStmt += tableName; 
1743     sqlStmt 
+= wxT(" ("); 
1745     // Append list of columns making up index 
1747     for (i 
= 0; i 
< numIndexColumns
; i
++) 
1749         sqlStmt 
+= pDb
->SQLColumnName(pIndexDefs
[i
].ColName
); 
1750 //        sqlStmt += pIndexDefs[i].ColName; 
1752         // MySQL requires a key length on VARCHAR keys 
1753         if ( pDb
->Dbms() == dbmsMY_SQL 
) 
1755             // Find the details on this column 
1757             for ( j 
= 0; j 
< m_numCols
; ++j 
) 
1759                 if ( wxStrcmp( pIndexDefs
[i
].ColName
, colDefs
[j
].ColName 
) == 0 ) 
1764             if ( colDefs
[j
].DbDataType 
==  DB_DATA_TYPE_VARCHAR
) 
1767                 s
.Printf(wxT("(%d)"), (int)(colDefs
[i
].SzDataObj 
/ sizeof(wxChar
))); 
1772         // Postgres and SQL Server 7 do not support the ASC/DESC keywords for index columns 
1773         if (!((pDb
->Dbms() == dbmsMS_SQL_SERVER
) && (wxStrncmp(pDb
->dbInf
.dbmsVer
,_T("07"),2)==0)) && 
1774             !(pDb
->Dbms() == dbmsFIREBIRD
) && 
1775             !(pDb
->Dbms() == dbmsPOSTGRES
)) 
1777             if (pIndexDefs
[i
].Ascending
) 
1778                 sqlStmt 
+= wxT(" ASC"); 
1780                 sqlStmt 
+= wxT(" DESC"); 
1783             wxASSERT_MSG(pIndexDefs
[i
].Ascending
, _T("Datasource does not support DESCending index columns")); 
1785         if ((i 
+ 1) < numIndexColumns
) 
1786             sqlStmt 
+= wxT(","); 
1789     // Append closing parentheses 
1790     sqlStmt 
+= wxT(")"); 
1792     pDb
->WriteSqlLog(sqlStmt
); 
1794 #ifdef DBDEBUG_CONSOLE 
1795     cout 
<< endl 
<< sqlStmt
.c_str() << endl 
<< endl
; 
1798     // Execute the CREATE INDEX statement 
1799     RETCODE retcode 
= SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
); 
1800     if (retcode 
!= SQL_SUCCESS
) 
1802         pDb
->DispAllErrors(henv
, hdbc
, hstmt
); 
1803         pDb
->RollbackTrans(); 
1808     // Commit the transaction and close the cursor 
1809     if (! pDb
->CommitTrans()) 
1811     if (! CloseCursor(hstmt
)) 
1814     // Index Created Successfully 
1817 }  // wxDbTable::CreateIndex() 
1820 /********** wxDbTable::DropIndex() **********/ 
1821 bool wxDbTable::DropIndex(const wxString 
&indexName
) 
1823     // NOTE: This function returns true if the Index does not exist, but 
1824     //       only for identified databases.  Code will need to be added 
1825     //       below for any other databases when those databases are defined 
1826     //       to handle this situation consistently 
1830     if (pDb
->Dbms() == dbmsACCESS 
|| pDb
->Dbms() == dbmsMY_SQL 
|| 
1831         pDb
->Dbms() == dbmsDBASE 
/*|| Paradox needs this syntax too when we add support*/) 
1832         sqlStmt
.Printf(wxT("DROP INDEX %s ON %s"), 
1833                        pDb
->SQLTableName(indexName
.c_str()).c_str(), 
1834                        pDb
->SQLTableName(tableName
.c_str()).c_str()); 
1835     else if ((pDb
->Dbms() == dbmsMS_SQL_SERVER
) || 
1836              (pDb
->Dbms() == dbmsSYBASE_ASE
) || 
1837              (pDb
->Dbms() == dbmsXBASE_SEQUITER
)) 
1838         sqlStmt
.Printf(wxT("DROP INDEX %s.%s"), 
1839                        pDb
->SQLTableName(tableName
.c_str()).c_str(), 
1840                        pDb
->SQLTableName(indexName
.c_str()).c_str()); 
1842         sqlStmt
.Printf(wxT("DROP INDEX %s"), 
1843                        pDb
->SQLTableName(indexName
.c_str()).c_str()); 
1845     pDb
->WriteSqlLog(sqlStmt
); 
1847 #ifdef DBDEBUG_CONSOLE 
1848     cout 
<< endl 
<< sqlStmt
.c_str() << endl
; 
1850     RETCODE retcode 
= SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
); 
1851     if (retcode 
!= SQL_SUCCESS
) 
1853         // Check for "Index not found" error and ignore 
1854         pDb
->GetNextError(henv
, hdbc
, hstmt
); 
1855         if (wxStrcmp(pDb
->sqlState
,wxT("S0012")))  // "Index not found" 
1857             // Check for product specific error codes 
1858             if (!((pDb
->Dbms() == dbmsSYBASE_ASA    
&& !wxStrcmp(pDb
->sqlState
,wxT("42000"))) ||  // v5.x (and lower?) 
1859                   (pDb
->Dbms() == dbmsSYBASE_ASE    
&& !wxStrcmp(pDb
->sqlState
,wxT("37000"))) || 
1860                   (pDb
->Dbms() == dbmsMS_SQL_SERVER 
&& !wxStrcmp(pDb
->sqlState
,wxT("S1000"))) || 
1861                   (pDb
->Dbms() == dbmsINTERBASE     
&& !wxStrcmp(pDb
->sqlState
,wxT("S1000"))) || 
1862                   (pDb
->Dbms() == dbmsMAXDB         
&& !wxStrcmp(pDb
->sqlState
,wxT("S1000"))) || 
1863                   (pDb
->Dbms() == dbmsFIREBIRD      
&& !wxStrcmp(pDb
->sqlState
,wxT("HY000"))) || 
1864                   (pDb
->Dbms() == dbmsSYBASE_ASE    
&& !wxStrcmp(pDb
->sqlState
,wxT("S0002"))) ||  // Base table not found 
1865                   (pDb
->Dbms() == dbmsMY_SQL        
&& !wxStrcmp(pDb
->sqlState
,wxT("42S12"))) ||  // tested by Christopher Ludwik Marino-Cebulski using v3.23.21beta 
1866                   (pDb
->Dbms() == dbmsPOSTGRES      
&& !wxStrcmp(pDb
->sqlState
,wxT("08S01"))) 
1869                 pDb
->DispNextError(); 
1870                 pDb
->DispAllErrors(henv
, hdbc
, hstmt
); 
1871                 pDb
->RollbackTrans(); 
1878     // Commit the transaction and close the cursor 
1879     if (! pDb
->CommitTrans()) 
1881     if (! CloseCursor(hstmt
)) 
1885 }  // wxDbTable::DropIndex() 
1888 /********** wxDbTable::SetOrderByColNums() **********/ 
1889 bool wxDbTable::SetOrderByColNums(UWORD first
, ... ) 
1891     int         colNumber 
= first
;  // using 'int' to be able to look for wxDB_NO_MORE_COLUN_NUMBERS 
1897     va_start(argptr
, first
);     /* Initialize variable arguments. */ 
1898     while (!abort 
&& (colNumber 
!= wxDB_NO_MORE_COLUMN_NUMBERS
)) 
1900         // Make sure the passed in column number 
1901         // is within the valid range of columns 
1903         // Valid columns are 0 thru m_numCols-1 
1904         if (colNumber 
>= m_numCols 
|| colNumber 
< 0) 
1910         if (colNumber 
!= first
) 
1911             tempStr 
+= wxT(","); 
1913         tempStr 
+= colDefs
[colNumber
].ColName
; 
1914         colNumber 
= va_arg (argptr
, int); 
1916     va_end (argptr
);              /* Reset variable arguments.      */ 
1918     SetOrderByClause(tempStr
); 
1921 }  // wxDbTable::SetOrderByColNums() 
1924 /********** wxDbTable::Insert() **********/ 
1925 int wxDbTable::Insert(void) 
1927     wxASSERT(!queryOnly
); 
1928     if (queryOnly 
|| !insertable
) 
1933     // Insert the record by executing the already prepared insert statement 
1935     retcode 
= SQLExecute(hstmtInsert
); 
1936     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
&& 
1937         retcode 
!= SQL_NEED_DATA
) 
1939         // Check to see if integrity constraint was violated 
1940         pDb
->GetNextError(henv
, hdbc
, hstmtInsert
); 
1941         if (! wxStrcmp(pDb
->sqlState
, wxT("23000")))  // Integrity constraint violated 
1942             return(DB_ERR_INTEGRITY_CONSTRAINT_VIOL
); 
1945             pDb
->DispNextError(); 
1946             pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
); 
1950     if (retcode 
== SQL_NEED_DATA
) 
1953         retcode 
= SQLParamData(hstmtInsert
, &pParmID
); 
1954         while (retcode 
== SQL_NEED_DATA
) 
1956             // Find the parameter 
1958             for (i
=0; i 
< m_numCols
; i
++) 
1960                 if (colDefs
[i
].PtrDataObj 
== pParmID
) 
1962                     // We found it.  Store the parameter. 
1963                     retcode 
= SQLPutData(hstmtInsert
, pParmID
, colDefs
[i
].SzDataObj
); 
1964                     if (retcode 
!= SQL_SUCCESS
) 
1966                         pDb
->DispNextError(); 
1967                         pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
); 
1973             retcode 
= SQLParamData(hstmtInsert
, &pParmID
); 
1974             if (retcode 
!= SQL_SUCCESS 
&& 
1975                 retcode 
!= SQL_SUCCESS_WITH_INFO
) 
1977                 // record was not inserted 
1978                 pDb
->DispNextError(); 
1979                 pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
); 
1985     // Record inserted into the datasource successfully 
1988 }  // wxDbTable::Insert() 
1991 /********** wxDbTable::Update() **********/ 
1992 bool wxDbTable::Update(void) 
1994     wxASSERT(!queryOnly
); 
2000     // Build the SQL UPDATE statement 
2001     BuildUpdateStmt(sqlStmt
, DB_UPD_KEYFIELDS
); 
2003     pDb
->WriteSqlLog(sqlStmt
); 
2005 #ifdef DBDEBUG_CONSOLE 
2006     cout 
<< endl 
<< sqlStmt
.c_str() << endl 
<< endl
; 
2009     // Execute the SQL UPDATE statement 
2010     return(execUpdate(sqlStmt
)); 
2012 }  // wxDbTable::Update() 
2015 /********** wxDbTable::Update(pSqlStmt) **********/ 
2016 bool wxDbTable::Update(const wxString 
&pSqlStmt
) 
2018     wxASSERT(!queryOnly
); 
2022     pDb
->WriteSqlLog(pSqlStmt
); 
2024     return(execUpdate(pSqlStmt
)); 
2026 }  // wxDbTable::Update(pSqlStmt) 
2029 /********** wxDbTable::UpdateWhere() **********/ 
2030 bool wxDbTable::UpdateWhere(const wxString 
&pWhereClause
) 
2032     wxASSERT(!queryOnly
); 
2038     // Build the SQL UPDATE statement 
2039     BuildUpdateStmt(sqlStmt
, DB_UPD_WHERE
, pWhereClause
); 
2041     pDb
->WriteSqlLog(sqlStmt
); 
2043 #ifdef DBDEBUG_CONSOLE 
2044     cout 
<< endl 
<< sqlStmt
.c_str() << endl 
<< endl
; 
2047     // Execute the SQL UPDATE statement 
2048     return(execUpdate(sqlStmt
)); 
2050 }  // wxDbTable::UpdateWhere() 
2053 /********** wxDbTable::Delete() **********/ 
2054 bool wxDbTable::Delete(void) 
2056     wxASSERT(!queryOnly
); 
2063     // Build the SQL DELETE statement 
2064     BuildDeleteStmt(sqlStmt
, DB_DEL_KEYFIELDS
); 
2066     pDb
->WriteSqlLog(sqlStmt
); 
2068     // Execute the SQL DELETE statement 
2069     return(execDelete(sqlStmt
)); 
2071 }  // wxDbTable::Delete() 
2074 /********** wxDbTable::DeleteWhere() **********/ 
2075 bool wxDbTable::DeleteWhere(const wxString 
&pWhereClause
) 
2077     wxASSERT(!queryOnly
); 
2084     // Build the SQL DELETE statement 
2085     BuildDeleteStmt(sqlStmt
, DB_DEL_WHERE
, pWhereClause
); 
2087     pDb
->WriteSqlLog(sqlStmt
); 
2089     // Execute the SQL DELETE statement 
2090     return(execDelete(sqlStmt
)); 
2092 }  // wxDbTable::DeleteWhere() 
2095 /********** wxDbTable::DeleteMatching() **********/ 
2096 bool wxDbTable::DeleteMatching(void) 
2098     wxASSERT(!queryOnly
); 
2105     // Build the SQL DELETE statement 
2106     BuildDeleteStmt(sqlStmt
, DB_DEL_MATCHING
); 
2108     pDb
->WriteSqlLog(sqlStmt
); 
2110     // Execute the SQL DELETE statement 
2111     return(execDelete(sqlStmt
)); 
2113 }  // wxDbTable::DeleteMatching() 
2116 /********** wxDbTable::IsColNull() **********/ 
2117 bool wxDbTable::IsColNull(UWORD colNumber
) const 
2120     This logic is just not right.  It would indicate true 
2121     if a numeric field were set to a value of 0. 
2123     switch(colDefs[colNumber].SqlCtype) 
2127         //case SQL_C_WXCHAR:  SQL_C_WXCHAR is covered by either SQL_C_CHAR or SQL_C_WCHAR 
2128             return(((UCHAR FAR *) colDefs[colNumber].PtrDataObj)[0] == 0); 
2130             return((  *((SWORD *) colDefs[colNumber].PtrDataObj))   == 0); 
2132             return((   *((UWORD*) colDefs[colNumber].PtrDataObj))   == 0); 
2134             return(( *((SDWORD *) colDefs[colNumber].PtrDataObj))   == 0); 
2136             return(( *((UDWORD *) colDefs[colNumber].PtrDataObj))   == 0); 
2138             return(( *((SFLOAT *) colDefs[colNumber].PtrDataObj))   == 0); 
2140             return((*((SDOUBLE *) colDefs[colNumber].PtrDataObj))   == 0); 
2141         case SQL_C_TIMESTAMP: 
2142             TIMESTAMP_STRUCT *pDt; 
2143             pDt = (TIMESTAMP_STRUCT *) colDefs[colNumber].PtrDataObj; 
2144             if (pDt->year == 0 && pDt->month == 0 && pDt->day == 0) 
2152     return (colDefs
[colNumber
].Null
); 
2153 }  // wxDbTable::IsColNull() 
2156 /********** wxDbTable::CanSelectForUpdate() **********/ 
2157 bool wxDbTable::CanSelectForUpdate(void) 
2162     if (pDb
->Dbms() == dbmsMY_SQL
) 
2165     if ((pDb
->Dbms() == dbmsORACLE
) || 
2166         (pDb
->dbInf
.posStmts 
& SQL_PS_SELECT_FOR_UPDATE
)) 
2171 }  // wxDbTable::CanSelectForUpdate() 
2174 /********** wxDbTable::CanUpdateByROWID() **********/ 
2175 bool wxDbTable::CanUpdateByROWID(void) 
2178  * NOTE: Returning false for now until this can be debugged, 
2179  *        as the ROWID is not getting updated correctly 
2183     if (pDb->Dbms() == dbmsORACLE) 
2188 }  // wxDbTable::CanUpdateByROWID() 
2191 /********** wxDbTable::IsCursorClosedOnCommit() **********/ 
2192 bool wxDbTable::IsCursorClosedOnCommit(void) 
2194     if (pDb
->dbInf
.cursorCommitBehavior 
== SQL_CB_PRESERVE
) 
2199 }  // wxDbTable::IsCursorClosedOnCommit() 
2203 /********** wxDbTable::ClearMemberVar() **********/ 
2204 void wxDbTable::ClearMemberVar(UWORD colNumber
, bool setToNull
) 
2206     wxASSERT(colNumber 
< m_numCols
); 
2208     switch(colDefs
[colNumber
].SqlCtype
) 
2214         //case SQL_C_WXCHAR:  SQL_C_WXCHAR is covered by either SQL_C_CHAR or SQL_C_WCHAR 
2215             ((UCHAR FAR 
*) colDefs
[colNumber
].PtrDataObj
)[0]    = 0; 
2218             *((SWORD 
*) colDefs
[colNumber
].PtrDataObj
)          = 0; 
2221             *((UWORD
*) colDefs
[colNumber
].PtrDataObj
)           = 0; 
2225             *((SDWORD 
*) colDefs
[colNumber
].PtrDataObj
)         = 0; 
2228             *((UDWORD 
*) colDefs
[colNumber
].PtrDataObj
)         = 0; 
2231             *((SFLOAT 
*) colDefs
[colNumber
].PtrDataObj
)         = 0.0f
; 
2234             *((SDOUBLE 
*) colDefs
[colNumber
].PtrDataObj
)        = 0.0f
; 
2236         case SQL_C_TIMESTAMP
: 
2237             TIMESTAMP_STRUCT 
*pDt
; 
2238             pDt 
= (TIMESTAMP_STRUCT 
*) colDefs
[colNumber
].PtrDataObj
; 
2249             pDtd 
= (DATE_STRUCT 
*) colDefs
[colNumber
].PtrDataObj
; 
2256             pDtt 
= (TIME_STRUCT 
*) colDefs
[colNumber
].PtrDataObj
; 
2264         SetColNull(colNumber
); 
2265 }  // wxDbTable::ClearMemberVar() 
2268 /********** wxDbTable::ClearMemberVars() **********/ 
2269 void wxDbTable::ClearMemberVars(bool setToNull
) 
2273     // Loop through the columns setting each member variable to zero 
2274     for (i
=0; i 
< m_numCols
; i
++) 
2275         ClearMemberVar((UWORD
)i
,setToNull
); 
2277 }  // wxDbTable::ClearMemberVars() 
2280 /********** wxDbTable::SetQueryTimeout() **********/ 
2281 bool wxDbTable::SetQueryTimeout(UDWORD nSeconds
) 
2283     if (SQLSetStmtOption(hstmtInsert
, SQL_QUERY_TIMEOUT
, nSeconds
) != SQL_SUCCESS
) 
2284         return(pDb
->DispAllErrors(henv
, hdbc
, hstmtInsert
)); 
2285     if (SQLSetStmtOption(hstmtUpdate
, SQL_QUERY_TIMEOUT
, nSeconds
) != SQL_SUCCESS
) 
2286         return(pDb
->DispAllErrors(henv
, hdbc
, hstmtUpdate
)); 
2287     if (SQLSetStmtOption(hstmtDelete
, SQL_QUERY_TIMEOUT
, nSeconds
) != SQL_SUCCESS
) 
2288         return(pDb
->DispAllErrors(henv
, hdbc
, hstmtDelete
)); 
2289     if (SQLSetStmtOption(hstmtInternal
, SQL_QUERY_TIMEOUT
, nSeconds
) != SQL_SUCCESS
) 
2290         return(pDb
->DispAllErrors(henv
, hdbc
, hstmtInternal
)); 
2292     // Completed Successfully 
2295 }  // wxDbTable::SetQueryTimeout() 
2298 /********** wxDbTable::SetColDefs() **********/ 
2299 bool wxDbTable::SetColDefs(UWORD index
, const wxString 
&fieldName
, int dataType
, void *pData
, 
2300                            SWORD cType
, int size
, bool keyField
, bool updateable
, 
2301                            bool insertAllowed
, bool derivedColumn
) 
2305     if (index 
>= m_numCols
)  // Columns numbers are zero based.... 
2307         tmpStr
.Printf(wxT("Specified column index (%d) exceeds the maximum number of columns (%d) registered for this table definition.  Column definition not added."), index
, m_numCols
); 
2313     if (!colDefs
)  // May happen if the database connection fails 
2316     if (fieldName
.length() > (unsigned int) DB_MAX_COLUMN_NAME_LEN
) 
2318         wxStrncpy(colDefs
[index
].ColName
, fieldName
, DB_MAX_COLUMN_NAME_LEN
); 
2319         colDefs
[index
].ColName
[DB_MAX_COLUMN_NAME_LEN
] = 0;  // Prevent buffer overrun 
2321         tmpStr
.Printf(wxT("Column name '%s' is too long. Truncated to '%s'."), 
2322                       fieldName
.c_str(),colDefs
[index
].ColName
); 
2327         wxStrcpy(colDefs
[index
].ColName
, fieldName
); 
2329     colDefs
[index
].DbDataType       
= dataType
; 
2330     colDefs
[index
].PtrDataObj       
= pData
; 
2331     colDefs
[index
].SqlCtype         
= cType
; 
2332     colDefs
[index
].SzDataObj        
= size
;  //TODO: glt ??? * sizeof(wxChar) ??? 
2333     colDefs
[index
].KeyField         
= keyField
; 
2334     colDefs
[index
].DerivedCol       
= derivedColumn
; 
2335     // Derived columns by definition would NOT be "Insertable" or "Updateable" 
2338         colDefs
[index
].Updateable       
= false; 
2339         colDefs
[index
].InsertAllowed    
= false; 
2343         colDefs
[index
].Updateable       
= updateable
; 
2344         colDefs
[index
].InsertAllowed    
= insertAllowed
; 
2347     colDefs
[index
].Null                 
= false; 
2351 }  // wxDbTable::SetColDefs() 
2354 /********** wxDbTable::SetColDefs() **********/ 
2355 wxDbColDataPtr
* wxDbTable::SetColDefs(wxDbColInf 
*pColInfs
, UWORD numCols
) 
2358     wxDbColDataPtr 
*pColDataPtrs 
= NULL
; 
2364         pColDataPtrs 
= new wxDbColDataPtr
[numCols
+1]; 
2366         for (index 
= 0; index 
< numCols
; index
++) 
2368             // Process the fields 
2369             switch (pColInfs
[index
].dbDataType
) 
2371                 case DB_DATA_TYPE_VARCHAR
: 
2372                    pColDataPtrs
[index
].PtrDataObj 
= new wxChar
[pColInfs
[index
].bufferSize
+(1*sizeof(wxChar
))]; 
2373                    pColDataPtrs
[index
].SzDataObj  
= pColInfs
[index
].bufferSize
+(1*sizeof(wxChar
)); 
2374                    pColDataPtrs
[index
].SqlCtype   
= SQL_C_WXCHAR
; 
2376                 case DB_DATA_TYPE_MEMO
: 
2377                    pColDataPtrs
[index
].PtrDataObj 
= new wxChar
[pColInfs
[index
].bufferSize
+(1*sizeof(wxChar
))]; 
2378                    pColDataPtrs
[index
].SzDataObj  
= pColInfs
[index
].bufferSize
+(1*sizeof(wxChar
)); 
2379                    pColDataPtrs
[index
].SqlCtype   
= SQL_C_WXCHAR
; 
2381                 case DB_DATA_TYPE_INTEGER
: 
2382                     // Can be long or short 
2383                     if (pColInfs
[index
].bufferSize 
== sizeof(long)) 
2385                       pColDataPtrs
[index
].PtrDataObj 
= new long; 
2386                       pColDataPtrs
[index
].SzDataObj  
= sizeof(long); 
2387                       pColDataPtrs
[index
].SqlCtype   
= SQL_C_SLONG
; 
2391                         pColDataPtrs
[index
].PtrDataObj 
= new short; 
2392                         pColDataPtrs
[index
].SzDataObj  
= sizeof(short); 
2393                         pColDataPtrs
[index
].SqlCtype   
= SQL_C_SSHORT
; 
2396                 case DB_DATA_TYPE_FLOAT
: 
2397                     // Can be float or double 
2398                     if (pColInfs
[index
].bufferSize 
== sizeof(float)) 
2400                         pColDataPtrs
[index
].PtrDataObj 
= new float; 
2401                         pColDataPtrs
[index
].SzDataObj  
= sizeof(float); 
2402                         pColDataPtrs
[index
].SqlCtype   
= SQL_C_FLOAT
; 
2406                         pColDataPtrs
[index
].PtrDataObj 
= new double; 
2407                         pColDataPtrs
[index
].SzDataObj  
= sizeof(double); 
2408                         pColDataPtrs
[index
].SqlCtype   
= SQL_C_DOUBLE
; 
2411                 case DB_DATA_TYPE_DATE
: 
2412                     pColDataPtrs
[index
].PtrDataObj 
= new TIMESTAMP_STRUCT
; 
2413                     pColDataPtrs
[index
].SzDataObj  
= sizeof(TIMESTAMP_STRUCT
); 
2414                     pColDataPtrs
[index
].SqlCtype   
= SQL_C_TIMESTAMP
; 
2416                 case DB_DATA_TYPE_BLOB
: 
2417                     wxFAIL_MSG(wxT("This form of ::SetColDefs() cannot be used with BLOB columns")); 
2418                     pColDataPtrs
[index
].PtrDataObj 
= /*BLOB ADDITION NEEDED*/NULL
; 
2419                     pColDataPtrs
[index
].SzDataObj  
= /*BLOB ADDITION NEEDED*/sizeof(void *); 
2420                     pColDataPtrs
[index
].SqlCtype   
= SQL_VARBINARY
; 
2423             if (pColDataPtrs
[index
].PtrDataObj 
!= NULL
) 
2424                 SetColDefs (index
,pColInfs
[index
].colName
,pColInfs
[index
].dbDataType
, pColDataPtrs
[index
].PtrDataObj
, pColDataPtrs
[index
].SqlCtype
, pColDataPtrs
[index
].SzDataObj
); 
2427                 // Unable to build all the column definitions, as either one of 
2428                 // the calls to "new" failed above, or there was a BLOB field 
2429                 // to have a column definition for.  If BLOBs are to be used, 
2430                 // the other form of ::SetColDefs() must be used, as it is impossible 
2431                 // to know the maximum size to create the PtrDataObj to be. 
2432                 delete [] pColDataPtrs
; 
2438     return (pColDataPtrs
); 
2440 } // wxDbTable::SetColDefs() 
2443 /********** wxDbTable::SetCursor() **********/ 
2444 void wxDbTable::SetCursor(HSTMT 
*hstmtActivate
) 
2446     if (hstmtActivate 
== wxDB_DEFAULT_CURSOR
) 
2447         hstmt 
= *hstmtDefault
; 
2449         hstmt 
= *hstmtActivate
; 
2451 }  // wxDbTable::SetCursor() 
2454 /********** wxDbTable::Count(const wxString &) **********/ 
2455 ULONG 
wxDbTable::Count(const wxString 
&args
) 
2461     // Build a "SELECT COUNT(*) FROM queryTableName [WHERE whereClause]" SQL Statement 
2462     sqlStmt  
= wxT("SELECT COUNT("); 
2464     sqlStmt 
+= wxT(") FROM "); 
2465     sqlStmt 
+= pDb
->SQLTableName(queryTableName
); 
2466 //    sqlStmt += queryTableName; 
2467 #if wxODBC_BACKWARD_COMPATABILITY 
2468     if (from 
&& wxStrlen(from
)) 
2474     // Add the where clause if one is provided 
2475 #if wxODBC_BACKWARD_COMPATABILITY 
2476     if (where 
&& wxStrlen(where
)) 
2481         sqlStmt 
+= wxT(" WHERE "); 
2485     pDb
->WriteSqlLog(sqlStmt
); 
2487     // Initialize the Count cursor if it's not already initialized 
2490         hstmtCount 
= GetNewCursor(false,false); 
2491         wxASSERT(hstmtCount
); 
2496     // Execute the SQL statement 
2497     if (SQLExecDirect(*hstmtCount
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
) != SQL_SUCCESS
) 
2499         pDb
->DispAllErrors(henv
, hdbc
, *hstmtCount
); 
2504     if (SQLFetch(*hstmtCount
) != SQL_SUCCESS
) 
2506         pDb
->DispAllErrors(henv
, hdbc
, *hstmtCount
); 
2510     // Obtain the result 
2511     if (SQLGetData(*hstmtCount
, (UWORD
)1, SQL_C_ULONG
, &count
, sizeof(count
), &cb
) != SQL_SUCCESS
) 
2513         pDb
->DispAllErrors(henv
, hdbc
, *hstmtCount
); 
2518     if (SQLFreeStmt(*hstmtCount
, SQL_CLOSE
) != SQL_SUCCESS
) 
2519         pDb
->DispAllErrors(henv
, hdbc
, *hstmtCount
); 
2521     // Return the record count 
2524 }  // wxDbTable::Count() 
2527 /********** wxDbTable::Refresh() **********/ 
2528 bool wxDbTable::Refresh(void) 
2532     // Switch to the internal cursor so any active cursors are not corrupted 
2533     HSTMT currCursor 
= GetCursor(); 
2534     hstmt 
= hstmtInternal
; 
2535 #if wxODBC_BACKWARD_COMPATABILITY 
2536     // Save the where and order by clauses 
2537     wxChar 
*saveWhere 
= where
; 
2538     wxChar 
*saveOrderBy 
= orderBy
; 
2540     wxString saveWhere 
= where
; 
2541     wxString saveOrderBy 
= orderBy
; 
2543     // Build a where clause to refetch the record with.  Try and use the 
2544     // ROWID if it's available, ow use the key fields. 
2545     wxString whereClause
; 
2546     whereClause
.Empty(); 
2548     if (CanUpdateByROWID()) 
2551         wxChar rowid
[wxDB_ROWID_LEN
+1]; 
2553         // Get the ROWID value.  If not successful retreiving the ROWID, 
2554         // simply fall down through the code and build the WHERE clause 
2555         // based on the key fields. 
2556         if (SQLGetData(hstmt
, (UWORD
)(m_numCols
+1), SQL_C_WXCHAR
, (UCHAR
*) rowid
, sizeof(rowid
), &cb
) == SQL_SUCCESS
) 
2558             whereClause 
+= pDb
->SQLTableName(queryTableName
); 
2559 //            whereClause += queryTableName; 
2560             whereClause 
+= wxT(".ROWID = '"); 
2561             whereClause 
+= rowid
; 
2562             whereClause 
+= wxT("'"); 
2566     // If unable to use the ROWID, build a where clause from the keyfields 
2567     if (wxStrlen(whereClause
) == 0) 
2568         BuildWhereClause(whereClause
, DB_WHERE_KEYFIELDS
, queryTableName
); 
2570     // Requery the record 
2571     where 
= whereClause
; 
2576     if (result 
&& !GetNext()) 
2579     // Switch back to original cursor 
2580     SetCursor(&currCursor
); 
2582     // Free the internal cursor 
2583     if (SQLFreeStmt(hstmtInternal
, SQL_CLOSE
) != SQL_SUCCESS
) 
2584         pDb
->DispAllErrors(henv
, hdbc
, hstmtInternal
); 
2586     // Restore the original where and order by clauses 
2588     orderBy 
= saveOrderBy
; 
2592 }  // wxDbTable::Refresh() 
2595 /********** wxDbTable::SetColNull() **********/ 
2596 bool wxDbTable::SetColNull(UWORD colNumber
, bool set
) 
2598     if (colNumber 
< m_numCols
) 
2600         colDefs
[colNumber
].Null 
= set
; 
2601         if (set
)  // Blank out the values in the member variable 
2602            ClearMemberVar(colNumber
, false);  // Must call with false here, or infinite recursion will happen 
2604         setCbValueForColumn(colNumber
); 
2611 }  // wxDbTable::SetColNull() 
2614 /********** wxDbTable::SetColNull() **********/ 
2615 bool wxDbTable::SetColNull(const wxString 
&colName
, bool set
) 
2618     for (colNumber 
= 0; colNumber 
< m_numCols
; colNumber
++) 
2620         if (!wxStricmp(colName
, colDefs
[colNumber
].ColName
)) 
2624     if (colNumber 
< m_numCols
) 
2626         colDefs
[colNumber
].Null 
= set
; 
2627         if (set
)  // Blank out the values in the member variable 
2628            ClearMemberVar((UWORD
)colNumber
,false);  // Must call with false here, or infinite recursion will happen 
2630         setCbValueForColumn(colNumber
); 
2637 }  // wxDbTable::SetColNull() 
2640 /********** wxDbTable::GetNewCursor() **********/ 
2641 HSTMT 
*wxDbTable::GetNewCursor(bool setCursor
, bool bindColumns
) 
2643     HSTMT 
*newHSTMT 
= new HSTMT
; 
2648     if (SQLAllocStmt(hdbc
, newHSTMT
) != SQL_SUCCESS
) 
2650         pDb
->DispAllErrors(henv
, hdbc
); 
2655     if (SQLSetStmtOption(*newHSTMT
, SQL_CURSOR_TYPE
, cursorType
) != SQL_SUCCESS
) 
2657         pDb
->DispAllErrors(henv
, hdbc
, *newHSTMT
); 
2664         if (!bindCols(*newHSTMT
)) 
2672         SetCursor(newHSTMT
); 
2676 }   // wxDbTable::GetNewCursor() 
2679 /********** wxDbTable::DeleteCursor() **********/ 
2680 bool wxDbTable::DeleteCursor(HSTMT 
*hstmtDel
) 
2684     if (!hstmtDel
)  // Cursor already deleted 
2688 ODBC 3.0 says to use this form 
2689     if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS) 
2692     if (SQLFreeStmt(*hstmtDel
, SQL_DROP
) != SQL_SUCCESS
) 
2694         pDb
->DispAllErrors(henv
, hdbc
); 
2702 }  // wxDbTable::DeleteCursor() 
2704 ////////////////////////////////////////////////////////////// 
2705 // wxDbGrid support functions 
2706 ////////////////////////////////////////////////////////////// 
2708 void wxDbTable::SetRowMode(const rowmode_t rowmode
) 
2710     if (!m_hstmtGridQuery
) 
2712         m_hstmtGridQuery 
= GetNewCursor(false,false); 
2713         if (!bindCols(*m_hstmtGridQuery
)) 
2717     m_rowmode 
= rowmode
; 
2720         case WX_ROW_MODE_QUERY
: 
2721             SetCursor(m_hstmtGridQuery
); 
2723         case WX_ROW_MODE_INDIVIDUAL
: 
2724             SetCursor(hstmtDefault
); 
2729 }  // wxDbTable::SetRowMode() 
2732 wxVariant 
wxDbTable::GetColumn(const int colNumber
) const 
2735     if ((colNumber 
< m_numCols
) && (!IsColNull((UWORD
)colNumber
))) 
2737         switch (colDefs
[colNumber
].SqlCtype
) 
2740     #if defined(SQL_WCHAR) 
2743     #if defined(SQL_WVARCHAR) 
2749                 val 
= (wxChar 
*)(colDefs
[colNumber
].PtrDataObj
); 
2753                 val 
= *(long *)(colDefs
[colNumber
].PtrDataObj
); 
2757                 val 
= (long int )(*(short *)(colDefs
[colNumber
].PtrDataObj
)); 
2760                 val 
= (long)(*(unsigned long *)(colDefs
[colNumber
].PtrDataObj
)); 
2763                 val 
= (long)(*(wxChar 
*)(colDefs
[colNumber
].PtrDataObj
)); 
2765             case SQL_C_UTINYINT
: 
2766                 val 
= (long)(*(wxChar 
*)(colDefs
[colNumber
].PtrDataObj
)); 
2769                 val 
= (long)(*(UWORD 
*)(colDefs
[colNumber
].PtrDataObj
)); 
2772                 val 
= (DATE_STRUCT 
*)(colDefs
[colNumber
].PtrDataObj
); 
2775                 val 
= (TIME_STRUCT 
*)(colDefs
[colNumber
].PtrDataObj
); 
2777             case SQL_C_TIMESTAMP
: 
2778                 val 
= (TIMESTAMP_STRUCT 
*)(colDefs
[colNumber
].PtrDataObj
); 
2781                 val 
= *(double *)(colDefs
[colNumber
].PtrDataObj
); 
2788 }  // wxDbTable::GetCol() 
2791 void wxDbTable::SetColumn(const int colNumber
, const wxVariant val
) 
2793     //FIXME: Add proper wxDateTime support to wxVariant.. 
2796     SetColNull((UWORD
)colNumber
, val
.IsNull()); 
2800         if ((colDefs
[colNumber
].SqlCtype 
== SQL_C_DATE
) 
2801             || (colDefs
[colNumber
].SqlCtype 
== SQL_C_TIME
) 
2802             || (colDefs
[colNumber
].SqlCtype 
== SQL_C_TIMESTAMP
)) 
2804             //Returns null if invalid! 
2805             if (!dateval
.ParseDate(val
.GetString())) 
2806                 SetColNull((UWORD
)colNumber
, true); 
2809         switch (colDefs
[colNumber
].SqlCtype
) 
2812     #if defined(SQL_WCHAR) 
2815     #if defined(SQL_WVARCHAR) 
2821                 csstrncpyt((wxChar 
*)(colDefs
[colNumber
].PtrDataObj
), 
2822                            val
.GetString().c_str(), 
2823                            colDefs
[colNumber
].SzDataObj
-1);  //TODO: glt ??? * sizeof(wxChar) ??? 
2827                 *(long *)(colDefs
[colNumber
].PtrDataObj
) = val
; 
2831                 *(short *)(colDefs
[colNumber
].PtrDataObj
) = (short)val
.GetLong(); 
2834                 *(unsigned long *)(colDefs
[colNumber
].PtrDataObj
) = val
.GetLong(); 
2837                 *(wxChar 
*)(colDefs
[colNumber
].PtrDataObj
) = val
.GetChar(); 
2839             case SQL_C_UTINYINT
: 
2840                 *(wxChar 
*)(colDefs
[colNumber
].PtrDataObj
) = val
.GetChar(); 
2843                 *(unsigned short *)(colDefs
[colNumber
].PtrDataObj
) = (unsigned short)val
.GetLong(); 
2845             //FIXME: Add proper wxDateTime support to wxVariant.. 
2848                     DATE_STRUCT 
*dataptr 
= 
2849                         (DATE_STRUCT 
*)colDefs
[colNumber
].PtrDataObj
; 
2851                     dataptr
->year   
= (SWORD
)dateval
.GetYear(); 
2852                     dataptr
->month  
= (UWORD
)(dateval
.GetMonth()+1); 
2853                     dataptr
->day    
= (UWORD
)dateval
.GetDay(); 
2858                     TIME_STRUCT 
*dataptr 
= 
2859                         (TIME_STRUCT 
*)colDefs
[colNumber
].PtrDataObj
; 
2861                     dataptr
->hour   
= dateval
.GetHour(); 
2862                     dataptr
->minute 
= dateval
.GetMinute(); 
2863                     dataptr
->second 
= dateval
.GetSecond(); 
2866             case SQL_C_TIMESTAMP
: 
2868                     TIMESTAMP_STRUCT 
*dataptr 
= 
2869                         (TIMESTAMP_STRUCT 
*)colDefs
[colNumber
].PtrDataObj
; 
2870                     dataptr
->year   
= (SWORD
)dateval
.GetYear(); 
2871                     dataptr
->month  
= (UWORD
)(dateval
.GetMonth()+1); 
2872                     dataptr
->day    
= (UWORD
)dateval
.GetDay(); 
2874                     dataptr
->hour   
= dateval
.GetHour(); 
2875                     dataptr
->minute 
= dateval
.GetMinute(); 
2876                     dataptr
->second 
= dateval
.GetSecond(); 
2880                 *(double *)(colDefs
[colNumber
].PtrDataObj
) = val
; 
2885     }  // if (!val.IsNull()) 
2886 }  // wxDbTable::SetCol() 
2889 GenericKey 
wxDbTable::GetKey() 
2894     blk 
= malloc(m_keysize
); 
2895     blkptr 
= (wxChar 
*) blk
; 
2898     for (i
=0; i 
< m_numCols
; i
++) 
2900         if (colDefs
[i
].KeyField
) 
2902             memcpy(blkptr
,colDefs
[i
].PtrDataObj
, colDefs
[i
].SzDataObj
); 
2903             blkptr 
+= colDefs
[i
].SzDataObj
; 
2907     GenericKey k 
= GenericKey(blk
, m_keysize
); 
2911 }  // wxDbTable::GetKey() 
2914 void wxDbTable::SetKey(const GenericKey
& k
) 
2920     blkptr 
= (wxChar 
*)blk
; 
2923     for (i
=0; i 
< m_numCols
; i
++) 
2925         if (colDefs
[i
].KeyField
) 
2927             SetColNull((UWORD
)i
, false); 
2928             memcpy(colDefs
[i
].PtrDataObj
, blkptr
, colDefs
[i
].SzDataObj
); 
2929             blkptr 
+= colDefs
[i
].SzDataObj
; 
2932 }  // wxDbTable::SetKey() 
2935 #endif  // wxUSE_ODBC