Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1177 lines
24 KiB

/*++
Copyright (c) 1997 Microsoft Corporation
Module Name:
cwabobj.cpp
Abstract:
Interface to the windows address book.
Environment:
Fax send wizard
Revision History:
10/23/97 -GeorgeJe-
Created it.
mm/dd/yy -author-
description
--*/
#include <windows.h>
#include <prsht.h>
#include <tchar.h>
#include <wab.h>
#include "faxui.h"
#include "cwabobj.h"
static
LPWSTR
DupUnicodeString(
LPWSTR pStr
);
static
LPWSTR
DupStringAnsiToUnicode(
LPSTR pAnsiStr
);
static
LPSPropValue
FindProp(
LPSPropValue rgprop,
ULONG cprop,
ULONG ulPropTag
);
static
AddRecipient(
PRECIPIENT *ppNewRecip,
LPWSTR DisplayName,
LPWSTR FaxNumber
);
static SizedSPropTagArray(5, sPropTags) =
{
5,
{
PR_DISPLAY_NAME_A,
PR_PRIMARY_FAX_NUMBER_A,
PR_HOME_FAX_NUMBER_A,
PR_BUSINESS_FAX_NUMBER_A,
PR_OBJECT_TYPE
}
};
CWabObj::CWabObj(
HINSTANCE hInstance
)
/*++
Routine Description:
Constructor for CWabObj class
Arguments:
hInstance - Instance handle
Return Value:
NONE
--*/
{
m_Initialized = FALSE;
m_lpAdrList = NULL;
m_hInstance = hInstance;
}
BOOL
CWabObj::Initialize(
VOID
)
/*++
Routine Description:
intialization function for CWabObj class
Arguments:
NONE
Return Value:
TRUE if the object is initialized successfully, else FALSE
--*/
{
TCHAR szDllPath[MAX_PATH];
HKEY hKey = NULL;
LONG rVal;
DWORD dwType;
DWORD cbData = MAX_PATH * sizeof(TCHAR);
HRESULT hr;
PCTSTR szDefaultPath = TEXT("%CommonProgramFiles%\\System\\wab32.dll");
m_Initialized = TRUE;
//
// get the path to wab32.dll
//
rVal = RegOpenKeyEx(
HKEY_LOCAL_MACHINE,
REGVAL_WABPATH,
0,
KEY_READ,
&hKey
);
if (rVal == ERROR_SUCCESS) {
rVal = RegQueryValueEx(
hKey,
TEXT(""),
NULL,
&dwType,
(LPBYTE) szDllPath,
&cbData
);
RegCloseKey( hKey );
}
if (rVal != ERROR_SUCCESS) {
ExpandEnvironmentStrings(szDefaultPath,szDllPath,sizeof(szDllPath)/sizeof(TCHAR));
}
m_hWab = LoadLibrary( szDllPath );
if (m_hWab != NULL) {
m_lpWabOpen = (LPWABOPEN) GetProcAddress( m_hWab , "WABOpen" );
} else {
m_lpWabOpen = (LPWABOPEN) NULL;
}
if (m_lpWabOpen == NULL) {
hr = E_FAIL;
goto exit;
}
//
// open the wab
//
hr = m_lpWabOpen( &m_lpAdrBook, &m_lpWABObject, 0, 0 );
exit:
if (HR_FAILED(hr)) {
m_lpAdrBook = NULL;
m_lpWABObject = NULL;
m_Initialized = FALSE;
if (m_hWab != NULL) {
FreeLibrary( m_hWab );
}
m_hWab = NULL;
}
return(m_Initialized);
}
CWabObj::~CWabObj()
/*++
Routine Description:
Destructor for CWabObj class
Arguments:
NONE
Return Value:
NONE
--*/
{
if (m_lpAdrBook) {
m_lpAdrBook->Release();
}
if (m_lpWABObject) {
m_lpWABObject->Release();
}
if ( m_hWab ) {
FreeLibrary( m_hWab );
}
}
BOOL
CWabObj::Address(
HWND hWnd,
PRECIPIENT pRecipients,
PRECIPIENT * ppNewRecip
)
/*++
Routine Description:
Bring up the address book UI. Prepopulate the to box with the entries in
pRecipient. Return the modified entries in ppNewRecip.
Arguments:
hWnd - window handle to parent window
pRecipients - list of recipients to look up
ppNewRecipients - list of new/modified recipients
Return Value:
TRUE if all recipients had a fax number.
FALSE if one or more of them didn't.
--*/
{
ADRPARM AdrParms = { 0 };
LPADRLIST tmp;
HRESULT hr;
DWORD i;
DWORD nRecips;
PRECIPIENT tmpRecipient;
ULONG DestComps[3] = { MAPI_TO, MAPI_CC, MAPI_BCC };
DWORD cDropped;
nRecips = 0;
tmpRecipient = pRecipients;
m_hWnd = hWnd;
m_PickNumber = 0;
//
// count recipients and set up initial address list
//
while (tmpRecipient) {
nRecips++;
tmpRecipient = (PRECIPIENT) tmpRecipient->pNext;
}
if (nRecips > 0) {
hr = m_lpWABObject->AllocateBuffer( CbNewADRLIST( nRecips ), (LPVOID *) &m_lpAdrList );
m_lpAdrList->cEntries = nRecips;
} else {
m_lpAdrList = NULL;
}
for (i = 0, tmpRecipient = pRecipients; i < nRecips; i++, tmpRecipient = (PRECIPIENT) tmpRecipient->pNext) {
LPADRENTRY lpAdrEntry = &m_lpAdrList->aEntries[i];
lpAdrEntry->cValues = 3;
hr = m_lpWABObject->AllocateBuffer( sizeof( SPropValue ) * 3, (LPVOID *) &lpAdrEntry->rgPropVals );
ZeroMemory( lpAdrEntry->rgPropVals, sizeof( SPropValue ) * 3 );
lpAdrEntry->rgPropVals[0].ulPropTag = PR_DISPLAY_NAME_A;
lpAdrEntry->rgPropVals[0].Value.lpszA = DupStringUnicodeToAnsi( lpAdrEntry->rgPropVals, tmpRecipient->pName );
lpAdrEntry->rgPropVals[1].ulPropTag = PR_RECIPIENT_TYPE;
lpAdrEntry->rgPropVals[1].Value.l = MAPI_TO;
lpAdrEntry->rgPropVals[2].ulPropTag = PR_PRIMARY_FAX_NUMBER_A;
lpAdrEntry->rgPropVals[2].Value.lpszA = DupStringUnicodeToAnsi( lpAdrEntry->rgPropVals, tmpRecipient->pAddress );
}
tmp = m_lpAdrList;
AdrParms.cDestFields = 1;
AdrParms.ulFlags = DIALOG_MODAL;
AdrParms.nDestFieldFocus = 0;
AdrParms.lpulDestComps = DestComps;
AdrParms.lpszCaption = TEXT( "" );
//
// Bring up the address book UI
//
hr = m_lpAdrBook->Address(
(ULONG *) &hWnd,
&AdrParms,
&m_lpAdrList
);
if (FAILED (hr) || !m_lpAdrList || m_lpAdrList->cEntries == 0) {
//
// in this case the user pressed cancel, so we skip resolving any of our addresses that aren't listed in the
// WAB
//
cDropped = 0;
goto skipresolve;
}
//
// Resolve names
//
hr = m_lpAdrBook->ResolveName ((ULONG_PTR)hWnd, 0, NULL, m_lpAdrList);
skipresolve:
tmp = m_lpAdrList;
if (m_lpAdrList) {
for (i = cDropped = 0; i < m_lpAdrList->cEntries; i++) {
LPADRENTRY lpAdrEntry = &m_lpAdrList->aEntries[i];
if (!InterpretAddress( lpAdrEntry->rgPropVals, lpAdrEntry->cValues, ppNewRecip )){
cDropped++;
}
}
//
// Clean up
//
for (ULONG iEntry = 0; iEntry < m_lpAdrList->cEntries; ++iEntry)
{
if(m_lpAdrList->aEntries[iEntry].rgPropVals)
m_lpWABObject->FreeBuffer(m_lpAdrList->aEntries[iEntry].rgPropVals);
}
m_lpWABObject->FreeBuffer(m_lpAdrList);
m_lpAdrList = NULL;
}
m_hWnd = NULL;
return cDropped == 0;
}
BOOL
CWabObj::InterpretAddress(
LPSPropValue SPropVal,
ULONG cValues,
PRECIPIENT *ppNewRecip
)
/*++
Routine Description:
Interpret the address book entry represented by SPropVal.
Arguments:
SPropVal - Property values for address book entry.
cValues - number of property values
ppNewRecip - new recipient list
Return Value:
TRUE if all of the entries have a fax number.
FALSE otherwise.
--*/
{
LPSPropValue lpSPropVal;
LPWSTR FaxNumber, DisplayName;
BOOL rVal = FALSE;
//
// get the object type
//
lpSPropVal = FindProp( SPropVal, cValues, PR_OBJECT_TYPE );
if (lpSPropVal) {
//
// If the object is a mail user, get the fax numbers and add the recipient
// to the list. If the object is a distribtion list, process it.
//
switch (lpSPropVal->Value.l) {
case MAPI_MAILUSER:
if(GetRecipientInfo( SPropVal, cValues, &FaxNumber, &DisplayName )) {
AddRecipient( ppNewRecip, DisplayName, FaxNumber );
rVal = TRUE;
}
break;
case MAPI_DISTLIST:
rVal = InterpretDistList( SPropVal, cValues, ppNewRecip );
}
return rVal;
} else {
//
// If there is no object type then this is valid entry that we queried on that went unresolved.
// We know that there is a fax number so add it.
//
if(GetRecipientInfo( SPropVal, cValues, &FaxNumber, &DisplayName )) {
AddRecipient( ppNewRecip, DisplayName, FaxNumber );
rVal = TRUE;
}
}
return rVal;
}
BOOL
CWabObj::InterpretDistList(
LPSPropValue SPropVal,
ULONG cValues,
PRECIPIENT * ppNewRecip
)
/*++
Routine Description:
Process a distribution list.
Arguments:
SPropVal - Property values for distribution list.
cValues - Number of properties.
ppNewRecip - New recipient list.
Return Value:
TRUE if all of the entries have a fax number.
FALSE otherwise.
--*/
#define EXIT_IF_FAILED(hr) { if (FAILED(hr)) goto ExitDistList; }
{
LPSPropValue lpPropVals;
LPSRowSet pRows = NULL;
LPDISTLIST lpMailDistList = NULL;
LPMAPITABLE pMapiTable = NULL;
ULONG ulObjType, cRows;
HRESULT hr;
BOOL rVal = FALSE;
lpPropVals = FindProp( SPropVal, cValues, PR_ENTRYID );
if (lpPropVals) {
LPENTRYID lpEntryId = (LPENTRYID) lpPropVals->Value.bin.lpb;
DWORD cbEntryId = lpPropVals->Value.bin.cb;
//
// Open the recipient entry
//
hr = m_lpAdrBook->OpenEntry(
cbEntryId,
lpEntryId,
(LPCIID) NULL,
0,
&ulObjType,
(LPUNKNOWN *) &lpMailDistList
);
EXIT_IF_FAILED( hr );
//
// Get the contents table of the address entry
//
hr = lpMailDistList->GetContentsTable(
MAPI_DEFERRED_ERRORS,
&pMapiTable
);
EXIT_IF_FAILED(hr);
//
// Limit the query to only the properties we're interested in
//
hr = pMapiTable->SetColumns((LPSPropTagArray) &sPropTags, 0);
EXIT_IF_FAILED(hr);
//
// Get the total number of rows
//
hr = pMapiTable->GetRowCount(0, &cRows);
EXIT_IF_FAILED(hr);
//
// Get the individual entries of the distribution list
//
hr = pMapiTable->SeekRow(BOOKMARK_BEGINNING, 0, NULL);
EXIT_IF_FAILED(hr);
hr = pMapiTable->QueryRows(cRows, 0, &pRows);
EXIT_IF_FAILED(hr);
hr = S_OK;
if (pRows && pRows->cRows) {
//
// Handle each entry of the distribution list in turn:
// for simple entries, call InterpretAddress
// for embedded distribution list, call this function recursively
//
for (cRows = 0; cRows < pRows->cRows; cRows++) {
LPSPropValue lpProps = pRows->aRow[cRows].lpProps;
ULONG cRowValues = pRows->aRow[cRows].cValues;
lpPropVals = FindProp( lpProps, cRowValues, PR_OBJECT_TYPE );
if (lpPropVals) {
switch (lpPropVals->Value.l) {
case MAPI_MAILUSER:
rVal = InterpretAddress( lpProps, cRowValues, ppNewRecip );
break;
case MAPI_DISTLIST:
rVal = InterpretDistList( lpProps, cRowValues, ppNewRecip );
}
}
}
}
}
ExitDistList:
//
// Perform necessary clean up before returning to caller
//
if (pRows) {
for (cRows = 0; cRows < pRows->cRows; cRows++) {
m_lpWABObject->FreeBuffer(pRows->aRow[cRows].lpProps);
}
m_lpWABObject->FreeBuffer(pRows);
}
if (pMapiTable)
pMapiTable->Release();
if (lpMailDistList)
lpMailDistList->Release();
return rVal;
}
INT_PTR
CALLBACK
ChooseFaxNumberDlgProc(
HWND hDlg,
UINT uMsg,
WPARAM wParam,
LPARAM lParam
)
/*++
Routine Description:
Dialog proc for choose fax number dialog.
Arguments:
lParam - pointer to PickFax structure.
Return Value:
Control id of selection.
--*/
{
PPICKFAX pPickFax = (PPICKFAX) lParam;
TCHAR buffer[MAX_TITLE_LEN];
switch (uMsg) {
case WM_INITDIALOG:
SetDlgItemText(hDlg, IDC_DISPLAY_NAME, pPickFax->DisplayName);
buffer[0] = 0;
GetDlgItemText(hDlg, IDC_BUSINESS_FAX, buffer, MAX_TITLE_LEN);
_tcscat(buffer, pPickFax->BusinessFax);
SetDlgItemText(hDlg, IDC_BUSINESS_FAX, buffer);
buffer[0] = 0;
GetDlgItemText(hDlg, IDC_HOME_FAX, buffer, MAX_TITLE_LEN);
_tcscat(buffer, pPickFax->HomeFax);
SetDlgItemText(hDlg, IDC_HOME_FAX, buffer);
CheckDlgButton(hDlg, IDC_BUSINESS_FAX, BST_CHECKED);
return TRUE;
case WM_COMMAND:
switch(LOWORD( wParam )){
case IDOK:
if (IsDlgButtonChecked(hDlg, IDC_BUSINESS_FAX) == BST_CHECKED) {
if (IsDlgButtonChecked(hDlg, IDC_ALWAYS_OPTION) == BST_CHECKED) {
EndDialog(hDlg, IDC_ALLBUS);
}
else {
EndDialog(hDlg, IDC_BUSINESS_FAX);
}
}
else if (IsDlgButtonChecked(hDlg, IDC_HOME_FAX) == BST_CHECKED) {
if (IsDlgButtonChecked(hDlg, IDC_ALWAYS_OPTION) == BST_CHECKED) {
EndDialog(hDlg, IDC_ALLHOME);
}
else {
EndDialog(hDlg, IDC_HOME_FAX);
}
}
break;;
}
break;
default:
return FALSE;
}
return FALSE;
}
#define StrPropOk( strprop ) ((strprop) && (strprop)->Value.lpszA && *(strprop)->Value.lpszA)
BOOL
CWabObj::GetRecipientInfo(
LPSPropValue SPropVals,
ULONG cValues,
LPWSTR * FaxNumber,
LPWSTR * DisplayName
)
/*++
Routine Description:
Get the fax number and display name properties.
Arguments:
SPropVal - Property values for distribution list.
cValues - Number of properties.
FaxNumber - pointer to pointer to string to hold the fax number.
DisplayName - pointer to pointer to string to hold the display name.
Return Value:
TRUE if there is a fax number and display name.
FALSE otherwise.
--*/
{
LPSPropValue lpPropVals;
LPSPropValue lpPropArray;
BOOL Result = FALSE;
PICKFAX PickFax = { 0 };
*FaxNumber = *DisplayName = NULL;
//
// Get the entryid and open the entry.
//
lpPropVals = FindProp( SPropVals, cValues, PR_ENTRYID );
if (lpPropVals) {
ULONG lpulObjType;
LPMAILUSER lpMailUser = NULL;
LPENTRYID lpEntryId = (LPENTRYID) lpPropVals->Value.bin.lpb;
DWORD cbEntryId = lpPropVals->Value.bin.cb;
HRESULT hr;
ULONG countValues;
hr = m_lpAdrBook->OpenEntry(
cbEntryId,
lpEntryId,
(LPCIID) NULL,
0,
&lpulObjType,
(LPUNKNOWN *) &lpMailUser
);
if (HR_SUCCEEDED(hr)) {
//
// Get the properties.
//
hr = ((IMailUser *) lpMailUser)->GetProps( (LPSPropTagArray) &sPropTags, 0, &countValues, &lpPropArray );
if (HR_SUCCEEDED(hr)) {
lpPropVals = FindProp( lpPropArray, countValues, PR_BUSINESS_FAX_NUMBER_A );
if (StrPropOk( lpPropVals )) {
PickFax.BusinessFax = DupStringAnsiToUnicode( lpPropVals->Value.lpszA );
}
lpPropVals = FindProp( lpPropArray, countValues, PR_HOME_FAX_NUMBER_A );
if (StrPropOk( lpPropVals )) {
PickFax.HomeFax = DupStringAnsiToUnicode( lpPropVals->Value.lpszA );
}
lpPropVals = FindProp( lpPropArray, countValues, PR_DISPLAY_NAME_A );
if (StrPropOk( lpPropVals )) {
*DisplayName = PickFax.DisplayName = DupStringAnsiToUnicode( lpPropVals->Value.lpszA );
}
//
// If there are two fax numbers, ask the user to pick one.
//
if (PickFax.BusinessFax && PickFax.HomeFax) {
int dlgResult;
if (m_PickNumber != 0) {
dlgResult = m_PickNumber;
} else {
dlgResult = (int)DialogBoxParam(
(HINSTANCE) m_hInstance,
MAKEINTRESOURCE( IDD_CHOOSE_FAXNUMBER ),
m_hWnd,
ChooseFaxNumberDlgProc,
(LPARAM) &PickFax
);
}
switch( dlgResult ) {
case IDC_ALLBUS:
m_PickNumber = IDC_BUSINESS_FAX;
// fall through
case IDC_BUSINESS_FAX:
MemFree( PickFax.HomeFax );
*FaxNumber = PickFax.BusinessFax;
break;
case IDC_ALLHOME:
m_PickNumber = IDC_HOME_FAX;
// fall through
case IDC_HOME_FAX:
MemFree( PickFax.BusinessFax );
*FaxNumber = PickFax.HomeFax;
break;
}
} else if (PickFax.BusinessFax) {
*FaxNumber = PickFax.BusinessFax;
} else if (PickFax.HomeFax) {
*FaxNumber = PickFax.HomeFax;
}
}
m_lpWABObject->FreeBuffer( lpPropArray );
}
if (lpMailUser) {
lpMailUser->Release();
}
} else {
// If there is no entryid, then this is a valid entry that we queried on that went unresolved
// add if anyway. In this case we know that PR_PRIMARY_FAX_NUMBER_A and PR_DISPLAY_NAME_A will be
// present.
lpPropVals = FindProp( SPropVals, cValues, PR_PRIMARY_FAX_NUMBER_A );
if (lpPropVals) {
*FaxNumber = DupStringAnsiToUnicode( lpPropVals->Value.lpszA );
}
lpPropVals = FindProp( SPropVals, cValues, PR_DISPLAY_NAME_A );
if (lpPropVals) {
*DisplayName = DupStringAnsiToUnicode( lpPropVals->Value.lpszA );
}
}
if (FaxNumber && DisplayName) {
return (*FaxNumber != 0 && *DisplayName != 0);
} else {
return FALSE;
}
}
LPSPropValue
FindProp(
LPSPropValue rgprop,
ULONG cprop,
ULONG ulPropTag
)
/*++
Routine Description:
Searches for a given property tag in a propset. If the given
property tag has type PT_UNSPECIFIED, matches only on the
property ID; otherwise, matches on the entire tag.
Arguments:
rgprop - Property values.
cprop - Number of properties.
ulPropTag - Property to search for.
Return Value:
Pointer to property desired property value or NULL.
--*/
{
BOOL f = PROP_TYPE(ulPropTag) == PT_UNSPECIFIED;
LPSPropValue pprop = rgprop;
if (!cprop || !rgprop)
return NULL;
while (cprop--)
{
if (pprop->ulPropTag == ulPropTag ||
(f && PROP_ID(pprop->ulPropTag) == PROP_ID(ulPropTag)))
return pprop;
++pprop;
}
return NULL;
}
LPSTR
CWabObj::DupStringUnicodeToAnsi(
LPVOID lpObject,
LPWSTR pUnicodeStr
)
/*++
Routine Description:
Convert a Unicode string to a multi-byte string
Arguments:
pUnicodeStr - Pointer to the Unicode string to be duplicated
Return Value:
Pointer to the duplicated multi-byte string
NOTE:
This is only need because the WAB is not Unicode enabled on NT.
This uses the WAB memory allocator so it must be freed with FreeBuffer.
--*/
{
INT nChar;
LPSTR pAnsiStr;
//
// Figure out how much memory to allocate for the multi-byte string
//
if (! (nChar = WideCharToMultiByte(CP_ACP, 0, pUnicodeStr, -1, NULL, 0, NULL, NULL)) ||
! HR_SUCCEEDED( m_lpWABObject->AllocateMore( nChar, lpObject, (LPVOID *) &pAnsiStr )))
{
return NULL;
}
//
// Convert Unicode string to multi-byte string
//
WideCharToMultiByte(CP_ACP, 0, pUnicodeStr, -1, pAnsiStr, nChar, NULL, NULL);
return pAnsiStr;
}
LPWSTR
DupStringAnsiToUnicode(
LPSTR pAnsiStr
)
/*++
Routine Description:
Convert a multi-byte string to a Unicode string
Arguments:
pAnsiStr - Pointer to the Ansi string to be duplicated
Return Value:
Pointer to the duplicated Unicode string
NOTE:
This is only need because MAPI is not Unicode enabled on NT.
This routine uses MemAlloc to allocate memory so the caller needs
to use MemFree.
--*/
{
INT nChar;
LPWSTR pUnicodeStr;
//
// Figure out how much memory to allocate for the Unicode string
//
if (! (nChar = MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, pAnsiStr, -1, NULL, 0)) ||
! ( pUnicodeStr = (LPWSTR) MemAlloc( nChar * sizeof(WCHAR) ) ))
{
return NULL;
}
//
// Convert Unicode string to multi-byte string
//
MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, pAnsiStr, -1, pUnicodeStr, nChar);
return pUnicodeStr;
}
LPWSTR
DupUnicodeString(
LPWSTR pStr
)
/*++
Routine Description:
Duplicate a Unicode string.
Arguments:
pStr - pointer to string to duplicate.
Return Value:
pointer to duplicated string.
--*/
{
LPWSTR NewStr;
NewStr = (LPWSTR) MemAlloc( (wcslen( pStr ) + 1) * sizeof (WCHAR));
wcscpy( NewStr, pStr );
return NewStr;
}
AddRecipient(
PRECIPIENT *ppNewRecip,
LPWSTR DisplayName,
LPWSTR FaxNumber
)
/*++
Routine Description:
Add a recipient to the recipient list.
Arguments:
ppNewRecip - pointer to pointer to list to add item to.
DisplayName - recipient name.
FaxNumber - recipient fax number.
Return Value:
NA
--*/
{
PRECIPIENT NewRecip;
NewRecip = (PRECIPIENT) MemAllocZ( sizeof( RECIPIENT ) );
if (NewRecip) {
NewRecip->pName = DisplayName;
NewRecip->pAddress = FaxNumber;
NewRecip->pNext = (LPVOID) *ppNewRecip;
*ppNewRecip = NewRecip;
}
return 0;
}
extern "C"
BOOL
CallWabAddress(
HWND hDlg,
PUSERMEM pUserMem,
PRECIPIENT * ppNewRecipient
)
/*++
Routine Description:
C wrapper for CWabObj->Address
Arguments:
hDlg - parent window handle.
pUserMem - pointer to USERMEM structure
ppNewRecipient - list to add new recipients to.
Return Value:
TRUE if all of the entries have a fax number.
FALSE otherwise.
--*/
{
LPWABOBJ lpCWabObj = (LPWABOBJ) pUserMem->lpWabInit;
return lpCWabObj->Address(
hDlg,
pUserMem->pRecipients,
ppNewRecipient
);
}
extern "C"
LPVOID
InitializeWAB(
HINSTANCE hInstance
)
/*++
Routine Description:
Initialize the WAB.
Arguments:
hInstance - instance handle.
Return Value:
NONE
--*/
{
LPWABOBJ lpWabObj = new CWabObj( hInstance );
if (lpWabObj) {
if (!lpWabObj->Initialize()) {
delete (lpWabObj);
lpWabObj = NULL;
}
}
return (LPVOID) lpWabObj;
}
extern "C"
VOID
UnInitializeWAB(
LPVOID lpVoid
)
/*++
Routine Description:
UnInitialize the WAB.
Arguments:
NONE
Return Value:
NONE
--*/
{
LPWABOBJ lpWabObj = (LPWABOBJ) lpVoid;
delete lpWabObj;
}