-       // Decrement the wxDB table count
-       if (pDb)
-               pDb->nTables--;
-
-       // Delete memory allocated for column definitions
-       if (colDefs)
-               delete [] colDefs;
-
-       // Free statement handles
-       if (!queryOnly)
-       {
-               if (hstmtInsert)
-                       if (SQLFreeStmt(hstmtInsert, SQL_DROP) != SQL_SUCCESS)
-                               pDb->DispAllErrors(henv, hdbc);
-               if (hstmtDelete)
-                       if (SQLFreeStmt(hstmtDelete, SQL_DROP) != SQL_SUCCESS)
-                               pDb->DispAllErrors(henv, hdbc);
-               if (hstmtUpdate)
-                       if (SQLFreeStmt(hstmtUpdate, SQL_DROP) != SQL_SUCCESS)
-                               pDb->DispAllErrors(henv, hdbc);
-       }
-       if (hstmtInternal)
-               if (SQLFreeStmt(hstmtInternal, SQL_DROP) != SQL_SUCCESS)
-                       pDb->DispAllErrors(henv, hdbc);
-
-       // Delete dynamically allocated cursors
-       if (hstmtDefault)
-               DeleteCursor(hstmtDefault);
-       if (hstmtCount)
-               DeleteCursor(hstmtCount);
-
-}  // wxTable::~wxTable()
-
-/********** wxTable::Open() **********/
-bool wxTable::Open(void)
-{
-       if (!pDb)
-               return FALSE;
-
-       int i;
-       char sqlStmt[DB_MAX_STATEMENT_LEN];
-
-       // Verify that the table exists in the database
-       if (!pDb->TableExists(tableName,NULL,tablePath))
-       {
-               char s[128];
-               sprintf(s, "Error opening '%s', table/view does not exist in the database.", tableName);
-               pDb->LogError(s);
-               return(FALSE);
-       }
-
-       // Bind the member variables for field exchange between
-       // the wxTable object and the ODBC record.
-       if (!queryOnly)
-       {
-               if (!bindInsertParams())                                        // Inserts
-                       return(FALSE);
-               if (!bindUpdateParams())                                        // Updates
-                       return(FALSE);
-       }
-       if (!bindCols(*hstmtDefault))                                   // Selects
-               return(FALSE);
-       if (!bindCols(hstmtInternal))                                   // Internal use only
-               return(FALSE);
-       /*
-        * Do NOT bind the hstmtCount cursor!!!
-        */
-
-       // Build an insert statement using parameter markers
-       if (!queryOnly && noCols > 0)
-       {
-               bool needComma = FALSE;
-               sprintf(sqlStmt, "INSERT INTO %s (", tableName);
-               for (i = 0; i < noCols; i++)
-               {
-                       if (! colDefs[i].InsertAllowed)
-                               continue;
-                       if (needComma)
-                               strcat(sqlStmt, ",");
-                       strcat(sqlStmt, colDefs[i].ColName);
-                       needComma = TRUE;
-               }
-               needComma = FALSE;
-               strcat(sqlStmt, ") VALUES (");
-               for (i = 0; i < noCols; i++)
-               {
-                       if (! colDefs[i].InsertAllowed)
-                               continue;
-                       if (needComma)
-                               strcat(sqlStmt, ",");
-                       strcat(sqlStmt, "?");
-                       needComma = TRUE;
-               }
-               strcat(sqlStmt, ")");
-
-//             pDb->WriteSqlLog(sqlStmt);
-
-               // Prepare the insert statement for execution
-               if (SQLPrepare(hstmtInsert, (UCHAR FAR *) sqlStmt, SQL_NTS) != SQL_SUCCESS)
-                       return(pDb->DispAllErrors(henv, hdbc, hstmtInsert));
-       }
-       
-       // Completed successfully
-       return(TRUE);
-
-}  // wxTable::Open()
-
-/********** wxTable::Query() **********/
-bool wxTable::Query(bool forUpdate, bool distinct)
-{
-
-       return(query(DB_SELECT_WHERE, forUpdate, distinct));
-
-}  // wxTable::Query()
-
-/********** wxTable::QueryBySqlStmt() **********/
-bool wxTable::QueryBySqlStmt(char *pSqlStmt)
-{
-       pDb->WriteSqlLog(pSqlStmt);
-
-       return(query(DB_SELECT_STATEMENT, FALSE, FALSE, pSqlStmt));
-
-}  // wxTable::QueryBySqlStmt()
-
-/********** wxTable::QueryMatching() **********/
-bool wxTable::QueryMatching(bool forUpdate, bool distinct)
-{
-
-       return(query(DB_SELECT_MATCHING, forUpdate, distinct));
-
-}  // wxTable::QueryMatching()
-
-/********** wxTable::QueryOnKeyFields() **********/
-bool wxTable::QueryOnKeyFields(bool forUpdate, bool distinct)
-{
-
-       return(query(DB_SELECT_KEYFIELDS, forUpdate, distinct));
-
-}  // wxTable::QueryOnKeyFields()
-
-/********** wxTable::query() **********/
-bool wxTable::query(int queryType, bool forUpdate, bool distinct, char *pSqlStmt)
-{
-       char sqlStmt[DB_MAX_STATEMENT_LEN];
-
-       // Set the selectForUpdate member variable
-       if (forUpdate)
-               // The user may wish to select for update, but the DBMS may not be capable
-               selectForUpdate = CanSelectForUpdate();
-       else
-               selectForUpdate = FALSE;
-
-       // Set the SQL SELECT string
-       if (queryType != DB_SELECT_STATEMENT)                           // A select statement was not passed in,
-       {                                                                                                                               // so generate a select statement.
-               GetSelectStmt(sqlStmt, queryType, distinct);
-               pDb->WriteSqlLog(sqlStmt);
-       }
-
-       // Make sure the cursor is closed first
-       if (! CloseCursor(hstmt))
-               return(FALSE);
-
-       // Execute the SQL SELECT statement
-       int retcode;
-
-       retcode = SQLExecDirect(hstmt, (UCHAR FAR *) (queryType == DB_SELECT_STATEMENT ? pSqlStmt : sqlStmt), SQL_NTS);
-       if (retcode != SQL_SUCCESS && retcode != SQL_SUCCESS_WITH_INFO)
-               return(pDb->DispAllErrors(henv, hdbc, hstmt));
-
-       // Completed successfully
-       return(TRUE);
-
-}  // wxTable::query()
-
-/********** wxTable::GetSelectStmt() **********/
-void wxTable::GetSelectStmt(char *pSqlStmt, int typeOfSelect, bool distinct)
-{
-       char whereClause[DB_MAX_WHERE_CLAUSE_LEN];
-
-       whereClause[0] = 0;
-
-       // Build a select statement to query the database
-       strcpy(pSqlStmt, "SELECT ");
-
-       // SELECT DISTINCT values only?
-       if (distinct)
-               strcat(pSqlStmt, "DISTINCT ");
-
-       // Was a FROM clause specified to join tables to the base table?
-       // Available for ::Query() only!!!
-       bool appendFromClause = FALSE;
-       if (typeOfSelect == DB_SELECT_WHERE && from && strlen(from))
-               appendFromClause = TRUE;
-
-       // Add the column list
-       int i;
-       for (i = 0; i < noCols; i++)
-       {
-               // If joining tables, the base table column names must be qualified to avoid ambiguity
-               if (appendFromClause)
-               {
-                       strcat(pSqlStmt, queryTableName);
-                       strcat(pSqlStmt, ".");
-               }
-               strcat(pSqlStmt, colDefs[i].ColName);
-               if (i + 1 < noCols)
-                       strcat(pSqlStmt, ",");
-       }
-
-       // If the datasource supports ROWID, get this column as well.  Exception: Don't retrieve
-       // the ROWID if querying distinct records.  The rowid will always be unique.
-       if (!distinct && CanUpdByROWID())
-       {
-               // If joining tables, the base table column names must be qualified to avoid ambiguity
-               if (appendFromClause)
-               {
-                       strcat(pSqlStmt, ",");
-                       strcat(pSqlStmt, queryTableName);
-                       strcat(pSqlStmt, ".ROWID");
-               }
-               else
-                       strcat(pSqlStmt, ",ROWID");
-       }
-
-       // Append the FROM tablename portion
-       strcat(pSqlStmt, " FROM ");
-       strcat(pSqlStmt, queryTableName);
-
-       // Sybase uses the HOLDLOCK keyword to lock a record during query.
-       // The HOLDLOCK keyword follows the table name in the from clause.
-       // Each table in the from clause must specify HOLDLOCK or
-       // NOHOLDLOCK (the default).  Note: The "FOR UPDATE" clause
-       // is parsed but ignored in SYBASE Transact-SQL.
-       if (selectForUpdate && (pDb->Dbms() == dbmsSYBASE_ASA || pDb->Dbms() == dbmsSYBASE_ASE))
-               strcat(pSqlStmt, " HOLDLOCK");
-
-       if (appendFromClause)
-               strcat(pSqlStmt, from);
-
-       // Append the WHERE clause.  Either append the where clause for the class
-       // or build a where clause.  The typeOfSelect determines this.
-       switch(typeOfSelect)
-       {
-       case DB_SELECT_WHERE:
-               if (where && strlen(where))     // May not want a where clause!!!
-               {
-                       strcat(pSqlStmt, " WHERE ");
-                       strcat(pSqlStmt, where);
-               }
-               break;
-       case DB_SELECT_KEYFIELDS:
-               GetWhereClause(whereClause, DB_WHERE_KEYFIELDS);
-               if (strlen(whereClause))
-               {
-                       strcat(pSqlStmt, " WHERE ");
-                       strcat(pSqlStmt, whereClause);
-               }
-               break;
-       case DB_SELECT_MATCHING:
-               GetWhereClause(whereClause, DB_WHERE_MATCHING);
-               if (strlen(whereClause))
-               {
-                       strcat(pSqlStmt, " WHERE ");
-                       strcat(pSqlStmt, whereClause);
-               }
-               break;
-       }
-
-       // Append the ORDER BY clause
-       if (orderBy && strlen(orderBy))
-       {
-               strcat(pSqlStmt, " ORDER BY ");
-               strcat(pSqlStmt, orderBy);
-       }
-
-       // SELECT FOR UPDATE if told to do so and the datasource is capable.  Sybase
-       // parses the FOR UPDATE clause but ignores it.  See the comment above on the
-       // HOLDLOCK for Sybase.
-       if (selectForUpdate && CanSelectForUpdate())
-               strcat(pSqlStmt, " FOR UPDATE");
-
-}  // wxTable::GetSelectStmt()
-
-/********** wxTable::getRec() **********/
-bool wxTable::getRec(UWORD fetchType)
-{
-       RETCODE retcode;
-
-#if wxODBC_FWD_ONLY_CURSORS
-
-       // Fetch the NEXT, PREV, FIRST or LAST record, depending on fetchType
-       UDWORD  cRowsFetched;
-       UWORD   rowStatus;
-       if ((retcode = SQLExtendedFetch(hstmt, fetchType, 0, &cRowsFetched, &rowStatus)) != SQL_SUCCESS)
-               if (retcode == SQL_NO_DATA_FOUND)
-                       return(FALSE);
-               else
-                       return(pDb->DispAllErrors(henv, hdbc, hstmt));
+
+    // Decrement the wxDb table count
+    if (pDb)
+        pDb->decrementTableCount();
+
+    // Delete memory allocated for column definitions
+    if (colDefs)
+        delete [] colDefs;
+
+    // Free statement handles
+    if (!queryOnly)
+    {
+        if (hstmtInsert)
+        {
+/*
+ODBC 3.0 says to use this form
+            if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS)
+*/
+            if (SQLFreeStmt(hstmtInsert, SQL_DROP) != SQL_SUCCESS)
+                pDb->DispAllErrors(henv, hdbc);
+        }
+
+        if (hstmtDelete)
+        {
+/*
+ODBC 3.0 says to use this form
+            if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS)
+*/
+            if (SQLFreeStmt(hstmtDelete, SQL_DROP) != SQL_SUCCESS)
+                pDb->DispAllErrors(henv, hdbc);
+        }
+
+        if (hstmtUpdate)
+        {
+/*
+ODBC 3.0 says to use this form
+            if (SQLFreeHandle(*hstmtDel, SQL_DROP) != SQL_SUCCESS)
+*/
+            if (SQLFreeStmt(hstmtUpdate, SQL_DROP) != SQL_SUCCESS)
+                pDb->DispAllErrors(henv, hdbc);
+        }
+    }
+
+    if (hstmtInternal)
+    {
+        if (SQLFreeStmt(hstmtInternal, SQL_DROP) != SQL_SUCCESS)
+            pDb->DispAllErrors(henv, hdbc);
+    }
+
+    // Delete dynamically allocated cursors
+    if (hstmtDefault)
+        DeleteCursor(hstmtDefault);
+
+    if (hstmtCount)
+        DeleteCursor(hstmtCount);
+
+    if (m_hstmtGridQuery)
+        DeleteCursor(m_hstmtGridQuery);
+
+}  // wxDbTable::cleanup()
+
+
+/***************************** PRIVATE FUNCTIONS *****************************/
+
+
+/********** wxDbTable::bindParams() **********/
+bool wxDbTable::bindParams(bool forUpdate)
+{
+    wxASSERT(!queryOnly);
+    if (queryOnly)
+        return(FALSE);
+
+    SWORD   fSqlType    = 0;
+    SDWORD  precision   = 0;
+    SWORD   scale       = 0;
+
+    // Bind each column of the table that should be bound
+    // to a parameter marker
+    int i;
+    UWORD colNo;
+
+    for (i=0, colNo=1; i < noCols; i++)
+    {
+        if (forUpdate)
+        {
+            if (!colDefs[i].Updateable)
+                continue;
+        }
+        else
+        {
+            if (!colDefs[i].InsertAllowed)
+                continue;
+        }
+
+        switch(colDefs[i].DbDataType)
+        {
+            case DB_DATA_TYPE_VARCHAR:
+                fSqlType = pDb->GetTypeInfVarchar().FsqlType;
+                precision = colDefs[i].SzDataObj;
+                scale = 0;
+                if (colDefs[i].Null)
+                    colDefs[i].CbValue = SQL_NULL_DATA;
+                else
+                    colDefs[i].CbValue = SQL_NTS;
+                break;
+            case DB_DATA_TYPE_INTEGER:
+                fSqlType = pDb->GetTypeInfInteger().FsqlType;
+                precision = pDb->GetTypeInfInteger().Precision;
+                scale = 0;
+                if (colDefs[i].Null)
+                    colDefs[i].CbValue = SQL_NULL_DATA;
+                else
+                    colDefs[i].CbValue = 0;
+                break;
+            case DB_DATA_TYPE_FLOAT:
+                fSqlType = pDb->GetTypeInfFloat().FsqlType;
+                precision = pDb->GetTypeInfFloat().Precision;
+                scale = pDb->GetTypeInfFloat().MaximumScale;
+                // SQL Sybase Anywhere v5.5 returned a negative number for the
+                // MaxScale.  This caused ODBC to kick out an error on ibscale.
+                // I check for this here and set the scale = precision.
+                //if (scale < 0)
+                // scale = (short) precision;
+                if (colDefs[i].Null)
+                    colDefs[i].CbValue = SQL_NULL_DATA;
+                else
+                    colDefs[i].CbValue = 0;
+                break;
+            case DB_DATA_TYPE_DATE:
+                fSqlType = pDb->GetTypeInfDate().FsqlType;
+                precision = pDb->GetTypeInfDate().Precision;
+                scale = 0;
+                if (colDefs[i].Null)
+                    colDefs[i].CbValue = SQL_NULL_DATA;
+                else
+                    colDefs[i].CbValue = 0;
+                break;
+            case DB_DATA_TYPE_BLOB:
+                fSqlType = pDb->GetTypeInfBlob().FsqlType;
+                precision = -1;
+                scale = 0;
+                if (colDefs[i].Null)
+                    colDefs[i].CbValue = SQL_NULL_DATA;
+                else
+                    colDefs[i].CbValue = SQL_LEN_DATA_AT_EXEC(colDefs[i].SzDataObj);
+                break;
+        }
+        if (forUpdate)
+        {
+            if (SQLBindParameter(hstmtUpdate, colNo++, SQL_PARAM_INPUT, colDefs[i].SqlCtype,
+                                 fSqlType, precision, scale, (UCHAR*) colDefs[i].PtrDataObj,
+                                 precision+1, &colDefs[i].CbValue) != SQL_SUCCESS)
+            {
+                return(pDb->DispAllErrors(henv, hdbc, hstmtUpdate));
+            }
+        }
+        else
+        {
+            if (SQLBindParameter(hstmtInsert, colNo++, SQL_PARAM_INPUT, colDefs[i].SqlCtype,
+                                 fSqlType, precision, scale, (UCHAR*) colDefs[i].PtrDataObj,
+                                 precision+1,&colDefs[i].CbValue) != SQL_SUCCESS)
+            {
+                return(pDb->DispAllErrors(henv, hdbc, hstmtInsert));
+            }
+        }
+    }
+
+    // Completed successfully
+    return(TRUE);
+
+}  // wxDbTable::bindParams()
+
+
+/********** wxDbTable::bindInsertParams() **********/
+bool wxDbTable::bindInsertParams(void)
+{
+    return bindParams(FALSE);
+}  // wxDbTable::bindInsertParams()
+
+
+/********** wxDbTable::bindUpdateParams() **********/
+bool wxDbTable::bindUpdateParams(void)
+{
+    return bindParams(TRUE);
+}  // wxDbTable::bindUpdateParams()
+
+
+/********** wxDbTable::bindCols() **********/
+bool wxDbTable::bindCols(HSTMT cursor)
+{
+    static SDWORD cb;
+
+    // Bind each column of the table to a memory address for fetching data
+    UWORD i;
+    for (i = 0; i < noCols; i++)
+    {
+        cb = colDefs[i].CbValue;
+        if (SQLBindCol(cursor, (UWORD)(i+1), colDefs[i].SqlCtype, (UCHAR*) colDefs[i].PtrDataObj,
+                       colDefs[i].SzDataObj, &cb ) != SQL_SUCCESS)
+          return (pDb->DispAllErrors(henv, hdbc, cursor));
+    }
+
+    // Completed successfully
+    return(TRUE);
+
+}  // wxDbTable::bindCols()
+
+
+/********** wxDbTable::getRec() **********/
+bool wxDbTable::getRec(UWORD fetchType)
+{
+    RETCODE retcode;
+
+    if (!pDb->FwdOnlyCursors())
+    {
+        // Fetch the NEXT, PREV, FIRST or LAST record, depending on fetchType
+        UDWORD  cRowsFetched;
+        UWORD   rowStatus;
+
+        retcode = SQLExtendedFetch(hstmt, fetchType, 0, &cRowsFetched, &rowStatus);
+        if (retcode != SQL_SUCCESS && retcode != SQL_SUCCESS_WITH_INFO)
+        {
+            if (retcode == SQL_NO_DATA_FOUND)
+                return(FALSE);
+            else
+                return(pDb->DispAllErrors(henv, hdbc, hstmt));
+        }
+        else
+        {
+            // Set the Null member variable to indicate the Null state
+            // of each column just read in.
+            int i;
+            for (i = 0; i < noCols; i++)
+                colDefs[i].Null = (colDefs[i].CbValue == SQL_NULL_DATA);
+        }
+    }
+    else
+    {
+        // Fetch the next record from the record set
+        retcode = SQLFetch(hstmt);
+        if (retcode != SQL_SUCCESS && retcode != SQL_SUCCESS_WITH_INFO)
+        {
+            if (retcode == SQL_NO_DATA_FOUND)
+                return(FALSE);
+            else
+                return(pDb->DispAllErrors(henv, hdbc, hstmt));
+        }
+        else
+        {
+            // Set the Null member variable to indicate the Null state
+            // of each column just read in.
+            int i;
+            for (i = 0; i < noCols; i++)
+                colDefs[i].Null = (colDefs[i].CbValue == SQL_NULL_DATA);
+        }
+    }
+
+    // Completed successfully
+    return(TRUE);
+
+}  // wxDbTable::getRec()
+
+
+/********** wxDbTable::execDelete() **********/
+bool wxDbTable::execDelete(const wxString &pSqlStmt)
+{
+    RETCODE retcode;
+
+    // Execute the DELETE statement
+    retcode = SQLExecDirect(hstmtDelete, (UCHAR FAR *) pSqlStmt.c_str(), SQL_NTS);
+
+    if (retcode == SQL_SUCCESS ||
+        retcode == SQL_NO_DATA_FOUND ||
+        retcode == SQL_SUCCESS_WITH_INFO)
+    {
+        // Record deleted successfully
+        return(TRUE);
+    }
+
+    // Problem deleting record
+    return(pDb->DispAllErrors(henv, hdbc, hstmtDelete));
+
+}  // wxDbTable::execDelete()
+
+
+/********** wxDbTable::execUpdate() **********/
+bool wxDbTable::execUpdate(const wxString &pSqlStmt)
+{
+    RETCODE retcode;
+
+    // Execute the UPDATE statement
+    retcode = SQLExecDirect(hstmtUpdate, (UCHAR FAR *) pSqlStmt.c_str(), SQL_NTS);
+
+    if (retcode == SQL_SUCCESS ||
+        retcode == SQL_NO_DATA_FOUND ||
+        retcode == SQL_SUCCESS_WITH_INFO)
+    {
+        // Record updated successfully
+        return(TRUE);
+    }
+    else if (retcode == SQL_NEED_DATA)
+    {
+        PTR pParmID;
+        while ((retcode = SQLParamData(hstmtUpdate, &pParmID) == SQL_NEED_DATA))
+        {
+            // Find the parameter
+            int i;
+            for (i=0; i < noCols; i++)
+            {
+                if (colDefs[i].PtrDataObj == pParmID)
+                {
+                    // We found it.  Store the parameter.
+                    retcode = SQLPutData(hstmtUpdate, pParmID, colDefs[i].SzDataObj);
+                    if (retcode != SQL_SUCCESS)
+                    {
+                        pDb->DispNextError();
+                        return pDb->DispAllErrors(henv, hdbc, hstmtUpdate);
+                    }
+                    break;
+                }
+            }
+        }
+        if (retcode == SQL_SUCCESS ||
+            retcode == SQL_NO_DATA_FOUND ||
+            retcode == SQL_SUCCESS_WITH_INFO)
+        {
+            // Record updated successfully
+            return(TRUE);
+        }
+    }
+
+    // Problem updating record
+    return(pDb->DispAllErrors(henv, hdbc, hstmtUpdate));
+
+}  // wxDbTable::execUpdate()
+
+
+/********** wxDbTable::query() **********/
+bool wxDbTable::query(int queryType, bool forUpdate, bool distinct, const wxString &pSqlStmt)
+{
+    wxString sqlStmt;
+
+    if (forUpdate)
+        // The user may wish to select for update, but the DBMS may not be capable
+        selectForUpdate = CanSelectForUpdate();
+    else
+        selectForUpdate = FALSE;
+
+    // Set the SQL SELECT string
+    if (queryType != DB_SELECT_STATEMENT)               // A select statement was not passed in,
+    {                                                   // so generate a select statement.
+        BuildSelectStmt(sqlStmt, queryType, distinct);
+        pDb->WriteSqlLog(sqlStmt);
+    }
+
+    // Make sure the cursor is closed first
+    if (!CloseCursor(hstmt))
+        return(FALSE);
+
+    // Execute the SQL SELECT statement
+    int retcode;
+    retcode = SQLExecDirect(hstmt, (UCHAR FAR *) (queryType == DB_SELECT_STATEMENT ? pSqlStmt.c_str() : sqlStmt.c_str()), SQL_NTS);
+    if (retcode != SQL_SUCCESS && retcode != SQL_SUCCESS_WITH_INFO)
+        return(pDb->DispAllErrors(henv, hdbc, hstmt));
+
+    // Completed successfully
+    return(TRUE);
+
+}  // wxDbTable::query()
+
+
+/***************************** PUBLIC FUNCTIONS *****************************/
+
+
+/********** wxDbTable::Open() **********/
+bool wxDbTable::Open(bool checkPrivileges, bool checkTableExists)
+{
+    if (!pDb)
+        return FALSE;
+
+    int i;
+    wxString sqlStmt;
+    wxString s;
+//    int NumKeyCols=0;
+
+    // Calculate the maximum size of the concatenated
+    // keys for use with wxDbGrid
+    m_keysize = 0;
+    for (i=0; i < noCols; i++)
+    {
+        if (colDefs[i].KeyField)
+        {
+//            NumKeyCols++;
+            m_keysize += colDefs[i].SzDataObj;
+        }
+    }
+
+    s.Empty();
+    // Verify that the table exists in the database
+    if (checkTableExists && !pDb->TableExists(tableName, pDb->GetUsername(), tablePath))
+    {
+        s = wxT("Table/view does not exist in the database");
+        if ( *(pDb->dbInf.accessibleTables) == wxT('Y'))
+            s += wxT(", or you have no permissions.\n");
+        else
+            s += wxT(".\n");
+    }
+    else if (checkPrivileges)
+    {
+        // Verify the user has rights to access the table.
+        // Shortcut boolean evaluation to optimize out call to
+        // TablePrivileges
+        //
+        // Unfortunately this optimization doesn't seem to be
+        // reliable!
+        if (// *(pDb->dbInf.accessibleTables) == 'N' &&
+            !pDb->TablePrivileges(tableName,wxT("SELECT"), pDb->GetUsername(), pDb->GetUsername(), tablePath))
+            s = wxT("Current logged in user does not have sufficient privileges to access this table.\n");
+    }
+
+    if (!s.IsEmpty())
+    {
+        wxString p;
+
+        if (!tablePath.IsEmpty())
+            p.Printf(wxT("Error opening '%s/%s'.\n"),tablePath.c_str(),tableName.c_str());
+        else
+            p.Printf(wxT("Error opening '%s'.\n"), tableName.c_str());
+
+        p += s;
+        pDb->LogError(p.GetData());
+
+        return(FALSE);
+    }
+
+    // Bind the member variables for field exchange between
+    // the wxDbTable object and the ODBC record.
+    if (!queryOnly)
+    {
+        if (!bindInsertParams())                    // Inserts
+            return(FALSE);
+
+        if (!bindUpdateParams())                    // Updates
+            return(FALSE);
+    }
+
+    if (!bindCols(*hstmtDefault))                   // Selects
+        return(FALSE);
+
+    if (!bindCols(hstmtInternal))                   // Internal use only
+        return(FALSE);
+
+     /*
+     * Do NOT bind the hstmtCount cursor!!!
+     */
+
+    // Build an insert statement using parameter markers
+    if (!queryOnly && noCols > 0)
+    {
+        bool needComma = FALSE;
+        sqlStmt.Printf(wxT("INSERT INTO %s ("),
+                       pDb->SQLTableName(tableName.c_str()).c_str());
+        for (i = 0; i < noCols; i++)
+        {
+            if (! colDefs[i].InsertAllowed)
+                continue;
+            if (needComma)
+                sqlStmt += wxT(",");
+            sqlStmt += pDb->SQLColumnName(colDefs[i].ColName);
+//            sqlStmt += colDefs[i].ColName;
+            needComma = TRUE;
+        }
+        needComma = FALSE;
+        sqlStmt += wxT(") VALUES (");
+
+        int insertableCount = 0;
+
+        for (i = 0; i < noCols; i++)
+        {
+            if (! colDefs[i].InsertAllowed)
+                continue;
+            if (needComma)
+                sqlStmt += wxT(",");
+            sqlStmt += wxT("?");
+            needComma = TRUE;
+            insertableCount++;
+        }
+        sqlStmt += wxT(")");
+
+        // Prepare the insert statement for execution
+        if (insertableCount)
+        {
+            if (SQLPrepare(hstmtInsert, (UCHAR FAR *) sqlStmt.c_str(), SQL_NTS) != SQL_SUCCESS)
+                return(pDb->DispAllErrors(henv, hdbc, hstmtInsert));
+        }
+        else
+            insertable= FALSE;
+    }
+
+    // Completed successfully
+    return(TRUE);
+
+}  // wxDbTable::Open()
+
+
+/********** wxDbTable::Query() **********/
+bool wxDbTable::Query(bool forUpdate, bool distinct)
+{
+
+    return(query(DB_SELECT_WHERE, forUpdate, distinct));
+
+}  // wxDbTable::Query()
+
+
+/********** wxDbTable::QueryBySqlStmt() **********/
+bool wxDbTable::QueryBySqlStmt(const wxString &pSqlStmt)
+{
+    pDb->WriteSqlLog(pSqlStmt);
+
+    return(query(DB_SELECT_STATEMENT, FALSE, FALSE, pSqlStmt));
+
+}  // wxDbTable::QueryBySqlStmt()
+
+
+/********** wxDbTable::QueryMatching() **********/
+bool wxDbTable::QueryMatching(bool forUpdate, bool distinct)
+{
+
+    return(query(DB_SELECT_MATCHING, forUpdate, distinct));
+
+}  // wxDbTable::QueryMatching()
+
+
+/********** wxDbTable::QueryOnKeyFields() **********/
+bool wxDbTable::QueryOnKeyFields(bool forUpdate, bool distinct)
+{
+
+    return(query(DB_SELECT_KEYFIELDS, forUpdate, distinct));
+
+}  // wxDbTable::QueryOnKeyFields()
+
+
+/********** wxDbTable::GetPrev() **********/
+bool wxDbTable::GetPrev(void)
+{
+    if (pDb->FwdOnlyCursors())
+    {
+        wxFAIL_MSG(wxT("GetPrev()::Backward scrolling cursors are not enabled for this instance of wxDbTable"));
+        return FALSE;
+    }
+    else
+        return(getRec(SQL_FETCH_PRIOR));
+
+}  // wxDbTable::GetPrev()
+
+
+/********** wxDbTable::operator-- **********/
+bool wxDbTable::operator--(int)
+{
+    if (pDb->FwdOnlyCursors())
+    {
+        wxFAIL_MSG(wxT("operator--:Backward scrolling cursors are not enabled for this instance of wxDbTable"));
+        return FALSE;
+    }
+    else
+        return(getRec(SQL_FETCH_PRIOR));
+
+}  // wxDbTable::operator--
+
+
+/********** wxDbTable::GetFirst() **********/
+bool wxDbTable::GetFirst(void)
+{
+    if (pDb->FwdOnlyCursors())
+    {
+        wxFAIL_MSG(wxT("GetFirst():Backward scrolling cursors are not enabled for this instance of wxDbTable"));
+        return FALSE;
+    }
+    else
+        return(getRec(SQL_FETCH_FIRST));
+
+}  // wxDbTable::GetFirst()
+
+
+/********** wxDbTable::GetLast() **********/
+bool wxDbTable::GetLast(void)
+{
+    if (pDb->FwdOnlyCursors())
+    {
+        wxFAIL_MSG(wxT("GetLast()::Backward scrolling cursors are not enabled for this instance of wxDbTable"));
+        return FALSE;
+    }
+    else
+        return(getRec(SQL_FETCH_LAST));
+
+}  // wxDbTable::GetLast()
+
+
+/********** wxDbTable::BuildDeleteStmt() **********/
+void wxDbTable::BuildDeleteStmt(wxString &pSqlStmt, int typeOfDel, const wxString &pWhereClause)
+{
+    wxASSERT(!queryOnly);
+    if (queryOnly)
+        return;
+
+    wxString whereClause;
+
+    whereClause.Empty();
+
+    // Handle the case of DeleteWhere() and the where clause is blank.  It should
+    // delete all records from the database in this case.
+    if (typeOfDel == DB_DEL_WHERE && (pWhereClause.Length() == 0))
+    {
+        pSqlStmt.Printf(wxT("DELETE FROM %s"),
+                        pDb->SQLTableName(tableName.c_str()).c_str());
+        return;
+    }
+
+    pSqlStmt.Printf(wxT("DELETE FROM %s WHERE "),
+                    pDb->SQLTableName(tableName.c_str()).c_str());
+
+    // Append the WHERE clause to the SQL DELETE statement
+    switch(typeOfDel)
+    {
+        case DB_DEL_KEYFIELDS:
+            // If the datasource supports the ROWID column, build
+            // the where on ROWID for efficiency purposes.
+            // e.g. DELETE FROM PARTS WHERE ROWID = '111.222.333'
+            if (CanUpdByROWID())
+            {
+                SDWORD cb;
+                wxChar   rowid[wxDB_ROWID_LEN+1];
+
+                // Get the ROWID value.  If not successful retreiving the ROWID,
+                // simply fall down through the code and build the WHERE clause
+                // based on the key fields.
+                if (SQLGetData(hstmt, (UWORD)(noCols+1), SQL_C_CHAR, (UCHAR*) rowid, wxDB_ROWID_LEN, &cb) == SQL_SUCCESS)
+                {
+                    pSqlStmt += wxT("ROWID = '");
+                    pSqlStmt += rowid;
+                    pSqlStmt += wxT("'");
+                    break;
+                }
+            }
+            // Unable to delete by ROWID, so build a WHERE
+            // clause based on the keyfields.
+            BuildWhereClause(whereClause, DB_WHERE_KEYFIELDS);
+            pSqlStmt += whereClause;
+            break;
+        case DB_DEL_WHERE:
+            pSqlStmt += pWhereClause;
+            break;
+        case DB_DEL_MATCHING:
+            BuildWhereClause(whereClause, DB_WHERE_MATCHING);
+            pSqlStmt += whereClause;
+            break;
+    }
+
+}  // BuildDeleteStmt()
+
+
+/***** DEPRECATED: use wxDbTable::BuildDeleteStmt(wxString &....) form *****/
+void wxDbTable::BuildDeleteStmt(wxChar *pSqlStmt, int typeOfDel, const wxString &pWhereClause)
+{
+    wxString tempSqlStmt;
+    BuildDeleteStmt(tempSqlStmt, typeOfDel, pWhereClause);
+    wxStrcpy(pSqlStmt, tempSqlStmt);
+}  // wxDbTable::BuildDeleteStmt()
+
+
+/********** wxDbTable::BuildSelectStmt() **********/
+void wxDbTable::BuildSelectStmt(wxString &pSqlStmt, int typeOfSelect, bool distinct)
+{
+    wxString whereClause;
+    whereClause.Empty();
+
+    // Build a select statement to query the database
+    pSqlStmt = wxT("SELECT ");
+
+    // SELECT DISTINCT values only?
+    if (distinct)
+        pSqlStmt += wxT("DISTINCT ");
+
+    // Was a FROM clause specified to join tables to the base table?
+    // Available for ::Query() only!!!
+    bool appendFromClause = FALSE;
+#if wxODBC_BACKWARD_COMPATABILITY
+    if (typeOfSelect == DB_SELECT_WHERE && from && wxStrlen(from))
+        appendFromClause = TRUE;