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.
 
 
 
 
 
 

2455 lines
71 KiB

///////////////////////////////////////////////////////////////////////////////
/* File: mapisend.cpp
Description: Implements the most basic MAPI email client to send a message
to one or more recipients. All operations are done without UI.
classes: CMapiSession
CMapiRecipients
CMapiMessage
CMapiMessageBody
MAPI
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
06/22/97 Added class MAPI. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
#include "precomp.hxx"
#pragma hdrstop
#include "mapisend.h"
//
// Global MAPI object to provide dynamic linking to MAPI32.DLL.
// All MAPI32.DLL functions are called through this object.
//
MAPI MAPI32;
//
// How many entries we grow the recipients list by when it is enlarged.
//
#ifdef DEBUG
//
// For debugging and development, stress the list-growth code by making
// it extend the list each time a recipient is added.
//
UINT CMapiRecipients::m_cGrowIncr = 1;
#else // !DEBUG
//
// For production builds, fix the growth increment at 3. 3 is arbitrary
// but is probably a good conservative guess. Want to avoid
// list growth in the typical scenario.
//
// 1 for the quota user.
// 1 for the volume administrator.
// 1 for the user's manager (? probably not)
//
// Note that the list growth code always runs at least once because
// we use it to create the initial list.
//
UINT CMapiRecipients::m_cGrowIncr = 3;
#endif // DEBUG
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::CMapiRecipients
Description: Constructor.
Arguments: None.
Returns: Nothing.
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
CMapiRecipients::CMapiRecipients(
BOOL bUnicode
) : m_pal(NULL),
m_bUnicode(bUnicode),
m_cMaxEntries(0)
{
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::~CMapiRecipients
Description: Destructor.
Frees the MAPI address list.
Arguments: None.
Returns: Nothing.
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
CMapiRecipients::~CMapiRecipients(
VOID
)
{
if (NULL != m_pal)
{
MAPI32.FreePadrlist(m_pal);
}
}
CMapiRecipients::CMapiRecipients(
const CMapiRecipients& rhs
) : m_pal(NULL),
m_bUnicode(FALSE),
m_cMaxEntries(0)
{
#ifdef UNICODE
m_bUnicode = TRUE;
#endif
*this = rhs;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::operator = (const CMapiRecipients& rhs)
Description: Assignment copy.
Arguments:
rhs - Reference to source recipient list.
Returns: Reference to destination recipient list.
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
CMapiRecipients&
CMapiRecipients::operator = (
const CMapiRecipients& rhs
)
{
if (this != &rhs)
{
BOOL bConvertStrings = m_bUnicode != rhs.m_bUnicode;
//
// Delete the current address list.
//
if (NULL != m_pal)
{
MAPI32.FreePadrlist(m_pal);
m_pal = NULL;
}
//
// Copy the Max entry count.
//
// NOTE: We DO NOT copy the m_bUnicode attribute.
// This attribute stays with the object for life.
//
m_cMaxEntries = rhs.m_cMaxEntries;
if (NULL != rhs.m_pal)
{
HRESULT hr = E_FAIL;
UINT cb;
cb = sizeof(ADRLIST) + ((m_cMaxEntries - 1) * sizeof(ADRENTRY));
hr = MAPI32.AllocateBuffer(cb, (LPVOID *)&m_pal);
if (SUCCEEDED(hr))
{
ZeroMemory(m_pal, cb); // Note: m_pal->cEntries is init'd to 0.
if (NULL != m_pal)
{
for (UINT i = 0; i < rhs.m_pal->cEntries && SUCCEEDED(hr); i++)
{
hr = CopyAdrListEntry(m_pal->aEntries[i],
rhs.m_pal->aEntries[i],
bConvertStrings);
if (SUCCEEDED(hr))
{
m_pal->cEntries++;
}
}
}
}
if (FAILED(hr))
{
//
// Something went wrong. Leave the object in an empty state.
//
if (NULL != m_pal)
MAPI32.FreePadrlist(m_pal);
m_pal = NULL;
m_cMaxEntries = 0;
}
}
}
return *this;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::Count
Description: Returns the count of valid entries in the address list.
Arguments: None.
Returns: Count of entries.
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
INT
CMapiRecipients::Count(
VOID
) const
{
return m_pal ? m_pal->cEntries : 0;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::AddRecipient
Description: Adds a new recipient/recipient-type pair to the address list.
Arguments:
pszEmailName - Name of recipient typically used as an email destination.
dwType - Recipient type. Can be one of the following:
MAPI_ORIG
MAPI_TO
MAPI_CC
MAPI_BCC
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiRecipients::AddRecipient(
LPCTSTR pszEmailName,
DWORD dwType
)
{
HRESULT hr = NO_ERROR;
if (NULL == m_pal || m_pal->cEntries == m_cMaxEntries)
{
//
// Either we're just starting up (no list created yet),
// or the current list is full. Grow the list.
//
hr = Grow(m_cGrowIncr);
}
if (SUCCEEDED(hr) && NULL != m_pal)
{
UINT cb;
INT i = m_pal->cEntries++;
m_pal->aEntries[i].ulReserved1 = 0;
m_pal->aEntries[i].cValues = 2;
//
// Allocate the SPropValue buffer for this new entry.
// Our entries have 2 values (name and type).
// Caller must call IAddrBook::ResolveName() to get the remaining
// recipient information.
//
cb = sizeof(SPropValue) * m_pal->aEntries[i].cValues;
hr = MAPI32.AllocateBuffer(cb, (LPVOID *)&m_pal->aEntries[i].rgPropVals);
if (SUCCEEDED(hr))
{
ZeroMemory(m_pal->aEntries[i].rgPropVals, cb);
//
// Allocate the buffer for the recipient email name string.
//
hr = MAPI32.AllocateMore((lstrlen(pszEmailName)+1) * sizeof(TCHAR),
(LPVOID)m_pal->aEntries[i].rgPropVals,
(LPVOID *)&m_pal->aEntries[i].rgPropVals[0].Value.LPSZ);
if (SUCCEEDED(hr))
{
//
// Store the recipient email name string.
//
m_pal->aEntries[i].rgPropVals[0].ulPropTag = PR_DISPLAY_NAME;
lstrcpy(m_pal->aEntries[i].rgPropVals[0].Value.LPSZ, pszEmailName);
//
// Store the recipient type (i.e. MAPI_TO, MAPI_CC etc).
//
m_pal->aEntries[i].rgPropVals[1].ulPropTag = PR_RECIPIENT_TYPE;
m_pal->aEntries[i].rgPropVals[1].Value.l = dwType;
}
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::Grow
Description: Increases the size of the address list, preserving any
existing list entries.
Arguments:
cGrowIncr - Number of entries to grow the list by.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiRecipients::Grow(
UINT cGrowIncr
)
{
HRESULT hr = E_FAIL;
LPADRLIST m_palNew = NULL;
UINT cb;
//
// Allocate the new buffer m_cGrowIncr entries larger than the
// current buffer. The (-1) is because the declaration of ADRLIST already
// includes one entry.
//
cb = sizeof(ADRLIST) + ((m_cMaxEntries + cGrowIncr - 1) * sizeof(ADRENTRY));
hr = MAPI32.AllocateBuffer(cb, (LPVOID *)&m_palNew);
if (SUCCEEDED(hr))
{
ZeroMemory(m_palNew, cb); // Note: m_palNew->cEntries is init'd to 0.
if (NULL != m_pal)
{
//
// We have an existing address list.
// Copy it to the new list buffer.
// If we fail the copy of an entry, we abort the loop and m_palNew->cEntries
// accurately reflects how many valid entries we have.
//
for (UINT i = 0; i < m_pal->cEntries && SUCCEEDED(hr); i++)
{
hr = CopyAdrListEntry(m_palNew->aEntries[i], m_pal->aEntries[i], FALSE);
if (SUCCEEDED(hr))
{
m_palNew->cEntries++;
}
}
}
}
if (SUCCEEDED(hr))
{
//
// Delete the original list (if it exists) and store the
// address of the new list in m_pal.
//
LPADRLIST palOrig = m_pal;
m_pal = m_palNew;
if (NULL != palOrig)
MAPI32.FreePadrlist(palOrig);
m_cMaxEntries += cGrowIncr;
}
else
{
//
// Something went wrong. Just delete the new list and keep the old one
// the way it was.
//
if (NULL != m_palNew)
MAPI32.FreePadrlist(m_palNew);
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::CopyAdrListEntry
Description: Copies a single list entry from one entry structure to another.
All entry-type-specific issues are addressed.
Arguments:
Dest - Reference to destination entry structure.
Src - Reference to source entry structure.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiRecipients::CopyAdrListEntry(
ADRENTRY& Dest,
ADRENTRY& Src,
BOOL bConvertStrings
)
{
HRESULT hr;
UINT cb = 0;
//
// Allocate buffer for the new entry and it's property values.
//
cb = sizeof(SPropValue) * Src.cValues;
hr = MAPI32.AllocateBuffer(cb, (LPVOID *)&Dest.rgPropVals);
if (SUCCEEDED(hr))
{
ZeroMemory(Dest.rgPropVals, cb);
Dest.cValues = 0;
//
// Copy each of the values.
// If we fail the copy of a value, we abort the loop and Dest.cValues
// accurately reflects how many valid prop values we have.
//
for (UINT i = 0; i < Src.cValues && SUCCEEDED(hr); i++)
{
hr = CopySPropVal((LPVOID)Dest.rgPropVals, // Base for new allocations
Dest.rgPropVals[i], // Destination SPropVal
Src.rgPropVals[i], // Source SPropVal
bConvertStrings);
if (SUCCEEDED(hr))
{
Dest.cValues++;
}
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::CopySPropVal
Description: Copies a single property value from one SPropValue object
to another. Used by CopyAdrListEntry to copy the individual properties
belonging to an entry.
Arguments:
pvBaseAlloc - Pointer to use as the "lpObject" argument in a call
to MAPIAllocateMore() if memory must be reallocated during
the copy process.
Dest - Reference to destination property value structure.
Src - Reference to source property value structure.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiRecipients::CopySPropVal(
LPVOID pvBaseAlloc,
SPropValue& Dest,
SPropValue& Src,
BOOL bConvertStrings
)
{
HRESULT hr = NO_ERROR;
BOOL bCopyTag = TRUE;
//
// Copy method varies depending on property type.
//
switch(PROP_TYPE(Src.ulPropTag))
{
case PT_I2:
Dest.Value.i = Src.Value.i;
break;
case PT_LONG:
Dest.Value.l = Src.Value.l;
break;
case PT_R4:
Dest.Value.flt = Src.Value.flt;
break;
case PT_DOUBLE:
Dest.Value.dbl = Src.Value.dbl;
break;
case PT_BOOLEAN:
Dest.Value.b = Src.Value.b;
break;
case PT_CURRENCY:
Dest.Value.cur = Src.Value.cur;
break;
case PT_APPTIME:
Dest.Value.at = Src.Value.at;
break;
case PT_SYSTIME:
Dest.Value.ft = Src.Value.ft;
break;
case PT_I8:
Dest.Value.li = Src.Value.li;
break;
case PT_ERROR:
Dest.Value.err = Src.Value.err;
break;
case PT_NULL:
case PT_OBJECT:
Dest.Value.x = Src.Value.x;
break;
case PT_STRING8:
if (bConvertStrings && m_bUnicode)
{
//
// The recipients list is unicode, the source is ANSI
// and we're supposed to convert strings. That means
// we need to convert from Ansi to Unicode.
//
hr = CopySPropValString(pvBaseAlloc,
&Dest.Value.lpszW,
Src.Value.lpszA);
Dest.ulPropTag = PROP_TAG(PT_UNICODE, PROP_ID(Src.ulPropTag));
bCopyTag = FALSE;
}
else
{
//
// No conversion required. Just a straight copy.
//
hr = CopyVarLengthSPropVal(pvBaseAlloc,
(LPVOID *)&Dest.Value.lpszA,
Src.Value.lpszA,
(lstrlenA(Src.Value.lpszA)+1) * sizeof(char));
}
break;
case PT_UNICODE:
if (bConvertStrings && !m_bUnicode)
{
//
// The recipients list is Ansi, the source is Unicode
// and we're supposed to convert strings. That means
// we need to convert from Unicode to Ansi.
//
hr = CopySPropValString(pvBaseAlloc,
&Dest.Value.lpszA,
Src.Value.lpszW);
Dest.ulPropTag = PROP_TAG(PT_STRING8, PROP_ID(Src.ulPropTag));
bCopyTag = FALSE;
}
else
{
//
// No conversion required. Just a straight copy.
//
hr = CopyVarLengthSPropVal(pvBaseAlloc,
(LPVOID *)&Dest.Value.lpszW,
Src.Value.lpszW,
(lstrlenW(Src.Value.lpszW)+1) * sizeof(WCHAR));
}
break;
case PT_BINARY:
hr = CopyVarLengthSPropVal(pvBaseAlloc,
(LPVOID *)&Dest.Value.bin.lpb,
Src.Value.bin.lpb,
Src.Value.bin.cb);
break;
case PT_CLSID:
hr = CopyVarLengthSPropVal(pvBaseAlloc,
(LPVOID *)&Dest.Value.lpguid,
Src.Value.lpguid,
sizeof(*(Src.Value.lpguid)));
break;
default:
//
// FEATURE: Assert here so we know if we should be handling
// some other property type.
//
hr = E_FAIL;
break;
}
if (SUCCEEDED(hr))
{
//
// Copy the common stuff.
//
Dest.dwAlignPad = Src.dwAlignPad;
if (bCopyTag)
{
Dest.ulPropTag = Src.ulPropTag;
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::CopyVarLengthSPropVal
Description: Copies a single variable-length property value item from
one location in memory to another. Memory for the destination is
automatically allocated. This function is used to copy prop values
of types STRING, BINARY etc.
Arguments:
pvBaseAlloc - Pointer to use as the "lpObject" argument in a call
to MAPIAllocateMore().
ppvDest - Address of pointer variable to receive the address of
the newly allocated buffer.
pvSrc - Address of source buffer.
cb - Number of bytes to copy from pvSrc to *ppvDest.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiRecipients::CopyVarLengthSPropVal(
LPVOID pvBaseAlloc,
LPVOID *ppvDest,
LPVOID pvSrc,
INT cb
)
{
HRESULT hr;
hr = MAPI32.AllocateMore(cb, pvBaseAlloc, ppvDest);
if (SUCCEEDED(hr))
{
CopyMemory(*ppvDest, pvSrc, cb);
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::CopySPropValString (ANSI -> UNICODE)
Description: Copies a string property value item from one location
to another; performing ANSI/UNICODE translations as required.
Arguments:
pvBaseAlloc - Pointer to use as the "lpObject" argument in a call
to MAPIAllocateMore().
ppszDestW - Address of pointer variable to receive the address of
the newly wide character buffer.
pszSrcA - Address of ANSI source buffer.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiRecipients::CopySPropValString(
LPVOID pvBaseAlloc,
LPWSTR *ppszDestW,
LPCSTR pszSrcA
)
{
HRESULT hr;
INT cchW = MultiByteToWideChar(CP_ACP,
0,
pszSrcA,
-1,
NULL,
0);
hr = MAPI32.AllocateMore(cchW * sizeof(WCHAR), pvBaseAlloc, (LPVOID *)ppszDestW);
if (SUCCEEDED(hr))
{
MultiByteToWideChar(CP_ACP,
0,
pszSrcA,
-1,
*ppszDestW,
cchW);
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiRecipients::CopySPropValString (UNICODE -> ANSI)
Description: Copies a string property value item from one location
to another; performing ANSI/UNICODE translations as required.
Arguments:
pvBaseAlloc - Pointer to use as the "lpObject" argument in a call
to MAPIAllocateMore().
ppszDestA - Address of pointer variable to receive the address of
the newly ANSI character buffer.
pszSrcW - Address of UNICODE source buffer.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiRecipients::CopySPropValString(
LPVOID pvBaseAlloc,
LPSTR *ppszDestA,
LPCWSTR pszSrcW
)
{
HRESULT hr;
INT cchA = WideCharToMultiByte(CP_ACP,
0,
pszSrcW,
-1,
NULL,
0,
NULL, NULL);
hr = MAPI32.AllocateMore(cchA * sizeof(CHAR), pvBaseAlloc, (LPVOID *)ppszDestA);
if (SUCCEEDED(hr))
{
WideCharToMultiByte(CP_ACP,
0,
pszSrcW,
-1,
*ppszDestA,
cchA,
NULL, NULL);
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::CMapiSession
Description: Constructor.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
CMapiSession::CMapiSession(
VOID
) : m_pSession(NULL),
m_pDefMsgStore(NULL),
m_pOutBoxFolder(NULL),
m_bMapiInitialized(FALSE)
{
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::CMapiSession
Description: Constructor. Initializes MAPI.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
CMapiSession::~CMapiSession(
VOID
)
{
if (NULL != m_pOutBoxFolder)
{
m_pOutBoxFolder->Release();
}
if (NULL != m_pDefMsgStore)
{
m_pDefMsgStore->Release();
}
if (NULL != m_pSession)
{
m_pSession->Release();
}
if (m_bMapiInitialized)
{
//
// If MAPI was initialized, uninitialize it.
//
MAPI32.Uninitialize();
//
// Unload MAPI32.DLL.
//
MAPI32.Unload();
}
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::Initialize
Description: Initializes the session object by:
1. Initializing MAPI if neccessary,
2. Logging on to create the MAPI session object.
3. Open the default msg store.
4. Open the outbox folder in the default msg store.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiSession::Initialize(
VOID
)
{
HRESULT hr = NO_ERROR;
if (!m_bMapiInitialized)
{
//
// Load MAPI32.DLL.
//
MAPI32.Load();
//
// Initialize MAPI if it hasn't been initialized.
//
hr = MAPI32.Initialize(NULL);
if (SUCCEEDED(hr))
{
//
// Remember that MAPI has been initialized.
//
m_bMapiInitialized = TRUE;
}
}
if (m_bMapiInitialized)
{
//
// Attempt logon only if MAPI has been initialized.
//
DWORD dwLogonFlags = MAPI_TIMEOUT_SHORT | MAPI_USE_DEFAULT | MAPI_EXTENDED | MAPI_ALLOW_OTHERS;
#ifdef UNICODE
dwLogonFlags |= MAPI_UNICODE;
#endif
if (NULL != m_pSession)
{
//
// Release any previously-held session interface ptr.
//
m_pSession->Release();
m_pSession = NULL;
}
hr = MAPI32.LogonEx(0, // Hwnd for any UI.
NULL, // Profile name.
NULL, // Password.
dwLogonFlags, // Flags
&m_pSession); // Session obj ptr (out).
if (SUCCEEDED(hr))
{
ReportErrorsReturned(hr);
//
// We're logged on. Open the default msg store and out box folder.
//
hr = OpenDefMsgStore();
if (SUCCEEDED(hr))
{
hr = OpenOutBoxFolder();
}
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::OpenDefMsgStore
Description: Opens the session's default message store.
Stores the resulting IMsgStore ptr in m_pDefMsgStore.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiSession::OpenDefMsgStore(
VOID
)
{
HRESULT hr = E_FAIL;
SPropValue spv;
SizedSPropTagArray(2, sptCols) = {2, PR_ENTRYID, PR_DEFAULT_STORE};
SRestriction sres;
if (NULL != m_pSession)
{
LPMAPITABLE pTable;
hr = m_pSession->GetMsgStoresTable(0, &pTable);
ReportErrorsReturned(hr);
if (SUCCEEDED(hr))
{
//
// Find the entry ID for the default store in the session's
// msg stores table.
//
LPSRowSet pRow = NULL;
sres.rt = RES_PROPERTY;
sres.res.resProperty.relop = RELOP_EQ;
sres.res.resProperty.ulPropTag = PR_DEFAULT_STORE;
sres.res.resProperty.lpProp = &spv;
spv.ulPropTag = PR_DEFAULT_STORE;
spv.Value.b = TRUE;
hr = MAPI32.HrQueryAllRows(pTable, // Table ptr.
(LPSPropTagArray)&sptCols, // Column set
&sres, // Row restrictions
NULL, // Sort order set
0, // All rows.
&pRow); // Resulting row set (out)
ReportErrorsReturned(hr);
if (SUCCEEDED(hr))
{
SBinary sbEID = {0, NULL};
if (NULL != pRow &&
0 != pRow->cRows &&
0 != pRow->aRow[0].cValues &&
PR_ENTRYID == pRow->aRow[0].lpProps[0].ulPropTag)
{
sbEID = pRow->aRow[0].lpProps[0].Value.bin;
//
// Found the ID. Now open the store.
//
if (NULL != m_pDefMsgStore)
{
//
// Release any previously-held store interface ptr.
//
m_pDefMsgStore->Release();
m_pDefMsgStore = NULL;
}
hr = m_pSession->OpenMsgStore(0, // Hwnd for any UI
sbEID.cb, // Entry ID size.
(LPENTRYID)sbEID.lpb, // Entry ID ptr.
NULL, // Use Std iface.
MAPI_BEST_ACCESS, // read/write access
&m_pDefMsgStore); // Store ptr (out)
ReportErrorsReturned(hr);
}
else
{
hr = MAPI_E_NOT_FOUND;
}
MAPI32.FreeProws(pRow);
}
pTable->Release();
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::GetSessionUser
Description: Returns the address properties for the session user.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiSession::GetSessionUser(
LPSPropValue *ppProps,
ULONG *pcbProps
)
{
HRESULT hr = E_FAIL;
if (NULL != m_pSession)
{
ULONG cbEID = 0;
LPENTRYID lpEID = NULL;
//
// Get the "identity" of the session.
// In general, this is the user's entry ID.
//
hr = m_pSession->QueryIdentity(&cbEID, &lpEID);
if (SUCCEEDED(hr))
{
LPADRBOOK pAddrBook = NULL;
hr = GetAddressBook(&pAddrBook);
if (SUCCEEDED(hr))
{
ULONG ulObjType = 0;
IMAPIProp *pMailUser = NULL;
//
// Open the user's entry in the address book.
//
hr = pAddrBook->OpenEntry(cbEID,
lpEID,
NULL,
0,
&ulObjType,
(LPUNKNOWN *)&pMailUser);
if (SUCCEEDED(hr))
{
ULONG ulFlags = 0;
ULONG cProps = 0;
#ifdef UNICODE
//
// For unicode builds, we want UNICODE property strings.
//
ulFlags |= MAPI_UNICODE;
#endif
SizedSPropTagArray(5, tags) = { 5, PR_ADDRTYPE,
PR_DISPLAY_NAME,
PR_EMAIL_ADDRESS,
PR_ENTRYID,
PR_SEARCH_KEY };
//
// Retrieve the user properties and return them to
// the caller.
//
hr = pMailUser->GetProps((LPSPropTagArray)&tags, // Prop tags
ulFlags,
pcbProps, // Prop cnt (out)
ppProps); // Prop ptr (out)
ReportErrorsReturned(hr);
pMailUser->Release();
}
pAddrBook->Release();
}
MAPI32.FreeBuffer(lpEID);
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::OpenOutBoxFolder
Description: Opens the outbox folder in the default message store.
Stores the resulting IMAPIFolder ptr in m_pOutBoxFolder.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiSession::OpenOutBoxFolder(
VOID
)
{
HRESULT hr = E_FAIL;
if (NULL != m_pSession && NULL != m_pDefMsgStore)
{
LPSPropValue pProps = NULL;
ULONG ulObjType;
ULONG cProps;
ULONG ulFlags = 0;
#ifdef UNICODE
//
// For unicode builds, we want UNICODE property strings.
//
ulFlags |= MAPI_UNICODE;
#endif
SizedSPropTagArray(1, sptFolders) = { 1, PR_IPM_OUTBOX_ENTRYID };
//
// Retrieve the entry ID for the outbox in the default msg store.
//
hr = m_pDefMsgStore->GetProps((LPSPropTagArray)&sptFolders, // Prop tags
ulFlags,
&cProps, // Prop cnt (out)
&pProps); // Prop ptr (out)
ReportErrorsReturned(hr);
if (SUCCEEDED(hr))
{
if (0 != cProps && NULL != pProps)
{
if (pProps[0].ulPropTag == sptFolders.aulPropTag[0])
{
//
// Get the MAPI folder interface ptr for the outbox folder.
//
if (NULL != m_pOutBoxFolder)
{
//
// Release any previously-held outbox interface ptr.
//
m_pOutBoxFolder->Release();
m_pOutBoxFolder = NULL;
}
hr = m_pSession->OpenEntry(pProps[0].Value.bin.cb,
(LPENTRYID)pProps[0].Value.bin.lpb,
NULL,
MAPI_MODIFY,
&ulObjType,
(LPUNKNOWN *)&m_pOutBoxFolder);
ReportErrorsReturned(hr);
}
else
{
hr = MAPI_E_NOT_FOUND;
}
MAPI32.FreeBuffer(pProps);
}
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::Send
Description: Sends a message to a list of recipients. There are 3
versions that allow simple and complex formatting of the
message body:
1. First version lets you pass in simple text strings for the
subject and body text. This is the simplest way to send a
short, simple message.
2. Second version lets you pass in the subject as a simple text
string but then lets you create a complex message body as
a CMapiMessageBody object. This probably won't get used much.
3. The third version lets you create a CMapiMessage object
containing subject line and body text. This is the method
to use if you want to create a message with complex formatting.
Sending a message with CMapiSession::Send ensures that the
recipient list contains fully resolved names. If you send
a message with CMapiMessage::Send, you must resolve the names
before the Send call is made.
Arguments:
pAdrList - Address of MAPI ADRLIST structure containing the
list of recipient addresses. It is assumed that
the list contains unresolved recipients but this is not a
requirement.
pszSubject - Address of subject line text.
pszBody - Address of msg body text.
body - Reference to CMapiMessageBody object containing the message text.
msg - Reference to CMapiMessage object containing the subject line
and message body text.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
//
// Simplest form. Just give an address list, subject line and body
// text string.
//
HRESULT
CMapiSession::Send(
LPADRLIST pAdrList,
LPCTSTR pszSubject,
LPCTSTR pszBody
)
{
HRESULT hr;
//
// Create a local CMapiMessage object with the given subject
// line and body text. Then just send it.
//
CMapiMessage msg(m_pOutBoxFolder);
hr = msg.SetSubject(pszSubject);
if (SUCCEEDED(hr))
{
hr = msg.Append(pszBody);
if (SUCCEEDED(hr))
{
hr = Send(pAdrList, msg);
}
}
return hr;
}
//
// If you already have a CMapiMessageBody object created with text, this is the
// version you want to use.
//
HRESULT
CMapiSession::Send(
LPADRLIST pAdrList,
LPCTSTR pszSubject,
CMapiMessageBody& body
)
{
//
// Create a local CMapiMessage object with the given subject
// line and body text. Then just send it.
//
CMapiMessage msg(m_pOutBoxFolder, body, pszSubject);
return Send(pAdrList, msg);
}
//
// If you already have a CMapiMessage object create with subject line
// and body text, use this. The other versions of Send() eventually
// call this one.
//
HRESULT
CMapiSession::Send(
LPADRLIST pAdrList,
CMapiMessage& msg
)
{
HRESULT hr = E_FAIL;
hr = ResolveAddresses(pAdrList);
if (SUCCEEDED(hr))
{
hr = msg.Send(pAdrList);
DebugMsg(DM_ERROR, TEXT("CMapiSession::Send, Result = 0x%08X"), hr);
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::GetAddressBook
Description: Returns the session's address book pointer.
Arguments:
ppAdrBook - Address of pointer variable to receive pointer value.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiSession::GetAddressBook(
LPADRBOOK *ppAdrBook
)
{
HRESULT hr = E_POINTER;
if (NULL != m_pSession && NULL != ppAdrBook)
{
hr = m_pSession->OpenAddressBook(0, // Hwnd for UI
NULL, // Use std interface
AB_NO_DIALOG, // No UI.
ppAdrBook); // Book ptr (out)
ReportErrorsReturned(hr);
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::ReportErrorsReturned [static]
Description: When a MAPI function returns MAPI_W_ERRORS_RETURNED,
error information can be obtained by calling IMapiSession::GetLastError.
This function encapsulates the necessary behavior to get this error
information and dump it to the debugger. An option to write a
warning/error to the system event log is planned.
Arguments:
hr - HRESULT containing the error code.
bLogEvent [optional] - If True, write an event to the system event
log. This option is not in place yet.
Returns: Nothing.
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/16/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
VOID
CMapiSession::ReportErrorsReturned(
HRESULT hr,
BOOL bLogEvent // Unused at this time.
)
{
if (MAPI_W_ERRORS_RETURNED == hr)
{
LPMAPIERROR pMapiErrors = NULL;
DWORD dwFlags = 0;
#ifdef UNICODE
dwFlags |= MAPI_UNICODE;
#endif
hr = m_pSession->GetLastError(hr, dwFlags, &pMapiErrors);
if (S_OK == hr)
{
if (NULL != pMapiErrors)
{
DebugMsg(DM_ERROR, TEXT("MAPI returned errors.\n"));
DebugMsg(DM_ERROR, TEXT("\tVersion.......: %d"), pMapiErrors->ulVersion);
DebugMsg(DM_ERROR, TEXT("\tComponent.....: %s"), pMapiErrors->lpszComponent ?
pMapiErrors->lpszComponent :
TEXT("Not Provided"));
DebugMsg(DM_ERROR, TEXT("\tError.........: %s"), pMapiErrors->lpszError ?
pMapiErrors->lpszError :
TEXT("Not Provided"));
DebugMsg(DM_ERROR, TEXT("\tContext.......: %d"), pMapiErrors->ulContext);
DebugMsg(DM_ERROR, TEXT("\tLowLevel Error: %d\n"), pMapiErrors->ulLowLevelError);
if (bLogEvent)
{
//
// Open event log and write error?
//
}
MAPI32.FreeBuffer(pMapiErrors);
}
}
}
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::GetOutBoxFolder
Description: Returns the session's outbox folder pointer.
Arguments:
ppOutBoxFolder - Address of pointer variable to receive pointer value.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiSession::GetOutBoxFolder(
LPMAPIFOLDER *ppOutBoxFolder
)
{
HRESULT hr = E_POINTER;
if (NULL != m_pOutBoxFolder && NULL != ppOutBoxFolder)
{
*ppOutBoxFolder = m_pOutBoxFolder;
(*ppOutBoxFolder)->AddRef();
hr = NO_ERROR;
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiSession::ResolveAddresses
Description: Resolves names in an address list.
Arguments:
pAdrList - Pointer to the address list to be resolved.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiSession::ResolveAddresses(
LPADRLIST pAdrList
)
{
HRESULT hr;
LPADRBOOK pAdrBook = NULL;
hr = GetAddressBook(&pAdrBook);
if (SUCCEEDED(hr))
{
hr = pAdrBook->ResolveName(0, // Hwnd for UI
0, // Flags (no UI).
NULL, // Dlg title (none).
pAdrList); // ADRLIST ptr
ReportErrorsReturned(hr);
pAdrBook->Release();
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessageBody::CMapiMessageBody
Description: Constructors.
Arguments:
rhs - Reference to source CMapiMessageBody object in copy ctor.
Returns: Nothing.
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
CMapiMessageBody::CMapiMessageBody(
VOID
) : m_pStg(NULL),
m_pStm(NULL)
{
CommonConstruct();
}
CMapiMessageBody::CMapiMessageBody(
const CMapiMessageBody& rhs
) : m_pStg(NULL),
m_pStm(NULL)
{
CommonConstruct();
*this = rhs;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessageBody::~CMapiMessageBody
Description: Destructor.
Arguments: None.
Returns: Nothing.
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
CMapiMessageBody::~CMapiMessageBody(
VOID
)
{
if (NULL != m_pStm)
{
m_pStm->Release();
}
if (NULL != m_pStg)
{
m_pStg->Release();
}
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessageBody::operator =
Description: Assignment operator.
Arguments:
rhs - Reference to source CMapiMessageBody object.
Returns: Reference to *this.
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
CMapiMessageBody&
CMapiMessageBody::operator = (
const CMapiMessageBody& rhs
)
{
if (this != &rhs)
{
if (NULL != m_pStm && NULL != rhs.m_pStm)
{
HRESULT hr;
LPSTREAM pStmClone;
//
// Create a clone of the source stream so that we don't alter
// the source's seek ptr.
//
hr = rhs.m_pStm->Clone(&pStmClone);
if (SUCCEEDED(hr))
{
ULARGE_INTEGER ulSize = {0, 0};
LARGE_INTEGER lSeek = {0, 0};
//
// Reset the source stream seek ptr to the beginnging.
//
pStmClone->Seek(lSeek, STREAM_SEEK_SET, NULL);
//
// Truncate the destination stream to clear it.
//
m_pStm->SetSize(ulSize);
//
// Copy all of the source stream to the dest stream.
//
ulSize.LowPart = 0xFFFFFFFF;
hr = pStmClone->CopyTo(m_pStm, // Destination stream.
ulSize, // Write all bytes.
NULL, // pcbRead
NULL); // pcbWritten
pStmClone->Release();
}
}
}
return *this;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessageBody::CommonConstruct
Description: Performs operations common to all forms of constructors.
1. Creates Storage object.
2. Creates Stream "MSGBODY" in storage.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiMessageBody::CommonConstruct(
VOID
)
{
HRESULT hr;
DWORD grfMode = STGM_DIRECT | STGM_READWRITE |
STGM_CREATE | STGM_SHARE_EXCLUSIVE;
//
// Create the output doc file.
//
hr = StgCreateDocfile(NULL, // Temp file with unique name.
grfMode, // Access flags.
0, // Reserved
&m_pStg); // Stg ptr (out)
if (SUCCEEDED(hr))
{
//
// Create the stream in the doc file.
//
hr = m_pStg->CreateStream(L"MSGBODY", // Stream name.
grfMode, // Access flags.
0, // Reserved.
0, // Reserved.
&m_pStm); // Stream ptr (out)
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessageBody::Append
Description: Appends text to the msg body stream. Two versions are
provided. One accepts a nul-terminated format string while the other
accepts the resource ID for a string resource. Both formats allow
variable replacement arguments for replaceable arguments in the
format strings (i.e. %1, %2 etc.)
Arguments:
hInst - Module instance handle for string/message resource.
pszFmt - Address of format string.
idFmt - ID of format resource string. May be a string or message
resource.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiMessageBody::Append(
LPCTSTR pszFmt,
...
)
{
HRESULT hr;
va_list args;
va_start(args, pszFmt);
hr = Append(pszFmt, &args);
va_end(args);
return hr;
}
HRESULT
CMapiMessageBody::Append(
LPCTSTR pszFmt,
va_list *pargs
)
{
HRESULT hr = E_POINTER;
if (NULL != m_pStm)
{
CString str;
if (str.Format(pszFmt, pargs))
{
hr = m_pStm->Write((LPCTSTR)str, str.LengthBytes(), NULL);
}
}
return hr;
}
HRESULT
CMapiMessageBody::Append(
HINSTANCE hInst,
UINT idFmt,
...
)
{
HRESULT hr;
va_list(args);
va_start(args, idFmt);
hr = Append(hInst, idFmt, &args);
va_end(args);
return hr;
}
HRESULT
CMapiMessageBody::Append(
HINSTANCE hInst,
UINT idFmt,
va_list *pargs
)
{
HRESULT hr = E_POINTER;
if (NULL != m_pStm)
{
CString str;
if (str.Format(hInst, idFmt, pargs))
{
hr = m_pStm->Write((LPCTSTR)str, str.LengthBytes(), NULL);
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessage::CMapiMessage
Description: Constructors.
Arguments:
pFolder - Address of LPMAPIFOLDER object in which the message is
to be created.
body - Reference to a CMapiMessageBody object containing text for the
body of the message.
pszSubject [optional] - Address of message subject line string.
Returns: Nothing.
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
CMapiMessage::CMapiMessage(
LPMAPIFOLDER pFolder
) : m_pMsg(NULL)
{
CommonConstruct(pFolder);
}
CMapiMessage::CMapiMessage(
LPMAPIFOLDER pFolder,
CMapiMessageBody& body,
LPCTSTR pszSubject /* optional */
) : m_pMsg(NULL),
m_body(body)
{
CommonConstruct(pFolder);
if (NULL != pszSubject)
{
SetSubject(pszSubject);
}
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessage::~CMapiMessage
Description: Destructor.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
CMapiMessage::~CMapiMessage(
VOID
)
{
if (NULL != m_pMsg)
{
m_pMsg->Release();
}
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessage::CommonConstruct
Description: Performs operations common to all forms of constructors.
1. Creates the MAPI message object.
2. Sets the DELETE_AFTER_SUBMIT property (common to all messages).
Arguments:
pFolder - Address of LPMAPIFOLDER object in which the message is
to be created.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiMessage::CommonConstruct(
LPMAPIFOLDER pFolder
)
{
HRESULT hr = E_POINTER;
if (NULL != pFolder)
{
hr = pFolder->CreateMessage(NULL, 0, &m_pMsg);
if (SUCCEEDED(hr))
{
//
// Don't want sent message hanging around in users's outbox.
//
SPropValue rgProp[1];
rgProp[0].ulPropTag = PR_DELETE_AFTER_SUBMIT;
rgProp[0].Value.b = TRUE;
hr = m_pMsg->SetProps(ARRAYSIZE(rgProp), rgProp, NULL);
}
}
return hr;
}
HRESULT
CMapiMessage::SetProps(
ULONG cValues,
LPSPropValue lpPropArray,
LPSPropProblemArray *lppProblems
)
{
HRESULT hr = E_UNEXPECTED;
if (NULL != m_pMsg)
{
hr = m_pMsg->SetProps(cValues, lpPropArray, lppProblems);
}
return hr;
}
HRESULT
CMapiMessage::SaveChanges(
ULONG ulFlags
)
{
HRESULT hr = E_UNEXPECTED;
if (NULL != m_pMsg)
{
hr = m_pMsg->SaveChanges(ulFlags);
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessage::SetSubject
Description: Sets the PR_SUBJECT property of the message.
Arguments:
pszSubject - Address of new subject string.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiMessage::SetSubject(
LPCTSTR pszSubject
)
{
HRESULT hr = E_POINTER;
if (NULL != m_pMsg)
{
//
// Set the msg subject text property.
//
SPropValue spvProp;
spvProp.ulPropTag = PR_SUBJECT;
spvProp.Value.LPSZ = (LPTSTR)pszSubject;
hr = m_pMsg->SetProps(1, &spvProp, NULL);
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessage::SetRecipients
Description: Sets the recipients for the message. It is assumed that
the recipients in pAdrList have been resolved.
Arguments:
pAdrList - Address of MAPI address list containing resolved addresses.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiMessage::SetRecipients(
LPADRLIST pAdrList
)
{
HRESULT hr = E_POINTER;
if (NULL != m_pMsg)
{
hr = m_pMsg->ModifyRecipients(0, pAdrList);
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessage::Send
Description: Sends the message to a list of recipients.
Arguments:
pAdrList [optional] - Address of MAPI address list containing
resolved addresses. If this argument is NULL, the caller must
call SetRecipients before calling Send.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiMessage::Send(
LPADRLIST pAdrList /* optional */
)
{
HRESULT hr = E_POINTER;
if (NULL != m_pMsg)
{
hr = NO_ERROR;
if (NULL != pAdrList)
{
//
// If there's an address list, set the recipients.
// If not, the caller must call SetRecipients before calling Send().
// Otherwise, there will be no recipients to send it to.
//
hr = SetRecipients(pAdrList);
}
if (SUCCEEDED(hr))
{
LPSTREAM pPropStream;
hr = m_pMsg->OpenProperty(PR_BODY,
&IID_IStream,
STGM_READWRITE | STGM_DIRECT,
MAPI_CREATE | MAPI_MODIFY,
(LPUNKNOWN *)&pPropStream);
if (S_OK == hr)
{
LPSTREAM pMsgBodyStm = (LPSTREAM)m_body;
LPSTREAM pMsgBodyStmClone;
//
// Clone the body stream so that we don't alter it's seek ptr.
//
hr = pMsgBodyStm->Clone(&pMsgBodyStmClone);
if (SUCCEEDED(hr))
{
ULARGE_INTEGER ulSize = {0xFFFFFFFF, 0};
LARGE_INTEGER lSeek = {0, 0};
//
// Copy the msg body stream to the PR_BODY property.
//
pMsgBodyStmClone->Seek(lSeek, STREAM_SEEK_SET, NULL);
pMsgBodyStmClone->CopyTo(pPropStream, ulSize, NULL, NULL);
pPropStream->Commit(STGC_DEFAULT);
//
// Release temp streams.
//
pPropStream->Release();
pMsgBodyStmClone->Release();
//
// Send it!
// Note: Calling SaveChanges() is not required if the message
// is being sent immediately.
//
hr = m_pMsg->SubmitMessage(FORCE_SUBMIT);
}
}
}
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: CMapiMessage::Append
Description: Appends text to the msg body stream. Two versions are
provided. One accepts a nul-terminated format string while the other
accepts the resource ID for a string resource. Both formats allow
variable replacement arguments for replaceable arguments in the
format strings (i.e. %1, %2 etc.)
Arguments:
hInst - Module instance handle for string/message resource.
pszFmt - Address of format string.
idFmt - ID of format resource string. May be a string or message
resource.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/18/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
CMapiMessage::Append(
LPCTSTR pszFmt,
...
)
{
HRESULT hr;
va_list args;
va_start(args, pszFmt);
hr = m_body.Append(pszFmt, &args);
va_end(args);
return hr;
}
HRESULT
CMapiMessage::Append(
HINSTANCE hInst,
UINT idFmt,
...
)
{
HRESULT hr;
va_list(args);
va_start(args, idFmt);
hr = m_body.Append(hInst, idFmt, &args);
va_end(args);
return hr;
}
//
// Static members of class MAPI.
//
LONG MAPI::m_cLoadCount;
HINSTANCE MAPI::m_hmodMAPI;
LPMAPIINITIALIZE MAPI::m_pfnInitialize;
LPMAPILOGONEX MAPI::m_pfnLogonEx;
LPMAPIUNINITIALIZE MAPI::m_pfnUninitialize;
LPMAPIALLOCATEBUFFER MAPI::m_pfnAllocateBuffer;
LPMAPIALLOCATEMORE MAPI::m_pfnAllocateMore;
LPMAPIFREEBUFFER MAPI::m_pfnFreeBuffer;
LPMAPIHRQUERYALLROWS MAPI::m_pfnHrQueryAllRows;
LPMAPIFREEPADRLIST MAPI::m_pfnFreePadrlist;
LPMAPIFREEPROWS MAPI::m_pfnFreeProws;
///////////////////////////////////////////////////////////////////////////////
/* Function: MAPI::MAPI
Description: Constructor.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/22/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
MAPI::MAPI(
VOID
)
{
}
///////////////////////////////////////////////////////////////////////////////
/* Function: MAPI::~MAPI
Description: Destructor. Ensures MAPI32.DLL is unloaded.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/22/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
MAPI::~MAPI(
VOID
)
{
//
// m_cLoadCount should be 0 at this point if calls to Load() and
// Unload() are balanced.
//
ASSERT(0 == m_cLoadCount);
if (0 < m_cLoadCount)
{
//
// Calls to Load() and Unload() are not balanced due to programmer
// error or maybe an exception preventing a call to Unload().
// This will force Unload to call FreeLibrary().
//
m_cLoadCount = 1;
}
Unload();
}
///////////////////////////////////////////////////////////////////////////////
/* Function: MAPI::Load
Description: Load MAPI32.DLL and call GetProcAddress for all of the
MAPI32 functions we're interested in using. Maintains a reference
count so redundant calls to LoadLibrary are avoided.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/22/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
HRESULT
MAPI::Load(
VOID
)
{
Assert(0 <= m_cLoadCount);
if (0 == m_cLoadCount++)
{
m_hmodMAPI = ::LoadLibrary(TEXT("MAPI32.DLL"));
if (NULL != m_hmodMAPI)
{
m_pfnInitialize = (LPMAPIINITIALIZE) ::GetProcAddress(m_hmodMAPI, "MAPIInitialize");
m_pfnLogonEx = (LPMAPILOGONEX) ::GetProcAddress(m_hmodMAPI, "MAPILogonEx");
m_pfnUninitialize = (LPMAPIUNINITIALIZE) ::GetProcAddress(m_hmodMAPI, "MAPIUninitialize");
m_pfnAllocateBuffer = (LPMAPIALLOCATEBUFFER)::GetProcAddress(m_hmodMAPI, "MAPIAllocateBuffer");
m_pfnAllocateMore = (LPMAPIALLOCATEMORE) ::GetProcAddress(m_hmodMAPI, "MAPIAllocateMore");
m_pfnFreeBuffer = (LPMAPIFREEBUFFER) ::GetProcAddress(m_hmodMAPI, "MAPIFreeBuffer");
m_pfnHrQueryAllRows = (LPMAPIHRQUERYALLROWS)::GetProcAddress(m_hmodMAPI, "HrQueryAllRows@24");
m_pfnFreePadrlist = (LPMAPIFREEPADRLIST) ::GetProcAddress(m_hmodMAPI, "FreePadrlist@4");
m_pfnFreeProws = (LPMAPIFREEPROWS) ::GetProcAddress(m_hmodMAPI, "FreeProws@4");
}
}
return (NULL != m_hmodMAPI) ? NO_ERROR : E_FAIL;
}
///////////////////////////////////////////////////////////////////////////////
/* Function: MAPI::Unload
Description: Unloads MAPI32.DLL if the reference count drops to 0. If
the library is unloaded, all of the function pointers are
set to NULL.
Arguments: None.
Returns:
Revision History:
Date Description Programmer
-------- --------------------------------------------------- ----------
06/22/97 Initial creation. BrianAu
*/
///////////////////////////////////////////////////////////////////////////////
VOID
MAPI::Unload(
VOID
)
{
ASSERT(0 < m_cLoadCount);
if (0 == --m_cLoadCount)
{
if (NULL != m_hmodMAPI)
{
::FreeLibrary(m_hmodMAPI);
m_hmodMAPI = NULL;
}
m_pfnInitialize = NULL;
m_pfnLogonEx = NULL;
m_pfnUninitialize = NULL;
m_pfnAllocateBuffer = NULL;
m_pfnAllocateMore = NULL;
m_pfnFreeBuffer = NULL;
m_pfnHrQueryAllRows = NULL;
m_pfnFreePadrlist = NULL;
m_pfnFreeProws = NULL;
}
}
///////////////////////////////////////////////////////////////////////////////
// The remaining MAPI::XXXX functions are merely simple wrappers around the
// corresponding functions in MAPI32.DLL.
// See the MAPI SDK for information concerning their use.
///////////////////////////////////////////////////////////////////////////////
HRESULT
MAPI::LogonEx(
ULONG ulUIParam,
LPTSTR lpszProfileName,
LPTSTR lpszPassword,
FLAGS flFlags,
LPMAPISESSION FAR * lppSession
)
{
ASSERT(NULL != m_pfnLogonEx);
if (NULL != m_pfnLogonEx)
return (*m_pfnLogonEx)(ulUIParam, lpszProfileName, lpszPassword, flFlags, lppSession);
else
return E_POINTER;
}
HRESULT
MAPI::Initialize(
LPVOID lpMapiInit
)
{
ASSERT(NULL != m_pfnInitialize);
if (NULL != m_pfnInitialize)
return (*m_pfnInitialize)(lpMapiInit);
else
return E_POINTER;
}
VOID
MAPI::Uninitialize(
VOID
)
{
ASSERT(NULL != m_pfnUninitialize);
if (NULL != m_pfnUninitialize)
(*m_pfnUninitialize)();
}
SCODE
MAPI::AllocateBuffer(
ULONG cbSize,
LPVOID FAR * lppBuffer
)
{
ASSERT(NULL != m_pfnAllocateBuffer);
if (NULL != m_pfnAllocateBuffer)
return (*m_pfnAllocateBuffer)(cbSize, lppBuffer);
else
return E_POINTER;
}
SCODE
MAPI::AllocateMore(
ULONG cbSize,
LPVOID lpObject,
LPVOID FAR * lppBuffer
)
{
ASSERT(NULL != m_pfnAllocateMore);
if (NULL != m_pfnAllocateMore)
return (*m_pfnAllocateMore)(cbSize, lpObject, lppBuffer);
else
return E_POINTER;
}
ULONG
MAPI::FreeBuffer(
LPVOID lpBuffer
)
{
ASSERT(NULL != m_pfnFreeBuffer);
if (NULL != m_pfnFreeBuffer)
return (*m_pfnFreeBuffer)(lpBuffer);
else
return (ULONG)E_POINTER;
}
HRESULT
MAPI::HrQueryAllRows(
LPMAPITABLE lpTable,
LPSPropTagArray lpPropTags,
LPSRestriction lpRestriction,
LPSSortOrderSet lpSortOrderSet,
LONG crowsMax,
LPSRowSet FAR *lppRows
)
{
ASSERT(NULL != m_pfnHrQueryAllRows);
if (NULL != m_pfnHrQueryAllRows)
return (*m_pfnHrQueryAllRows)(lpTable,
lpPropTags,
lpRestriction,
lpSortOrderSet,
crowsMax,
lppRows);
else
return E_POINTER;
}
VOID
MAPI::FreePadrlist(
LPADRLIST lpAdrList
)
{
ASSERT(NULL != m_pfnFreePadrlist);
if (NULL != m_pfnFreePadrlist)
(*m_pfnFreePadrlist)(lpAdrList);
}
VOID
MAPI::FreeProws(
LPSRowSet lpRows
)
{
ASSERT(NULL != m_pfnFreeProws);
if (NULL != m_pfnFreeProws)
(*m_pfnFreeProws)(lpRows);
}