Article ID: 141802
Article Last Modified on 11/21/2006
119591 How to obtain Microsoft support files from online services
DYNC32.EXE -d
BOOL CDyncol32Set::Open(UINT nOpenType, LPCSTR lpszSql, DWORD
dwOptions)
{
// use one CDatabase Object so we only prompt once for database Open
// ***** allocated in constructor...
if (! m_pDatabase->Open(NULL, FALSE, FALSE, "ODBC;"))
return FALSE;
CTables tables(m_pDatabase);
// prompt for a table to open
CTablesDialog tableName(NULL,m_pDatabase);
if (IDCANCEL == tableName.DoModal())
return FALSE;
// Specify the table to look at
m_pColumns = new CColumns(m_pDatabase);
m_pColumns->m_strTableNameParam = tableName.m_strTableNameSelected;
// Set the database to be the CDynaSet's database if one hasn't
// already been set (this should be the case)
if (m_pColumns->m_pDatabase == NULL)
m_pColumns->m_pDatabase = m_pDatabase;
// Open the recordset to get the column info
if (!m_pColumns->Open(CRecordset::forwardOnly, NULL,
CRecordset::readOnly))
return FALSE;
// Initialize the number of fields dynamically allocated to CDynaSet
m_nFields = 0;
// Loop until we've seen all the columns
while (!m_pColumns->IsEOF())
{
// Allocate a new CColumnData object for the current column
CColumnData *pData = new CColumnData;
// Store the colmun information
pData->m_nDataType = m_pColumns->m_nDataType;
pData->m_strColumnName = "[";
pData->m_strColumnName += m_pColumns->m_strColumnName;
pData->m_strColumnName +="]";
// Allocate an object of the appropriate type to store
// the column data
switch(pData->m_nDataType)
{
case SQL_BIT:
pData->m_pData = (void *)new BOOL;
break;
case SQL_TINYINT:
pData->m_pData = (void *)new BYTE;
break;
case SQL_SMALLINT:
pData->m_pData = (void *)new int;
break;
...
default:
ASSERT(FALSE);
}
// Add the column descriptor to the list and
// increment the number of columns in the
CDynaSet
m_pList.AddTail(pData);
m_nFields++;
// Get the next column's information
m_pColumns->MoveNext();
}
// Free the HSTMT used to get the table info
RETCODE nRetCode;
AFX_SQL_SYNC(::SQLFreeStmt(m_pColumns->m_hstmt, SQL_CLOSE));
// Return the base class if we got this far
lpszSql = m_pColumns->m_strTableNameParam;
// we don't need the columns recordset any more
delete m_pColumns;
return CRecordset::Open(nOpenType, lpszSql, dwOptions);
}
void CDyncol32Set::DoFieldExchange(CFieldExchange* pFX)
{
// Set the type of exhange; same as AppWizard generated
pFX->SetFieldType(CFieldExchange::outputColumn);
// Get a pointer to the first CColumnData object in the list
POSITION rPos = m_pList.GetHeadPosition();
CColumnData *pData = (CColumnData *)m_pList.GetNext(rPos);
// Loop until we've traversed all the columns
while (pData)
{
// Call the appropriate RFX routine for the column's type
switch(pData->m_nDataType)
{
case SQL_BIT:
RFX_Bool(pFX, pData->m_strColumnName, *((BOOL *)(pData-
>m_pData)));
break;
case SQL_TINYINT:
RFX_Byte(pFX, pData->m_strColumnName, *((BYTE *)(pData-
>m_pData)));
break;
case SQL_SMALLINT:
RFX_Int(pFX, pData->m_strColumnName, *((int *)(pData-
>m_pData)));
break;
...
default:
ASSERT(FALSE);
}
// Set pData to NULL if that was the last column
if (rPos)
pData = (CColumnData *)m_pList.GetNext(rPos);
else
pData = NULL;
}
}
void CDyncol32View::RefreshData()
{
VARIANT var;
CRect rect;
LVITEM lvitem;
int iActualItem = 0;
POSITION rPos ;
CString csTemp;
CColumnData *pData = NULL;
VariantInit(&var);
var.vt = VT_ERROR;
var.scode = DISP_E_PARAMNOTFOUND;
if (!m_pSet->IsOpen())
return;
// Get pointer to CDynaSet's CColumnData list.
CPtrList *pList = &m_pSet->m_pList;
m_lstData.GetWindowRect(&rect);
rPos = pList->GetHeadPosition();
pData = (CColumnData *)pList->GetNext(rPos);
// Navigate through the CCOlumnData to get the field names.
// Set the columns header text as field names.
for(int nCol= 0; nCol < (int)m_pSet->m_nFields;nCol++)
{
csTemp = pData->m_strColumnName.AllocSysString();
m_lstData.InsertColumn(0, csTemp, LVCFMT_LEFT,rect.Width() * 1/3,
0);
if (rPos)
pData = (CColumnData *)pList->GetNext(rPos);
else
pData = NULL;
}
// Navigate through the recordset.
for (int iItem = 0; m_pSet->IsEOF() != TRUE < 20; iItem++) // insert
// the items and subitems into the list view.
{
// Get the first element.
rPos = pList->GetHeadPosition();
pData = (CColumnData *)pList->GetNext(rPos);
// Navigate through the column list
// and put the data to the list control.
for (int iSubItem = 0; iSubItem < (int)m_pSet->m_nFields;
iSubItem++)
{
lvitem.mask = LVIF_TEXT | (iSubItem == 0? LVIF_IMAGE : 0);
lvitem.iItem = (iSubItem == 0)? iItem : iActualItem;
lvitem.iSubItem = iSubItem;
csTemp = GetData(pData);
// calculate the main and sub-item strings
// for the current item.
(lvitem.pszText) = csTemp.GetBuffer(csTemp.GetLength());
if (iSubItem == 0)
iActualItem = m_lstData.InsertItem(&lvitem); // insert new
// item.
else
m_lstData.SetItem(&lvitem); // modify existing item
// (the sub-item text).
if (rPos)
pData = (CColumnData *)pList->GetNext(rPos);
else
pData = NULL;
}
m_pSet->MoveNext();
}
}
// Function to convert the data in pData to Cstring.
CString CDyncol32View::GetData(CColumnData *pData)
{
CString csTemp;
CString strBuffer;
CString *pSet;
switch(pData->m_nDataType)
{
case SQL_BIT:
strBuffer = *((BOOL *) (pData->m_pData)) ? _T("TRUE") :
_T("FALSE");
csTemp = strBuffer;
break;
case SQL_TINYINT:
{
char *pBuffer = strBuffer.GetBuffer(32);
_itoa(*((BYTE *)(pData->m_pData)), pBuffer, 10);
}
strBuffer.ReleaseBuffer();
csTemp= strBuffer;
break;
case SQL_SMALLINT:
case SQL_INTEGER:
{
char *pBuffer = strBuffer.GetBuffer(32);
_itoa(*((int *)(pData->m_pData)), pBuffer, 10);
}
strBuffer.ReleaseBuffer();
csTemp = strBuffer;
break;
case SQL_REAL:
{
char *pBuffer = strBuffer.GetBuffer(32);
sprintf(pBuffer, "%#f", *((float *) (pData->m_pData)));
}
strBuffer.ReleaseBuffer();
csTemp = strBuffer;
break;
case SQL_FLOAT:
case SQL_DOUBLE:
{
char *pBuffer = strBuffer.GetBuffer(32);
sprintf(pBuffer, "%#f", *((double *) (pData->m_pData)));
}
strBuffer.ReleaseBuffer();
csTemp = strBuffer;
break;
case SQL_DATE:
case SQL_TIME:
case SQL_TIMESTAMP:
strBuffer = ((CTime *)(pData->m_pData))->Format( "%c" );
csTemp = strBuffer;
break;
case SQL_BINARY:
case SQL_VARBINARY:
csTemp = "Binary Field";
break;
case SQL_DECIMAL: // ODBC default xfer type
case SQL_NUMERIC: // ODBC default xfer type
case SQL_CHAR:
case SQL_VARCHAR:
pSet = (CString *)(pData->m_pData);
csTemp = *pSet;
break;
case SQL_LONGVARCHAR:
case SQL_LONGVARBINARY:
csTemp = "Longvarchar or Longvarbinary";
break;
default:
ASSERT(FALSE);
}
return csTemp;
}
The most fundamental concepts of the sample are contained within the
CColumns and CColumnData objects. The first allows DYNC32 to determine the name
and data type for each column in the table. With this information it is able to
allocate a linked list of CColumnData objects. Each CColumnData object is a
descriptor that contains the name, type and a pointer to a data object of the
correct type. Once this list has been constructed, traversal routines must be
put in the DoFieldExchange() and DoDataExchange() routines to perform the
appropriate RFX/DDX function with the allocated data object. Additional query words: Dync32 kbvc400 kbvc410 kbvc420 kbvc500 kbvc600
Keywords: kbdownload kbcode kbdatabase kbfile kbmdacnosweep kbsample kbusage KB141802