Dinamicamente colunas determinando retornadas para o consumidor
As macros PROVIDER_COLUMN_ENTRY manipulam normalmente o IColumnsInfo::GetColumnsInfo telefonar.No entanto, porque um consumidor pode optar por usar marcadores, o provedor deve ser capaz de alterar as colunas retornadas dependendo se o consumidor solicita um indicador.
Para manipular o IColumnsInfo::GetColumnsInfo chamar, exclua PROVIDER_COLUMN_MAP, que define uma função GetColumnInfo, da CAgentMan usuário registrar no MyProviderRS.h e substitua-a definição para o seu próprio GetColumnInfo função:
////////////////////////////////////////////////////////////////////////
// MyProviderRS.H
class CAgentMan
{
public:
DWORD dwBookmark;
TCHAR szCommand[256];
TCHAR szText[256];
TCHAR szCommand2[256];
TCHAR szText2[256];
static ATLCOLUMNINFO* GetColumnInfo(void* pThis, ULONG* pcCols);
bool operator==(const CAgentMan& am)
{
return (lstrcmpi(szCommand, am.szCommand) == 0);
}
};
Em seguida, implemente o GetColumnInfo função em MyProviderRS.cpp, sistema autônomo mostra o código a seguir.
GetColumnInfo verifica primeiro se a propriedade de banco de dados OLE DB DBPROP_BOOKMARKS é definida.Para obter a propriedade, GetColumnInfo usa um ponteiro (pRowset) para o objeto de conjunto de linhas. The pThis ponteiro representa a classe que criou o conjunto de linhas, que é a classe onde o MAP de propriedade é armazenado. GetColumnInfo typecasts o pThis ponteiro para um RMyProviderRowset ponteiro.
Para verificar se o DBPROP_BOOKMARKS propriedade GetColumnInfo usa o IRowsetInfo interface, que pode ser obtido chamando QueryInterface sobre o pRowset interface. sistema autônomo alternativa, você pode usar um ATL CComQIPtr método em vez disso.
////////////////////////////////////////////////////////////////////
// MyProviderRS.cpp
ATLCOLUMNINFO* CAgentMan::GetColumnInfo(void* pThis, ULONG* pcCols)
{
static ATLCOLUMNINFO _rgColumns[5];
ULONG ulCols = 0;
// Check the property flag for bookmarks; if it is set, set the zero
// ordinal entry in the column map with the bookmark information.
CAgentRowset* pRowset = (CAgentRowset*) pThis;
CComQIPtr<IRowsetInfo, &IID_IRowsetInfo> spRowsetProps = pRowset;
CDBPropIDSet set(DBPROPSET_ROWSET);
set.AddPropertyID(DBPROP_BOOKMARKS);
DBPROPSET* pPropSet = NULL;
ULONG ulPropSet = 0;
HRESULT hr;
if (spRowsetProps)
hr = spRowsetProps->GetProperties(1, &set, &ulPropSet, &pPropSet);
if (pPropSet)
{
CComVariant var = pPropSet->rgProperties[0].vValue;
CoTaskMemFree(pPropSet->rgProperties);
CoTaskMemFree(pPropSet);
if (SUCCEEDED(hr) && (var.boolVal == VARIANT_TRUE))
{
ADD_COLUMN_ENTRY_EX(ulCols, OLESTR("Bookmark"), 0, sizeof(DWORD),
DBTYPE_BYTES, 0, 0, GUID_NULL, CAgentMan, dwBookmark,
DBCOLUMNFLAGS_ISBOOKMARK)
ulCols++;
}
}
// Next, set the other columns up.
ADD_COLUMN_ENTRY(ulCols, OLESTR("Command"), 1, 256, DBTYPE_STR, 0xFF, 0xFF,
GUID_NULL, CAgentMan, szCommand)
ulCols++;
ADD_COLUMN_ENTRY(ulCols, OLESTR("Text"), 2, 256, DBTYPE_STR, 0xFF, 0xFF,
GUID_NULL, CAgentMan, szText)
ulCols++;
ADD_COLUMN_ENTRY(ulCols, OLESTR("Command2"), 3, 256, DBTYPE_STR, 0xFF, 0xFF,
GUID_NULL, CAgentMan, szCommand2)
ulCols++;
ADD_COLUMN_ENTRY(ulCols, OLESTR("Text2"), 4, 256, DBTYPE_STR, 0xFF, 0xFF,
GUID_NULL, CAgentMan, szText2)
ulCols++;
if (pcCols != NULL)
*pcCols = ulCols;
return _rgColumns;
}
Este exemplo usa uma matriz estática para conter as informações de coluna.Se o consumidor não deseja que a coluna de indicador, uma entrada na matriz é não utilizada.Para lidar com as informações, você pode criar duas macros de array: ADD_COLUMN_ENTRY e ADD_COLUMN_ENTRY_EX.ADD_COLUMN_ENTRY_EX assume um parâmetro extra, flags, que é necessário se você designar uma coluna de indicador.
////////////////////////////////////////////////////////////////////////
// MyProviderRS.h
#define ADD_COLUMN_ENTRY(ulCols, name, ordinal, colSize, type, precision,
scale, guid, dataClass, member) \
_rgColumns[ulCols].pwszName = (LPOLESTR)name; \
_rgColumns[ulCols].pTypeInfo = (ITypeInfo*)NULL; \
_rgColumns[ulCols].iOrdinal = (ULONG)ordinal; \
_rgColumns[ulCols].dwFlags = 0; \
_rgColumns[ulCols].ulColumnSize = (ULONG)colSize; \
_rgColumns[ulCols].wType = (DBTYPE)type; \
_rgColumns[ulCols].bPrecision = (BYTE)precision; \
_rgColumns[ulCols].bScale = (BYTE)scale; \
_rgColumns[ulCols].cbOffset = offsetof(dataClass, member);
#define ADD_COLUMN_ENTRY_EX(ulCols, name, ordinal, colSize, type,
precision, scale, guid, dataClass, member, flags) \
_rgColumns[ulCols].pwszName = (LPOLESTR)name; \
_rgColumns[ulCols].pTypeInfo = (ITypeInfo*)NULL; \
_rgColumns[ulCols].iOrdinal = (ULONG)ordinal; \
_rgColumns[ulCols].dwFlags = flags; \
_rgColumns[ulCols].ulColumnSize = (ULONG)colSize; \
_rgColumns[ulCols].wType = (DBTYPE)type; \
_rgColumns[ulCols].bPrecision = (BYTE)precision; \
_rgColumns[ulCols].bScale = (BYTE)scale; \
_rgColumns[ulCols].cbOffset = offsetof(dataClass, member); \
memset(&(_rgColumns[ulCols].columnid), 0, sizeof(DBID)); \
_rgColumns[ulCols].columnid.uName.pwszName = (LPOLESTR)name;
No GetColumnInfo função, a macro de indicador é usada da seguinte forma:
ADD_COLUMN_ENTRY_EX(ulCols, OLESTR("Bookmark"), 0, sizeof(DWORD),
DBTYPE_BYTES, 0, 0, GUID_NULL, CAgentMan, dwBookmark,
DBCOLUMNFLAGS_ISBOOKMARK)
Agora você pode compilar e executar provedor avançado.Para testar o provedor, modifique o consumidor de teste conforme descrito em Implementação de um consumidor Simple.Execute o teste de consumidor com o provedor.Verifique se que o consumidor de teste recupera as seqüências de caracteres apropriadas do provedor de quando você clicar na Executar botão a Consumidor de teste caixa de diálogo.