|

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(""))
{
if (strAttachmentFileName.IsEmpty())
return false;
if (!hWndParent || !::IsWindow(hWndParent))
return false;
HINSTANCE hMAPI = ::LoadLibraryA(_T("MAPI32.DLL"));
if (!hMAPI)
return false;
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;
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.
| You must Sign In to use this message board. |
|
| | Msgs 1 to 25 of 55 (Total in Forum: 55) (Refresh) | FirstPrevNext |
|
 |
|
|
 |
|
|
what must to modify at the code to send more than one mail address. Thanks
modified on Wednesday, July 2, 2008 5:32 PM
|
| Sign In·View Thread·PermaLink | 3.00/5 (2 votes) |
|
|
|
 |
|
|
An "article" which just copies MSDN code or hundreds of Usenet codes posted for 13 years ? Is it a joke ? Pathetic 
|
| Sign In·View Thread·PermaLink | 1.00/5 (4 votes) |
|
|
|
 |
|
|
Lotus Notes (v7 and v8) ignores lpszFileName when attaching a file with Simple MAPI. Instead, Notes uses the filename from lpszPathName as the displayable attachment filename seen by the recipients.
Caveat: If a single file is attached with lpszPathName ending in ".tmp", Notes does use lpszFileName as displayed name. But, attempting to attach more than one file with name ending in ".tmp" causes "Notes MailMan error" to pop up. In order to attach multiple files, each lpszPathName must end in other than ".tmp", e.g., ".temp" or ".jpg". But, Notes disregards lpszFileName unless lpszPathName ends with ".tmp".
Outlook correctly honors lpszFileName.
As a workaround for Notes, I could use the desired displayable name as the filename portion of lpszPathName. For example, my app could download URL to \docs and settings\joe\Local Settings\Temp\Harry Potter.txt. But, that doesn't work when lpszPathName contains special unicode characters.
Does anyone have a solution?
http://msdn2.microsoft.com/en-us/library/ms530443.aspx says:
lpszPathName
Pointer to the fully qualified path of the attached file. This path should include the disk drive letter and directory name.
lpszFileName
Pointer to the attachment filename seen by the recipient, which may differ from the filename in the lpszPathName member if temporary files are being used. If the lpszFileName member is empty or NULL, the filename from lpszPathName is used.
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
Thanks your code!
But I have the same question about "why it saves the draft into inbox folder?"
Thank you!
Peter
----------------------------------------------------------- Two Questions? Sstar 23:13 1 May '06 2. When the Mail dialog pops up, instead of sending it, if I click on the 'save' button, found that the mail is saved into my INBOX folder. By right, the mail should be saved into the DRAFT folder, shouldn't it? Is there any option that we could set somewhere to correct this problem when using the MAPI_SENDMAIL function? I am using Ms Outlook 2003.
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
Sorry for the Newbie question, but I have tried to make a simple test of this using both a console application project and a windows application project in Visual Studio 2005 with no luck. I'm not sure what headers to include to make this work, but both projects fail compiling mapi.h (I get 198 errors - first few shown below).
thanks charles
Error 1 error C2143: syntax error : missing ';' before '*' C:\Program Files\Microsoft Visual Studio 8\VC\PlatformSDK\include\mapi.h 43 Error 2 error C4430: missing type specifier - int assumed. Note: C++ does not support default-int C:\Program Files\Microsoft Visual Studio 8\VC\PlatformSDK\include\mapi.h 43 Error 3 error C2146: syntax error : missing ';' before identifier 'LHANDLE' C:\Program Files\Microsoft Visual Studio 8\VC\PlatformSDK\include\mapi.h 48 Error 4 error C4430: missing type specifier - int assumed. Note: C++ does not support default-int C:\Program Files\Microsoft Visual Studio 8\VC\PlatformSDK\include\mapi.h 48 Error 5 error C4430: missing type specifier - int assumed. Note: C++ does not support default-int C:\Program Files\Microsoft Visual Studio 8\VC\PlatformSDK\include\mapi.h 48 Error 6 error C2143: syntax error : missing ';' before '*' C:\Program Files\Microsoft Visual Studio 8\VC\PlatformSDK\include\mapi.h 48
and so on
|
| Sign In·View Thread·PermaLink | 5.00/5 (1 vote) |
|
|
|
 |
|
|
 |
|
|
I did figure out I need ATL/WTL (it was even mentioned in the original post) but still get various errors I can't resolve. Would someone mind making a quick project that compiles for this so we can see how to satisfy the dependencies?
|
| Sign In·View Thread·PermaLink | 2.00/5 (2 votes) |
|
|
|
 |
|
|
Has anyone sent an html formatted email?
I have this:
MapiMessage Msg;
Msg.lpszNoteText = szBody; // body
//If I sent text in lpszNoteText, then email format will be plain text //If I sent lpsZNoteText empty, then email format will be html
So, I want to send a link in the body, something like:
<a href:"http://www.codeproject.com">Link to codeproject</a>
Can anyone help me on this. Any advice would be greatly appreciated.
Thanks.
|
| Sign In·View Thread·PermaLink | 5.00/5 (1 vote) |
|
|
|
 |
|
|
 |
|
|
this code works great !!! thaaanks But i 've one problem We run our solution on a CITRIX server and so i've another profile then the profile on my local workstation.
so everytime i send an email a dialog appears where i should coose the profile.
How can i change the profile name or set the default profile name
Maybe someone can give me an example ? Pleeaasee !!
I don't know how to do
greets Robert
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
 |
|
|
 |
|
|
When my outlook is off, MAPISendMail does not work properly. When i call MAPISendMail, the new mail dialog is shown, subject, to and attachment parts are filled. I click send button and the new mail window terminates. But the mail is not sent until i click send-receive button on outlook. With mailto call, the mail is sent directly without clicking send-receive button. But mailto call does not support attachments.
|
| Sign In·View Thread·PermaLink | 2.00/5 (1 vote) |
|
|
|
 |
|
|
thanks for the code,it was more than a bunch of help. but my platform sdk (Win2K3 R2) has the Mapi structures defined using LPSTR,rather than LPTSTR,which gives me trouble especially about non-english characters in the message.
i hopelessly tried changing the header file but as expected without result. does anyone have any idea about this matter?
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
Here is a UNICODE only version. I didn't bother setting up for TCHAR template types.
#ifndef __SENDFILETO_H__ #define __SENDFILETO_H__
#include <mapi.h> #include
class CSendFileTo { public: bool SendMail(HWND hWndParent, CString const &strAttachmentFileName, CString const &strSubject=_T("")) { if (strAttachmentFileName.IsEmpty()) return false;
if (!hWndParent || !::IsWindow(hWndParent)) return false;
HINSTANCE hMAPI = ::LoadLibrary(_T("MAPI32.DLL")); if (!hMAPI) return false;
ULONG (PASCAL *SendMail)(ULONG, ULONG_PTR, MapiMessage*, FLAGS, ULONG); (FARPROC&)SendMail = GetProcAddress(hMAPI, "MAPISendMail");
if (!SendMail) return false;
char szFileName[_MAX_PATH]; char szPath[_MAX_PATH]; char szSubject[_MAX_PATH]; USES_CONVERSION;
::strcpy(szFileName, W2A(strAttachmentFileName.GetString())); ::strcpy(szPath, W2A(strAttachmentFileName.GetString())); ::strcpy(szSubject, W2A(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;
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
|
| Sign In·View Thread·PermaLink | 3.50/5 (2 votes) |
|
|
|
 |
|
|
yes this seems to handle the WCHAR characters.
but the problem for using USES_CONVERSION macros is they tend to change the non-english characters to the english ones. like "ı" to "i" and "ğ" to "g".
so i think i'll need a win32 api that directly supports WCHAR, which i think is not available currently.
|
| Sign In·View Thread·PermaLink | 1.00/5 (1 vote) |
|
|
|
 |
|
|
 |
|
|
 |
|
|
 |
|
|
A .NET solution is slightly more tricky since we need to marshal data between managed and unmanaged code for the Win32 MAPI calls. Here's a class file you can use. I will update the article to include this along with a sample C# test application.
using System; using System.Runtime.InteropServices; using System.IO;
namespace SendFileTo { class MAPI { private const int MAPI_LOGON_UI = 0x00000001; private const int MAPI_DIALOG = 0x00000008;
public static int SendMail(string strAttachmentFileName, string strSubject) { IntPtr session = new IntPtr(0); IntPtr winhandle = new IntPtr(0);
MapiMessage msg = new MapiMessage(); msg.subject = strSubject;
int sizeofMapiDesc = Marshal.SizeOf(typeof(MapiFileDesc)); IntPtr pMapiDesc = Marshal.AllocHGlobal(sizeofMapiDesc);
MapiFileDesc fileDesc = new MapiFileDesc(); fileDesc.position = -1; int ptr = (int)pMapiDesc;
string path = strAttachmentFileName; fileDesc.name = Path.GetFileName(path); fileDesc.path = path; Marshal.StructureToPtr(fileDesc, (IntPtr)ptr, false);
msg.files = pMapiDesc; msg.fileCount = 1;
return MAPISendMail(session, winhandle, msg, MAPI_LOGON_UI | MAPI_DIALOG, 0); }
[DllImport("MAPI32.DLL")] private static extern int MAPISendMail(IntPtr sess, IntPtr hwnd, MapiMessage message, int flg, int rsv);
}
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)] public class MapiMessage { public int reserved; public string subject; public string noteText; public string messageType; public string dateReceived; public string conversationID; public int flags; public IntPtr originator; public int recipCount; public IntPtr recips; public int fileCount; public IntPtr files; }
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)] public class MapiFileDesc { public int reserved; public int flags; public int position; public string path; public string name; public IntPtr type; } }
You can use it something like this.
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Text; using System.Windows.Forms; using SendFileTo;
namespace Test { public partial class Form1 : Form { public Form1() { InitializeComponent(); }
private void Form1_Load_1(object sender, EventArgs e) { MAPI.SendMail("c://documents//menu.doc", "Here's the lunch menu"); } } }
-- modified at 10:14 Tuesday 12th December, 2006
|
| Sign In·View Thread·PermaLink | 2.00/5 (1 vote) |
|
|
|
 |
|
|
 |
|
|
 |
|
|
Here's the code. The only code I'm a little concerned with is:
int ptr = (int)ptrMapiDesc + (iFilePtr * Marshal.SizeOf(typeof(MapiFileDesc)));
This may not be safe. But I tested it and it works.
Good luck, John
using System; using System.Runtime.InteropServices; using System.IO;
namespace SendFileTo { class MAPI { private const int MAPI_LOGON_UI = 0x00000001; private const int MAPI_DIALOG = 0x00000008;
public static int SendMail(string[] aAttachments, string sSubject, string sBody) { IntPtr ptrSession = new IntPtr(0); IntPtr ptrWinHandle = new IntPtr(0); int iFiles = 0; int iFilePtr = 0;
// Create a message MapiMessage msg = new MapiMessage(); msg.subject = sSubject; msg.noteText = sBody;
// Get the number of files iFiles = aAttachments.GetLength(0); MapiFileDesc[] fileDesc = new MapiFileDesc[iFiles];
int iSizeofMapiDesc = Marshal.SizeOf(typeof(MapiFileDesc)) * iFiles; IntPtr ptrMapiDesc = Marshal.AllocHGlobal(iSizeofMapiDesc); foreach(string sAttachment in aAttachments) { fileDesc[iFilePtr] = new MapiFileDesc(); fileDesc[iFilePtr].position = -1; int ptr = (int)ptrMapiDesc + (iFilePtr * Marshal.SizeOf(typeof(MapiFileDesc))); string sPath = sAttachment; fileDesc[iFilePtr].name = Path.GetFileName(sPath); fileDesc[iFilePtr].path = sPath; Marshal.StructureToPtr(fileDesc[iFilePtr], (IntPtr)ptr, false);
iFilePtr++; }
msg.files = ptrMapiDesc; msg.fileCount = iFiles;
return(MAPISendMail(ptrSession, ptrWinHandle, msg, MAPI_LOGON_UI | MAPI_DIALOG, 0)); }
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)] public class MapiMessage { public int reserved; public string subject; public string noteText; public string messageType; public string dateReceived; public string conversationID; public int flags; public IntPtr originator; public int recipCount; public IntPtr recips; public int fileCount; public IntPtr files; }
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)] public class MapiFileDesc { public int reserved; public int flags; public int position; public string path; public string name; public IntPtr type; } }
|
| Sign In·View Thread·PermaLink | |
|
|
|
 |
|
|
 |
|
|
General News Question Answer Joke Rant Admin
|