Click here to Skip to main content
Click here to Skip to main content

Programmatically adding attachments to emails

By , 18 Jan 2006
 

Introduction

Having read Stephane Rodriguez's excellent and rather clever solution to programmatically adding attachments to Outlook emails, I realized how subtly useful this functionality was. Many was the time I have needed a way to send an application's document to another user. Obviously, it could be done by writing your own dialogs that mimic the mail client dialogs, but you have to hook up the address book and so on. It's far better to exploit the existing mail client software installed on the PC. I needed exactly this for an application so it could have File->Send To functionality - obviously, it needs attachments to work.

SendTo or MailTo

Stephane's article makes the perfectly valid point that using the SendTo approach is functionally better than using the mailto trick. Another compelling reason for not using mailto is it doesn't support attachments. The RFC mailto protocol is simple and doesn't specify attachments.

However, you commonly see code trying to use mailto like this:

mailto:shiver@metimbers.com?Subject=Ahoy there 
  shipmate&Body=Here's the shipping 
  manifest&Attach="D:\manifest.doc"

It probably won't attach the document because you are at the liberty of the email client to implement the mailto protocol and include parsing for the attachment clause. You may not know what mail client is installed on the PC, so it may not always work - Outlook certainly doesn't support attachments using mailto.

So what's the best (and easiest) way to do it

Stephane's solution is neat. It simulates a drag 'n drop into Outlook by using an unpublished mail helper COM object. However, by using unpublished functionality, you are at the mercy of the vendor (in this case Microsoft) changing things! And this is what has happened. The drag drop code works with no problem at all when using Outlook 2002/2003 on Win2K, but throws a 'Privileged exception' when running on XP. The presumption is there are permission issues with this combination of OS and the version of Outlook.

I needed a mail client version independent solution that supported attachments, so I chose to investigate MAPI. As it turns out, the answer is pretty simple, which is wrapped up in the CSendFileTo class.

#ifndef __SENDFILETO_H__
#define __SENDFILETO_H__

#include <mapi.h>

class CSendFileTo
{
public:
    bool SendMail(HWND hWndParent, 
         CString const &strAttachmentFileName, 
         CString const &strSubject=_T(""))
    {
        // The attachment must exist as a file on the system
        // or MAPISendMail will fail, so......
        if (strAttachmentFileName.IsEmpty())
            return false;

        // You may want to remove this check, but if a valid
        // HWND is passed in, the mail dialog will be made
        // modal to it's parent.
        if (!hWndParent || !::IsWindow(hWndParent))
            return false;

        HINSTANCE hMAPI = ::LoadLibraryA(_T("MAPI32.DLL"));
        if (!hMAPI)
            return false;

        // Grab the exported entry point for the MAPISendMail function
        ULONG (PASCAL *SendMail)(ULONG, ULONG_PTR, 
                      MapiMessage*, FLAGS, ULONG);
        (FARPROC&)SendMail = GetProcAddress(hMAPI, 
                              _T("MAPISendMail"));

        if (!SendMail)
            return false;

        TCHAR szFileName[_MAX_PATH];
        TCHAR szPath[_MAX_PATH];
        TCHAR szSubject[_MAX_PATH];
        ::StrCpy(szFileName, strAttachmentFileName.GetString());
        ::StrCpy(szPath, strAttachmentFileName.GetString());
        ::StrCpy(szSubject, strSubject.GetString());

        MapiFileDesc fileDesc;
        ::ZeroMemory(&fileDesc, sizeof(fileDesc));
        fileDesc.nPosition = (ULONG)-1;
        fileDesc.lpszPathName = szPath;
        fileDesc.lpszFileName = szFileName;

        MapiMessage message;
        ::ZeroMemory(&message, sizeof(message));
        message.lpszSubject = szSubject;
        message.nFileCount = 1;
        message.lpFiles = &fileDesc;

        // Ok to send
        int nError = SendMail(0, (ULONG_PTR)hWndParent, 
               &message, MAPI_LOGON_UI|MAPI_DIALOG, 0);

        if (nError != SUCCESS_SUCCESS && 
            nError != MAPI_USER_ABORT && 
            nError != MAPI_E_LOGIN_FAILURE)
              return false;

        return true;
    }
};

#endif

Example use

This nugatory code fragment shows how easy it is to use.

#include "SendFileTo.h"
...
...

CSendFileTo sendTo;
sendTo.(m_hWnd, _T("c://documents//menu.doc"), 
                _T("Here's the lunch menu"));

...
...

This is all straightforward, but there are a few points to note.

  • If the attachment doesn't exist as a file on the file system, the call to MAPISendMail will fail with MAPI_E_ATTACHMENT_NOT_FOUND. Hence the check at the start of the SendMail call.
  • By passing the parent HWND to the MAPISendMail function, the email client is supposed to make the send mail dialog modal to the given HWND. You may want to remove this modalness(?) and simply use HWND_DESKTOP.
  • When the MAPISendMail call is made, it doesn't send the mail, it just pops up the email client dialog with the optional subject line set and the attachment attached.
  • This code was written to compile and work with WTL (it rocks), but will work equally well using MFC.

I've successfully tested this with Outlook 2002 and 2003 on Win2K and XP. I'd be interested in hearing if it works with other mail clients I don't have access to - Eudora etc.

This class could probably do with some more error checking but I'll leave that as an exercise for the reader.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

About the Author

David M Brooks
Web Developer
United Kingdom United Kingdom
Member
When Dave isn't trying to play drums, fly helicopters or race cars, he can be found trying to be CTO at www.eventility.co.uk He must try harder!
 
You can read Dave's ramblings in his blog Aliens Ate My GUI
 
Or, if you can be bothered, he twitters on BoomerDave

Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
Hint: For improved responsiveness ensure Javascript is enabled and choose 'Normal' from the Layout dropdown and hit 'Update'.
You must Sign In to use this message board.
Search this forum  
    Spacing  Noise  Layout  Per page   
QuestionNice, works now with wxwidgets as wellmemberMember 90858329 Jul '12 - 8:12 
Questionsendto with attachment solutionmembermanywebbs9 Sep '11 - 18:14 
GeneralMy vote of 5memberitz_faraz5 May '11 - 19:31 
GeneralI hate to nag but this code sample is bad...memberdc_200025 Apr '11 - 17:46 
QuestionCan I use this solution for my problem....memberdkperez21 Jan '10 - 12:16 
Tabular html form into which a user enters 1 or more titles and matching file spec. In this case the filespec is the full path, filename, and extension on the user's local system... SO, for an image called sunrise.jpg what they'd have to enter is something like
d:\images\morning\sunrise.jpg.
 
When done the user hits SUBMIT and everything gets sent to php and stored.
 
I need them to email the files they've entered to an address, and I'd prefer to do this from the client so I don't have to upload a bunch of large image files to the server.
 
So, I'd like to use some form of client emailer, preferably one that makes use of the mail engine on the user's PC so they don't have to wait while a server "mail" synchronously processes and sends a bunch of images (seems very slow)......
 
If it WILL work in this case, can someone provide very specific information about where the code has to go and how to use whatever does the call in html...
 
thanks.
GeneralWARNING = THIS CODE PROBABLY UNSAFEmembersashasawchai1 Dec '08 - 22:21 
GeneralRe: WARNING = THIS CODE PROBABLY UNSAFEmemberDavid M Brooks27 Aug '09 - 5:14 
Generalto send more than 1 mail address [modified]memberArielR2 Jul '08 - 10:08 
GeneralRe: to send more than 1 mail addressmemberpine_le11 Apr '10 - 4:28 
Questionwhat for ?memberkilt25 Jun '08 - 19:59 
QuestionLotus Notes ignores lpszFileNamemembertiggerlily29 Nov '07 - 23:08 
Questionwhy it saves the draft into inbox folder?memberxuxiaohui7 Nov '07 - 22:37 
QuestionNewbie problem - can't compile (get errors in mapi.h)membercgalpin1 Nov '07 - 4:59 
AnswerRe: Newbie problem - can't compile (get errors in mapi.h)memberVarchas R S4 Nov '07 - 23:03 
GeneralRe: Newbie problem - can't compile (get errors in mapi.h)membercgalpin6 Nov '07 - 7:42 
QuestionSending html formatted emailmemberkryzzozz7810 Oct '07 - 9:09 
GeneralRe: Sending html formatted emailmemberMember 216347031 Mar '08 - 4:56 
Generalprofile problem with MAPImemberrob010329 Jul '07 - 23:48 
GeneralRe: profile problem with MAPImemberBhushanKalse27 Aug '07 - 20:28 
Generalthanks!membercronky7812 Jul '07 - 20:17 
Generalsending problemmember133mmxtr11 Jun '07 - 2:20 
QuestionWCHAR version of Mapi32.dll ?? [modified]memberajitatif angajetor24 Jan '07 - 2:49 
AnswerRe: WCHAR version of Mapi32.dll ??memberhannahb21 Mar '07 - 3:22 
GeneralRe: WCHAR version of Mapi32.dll ??memberajitatif angajetor27 Mar '07 - 6:21 
Generalwants demo & sourcemembershital_harode@rediffmail.com17 Nov '06 - 23:24 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Permalink | Advertise | Privacy | Mobile
Web03 | 2.6.130516.1 | Last Updated 18 Jan 2006
Article Copyright 2006 by David M Brooks
Everything else Copyright © CodeProject, 1999-2013
Terms of Use
Layout: fixed | fluid