Lotus automation classes provide a rich toolset for working with Lotus Notes client. If using from VBScript the syntax will be very similar to LotusScript. Even LotusScript documentation can be used to reference available Lotus classes. It can be found in Domino Designer Help or IBM site online.
In this article we will talk about Lotus Notes automation classes such as "Notes.NotesSession" and "Notes.NotesUIWorkspace". These classes were around in Lotus for a long time, since version 5.x, and they are still present in Lotus Notes 8.5, even in Eclipse based standard edition.
There are enough NotesUIWorkspace and NotesSession usage samples available on the Net written in Visual Basic 6.0 and VBA for Microsoft Access and Excel. This article should fill the gap of guides to access Lotus automation classes in C++ and ATL.
The most popular sample in VB is sending an email. I don't have Visual Basic 6.0 installed so I've adopted it to VBScript. Here is a code:
' VB Script Document ' http://lotusdesk.blogspot.com/ option explicit Dim objSession, objMailDB, objMailDoc Set objSession = CreateObject("Notes.NotesSession") Set objMailDB = objSession.GetDatabase("", "") 'Open database if it's not ready If Not objMailDB.IsOpen Then objMailDB.OpenMail End If Set objMailDoc = objMailDB.CreateDocument Dim strRecipient strRecipient = "notescoder@gmail.com" objMailDoc.Form = "Memo" objMailDoc.SendTo = strRecipient objMailDoc.Subject = "Test Subject Line" objMailDoc.Body = "Some Text" ' Let's set it to True to get a copy in Sent folder. objMailDoc.SaveMessageOnSend = True objMailDoc.PostedDate = Now() objMailDoc.Send 0, strRecipient
Let's do the same in C++. Since Notes.NotesSession, Notes.NotesUIWorkspace and associated interfaces support late binding only we need to use IDispatch method pair GetIDOfName() and Invoke() to access their properties and methods. Luckily ATL provides a class CComDispatchDriver which can handle GetIDOfName() and Invoke() pair for us.
CComDispatchDriver ptrSession; HRESULT hr = ptrSession.CoCreateInstance(OLESTR("Notes.NotesSession")); OleRun(ptrSession); _variant_t serverName(OLESTR("")), fileName(OLESTR("")); _variant_t varMailDb; hr = ptrSession.Invoke2(OLESTR("GetDatabase"), &serverName, &fileName, &varMailDb); if(FAILED(hr)) { return FALSE; } return TRUE;
In this sample we have used ProgId to create NotesSession object. We can declare and use CLSID instead:
// {29131401-2EED-1069-BF5D-00DD011186B7} static const GUID CLSID_NotesSession = { 0x29131401, 0x2EED, 0x1069, { 0xBF, 0x5D, 0x00, 0xDD, 0x01, 0x11, 0x86, 0xB7 } };
This class identifier didn't change since Lotus Notes 6.0, maybe for earlier version it's also the same. So it's safe to use it. Here is another CLSID for NotesUIWorkspace:
// {29131502-2EED-1069-BF5D-00DD011186B7} static const GUID CLSID_NotesUiWorkspace = { 0x29131502, 0x2EED, 0x1069, { 0xBF, 0x5D, 0x00, 0xDD, 0x01, 0x11, 0x86, 0xB7 } };
Let's put object creation and OleRun call into a separate function so we can reuse it later for creation of NotesUIWorkspace class:
CComDispatchDriver CreateObject(const CLSID &clsid) { CComDispatchDriver ptrDispatch; HRESULT hr = ptrDispatch.CoCreateInstance(clsid); if(SUCCEEDED(hr)) OleRun(ptrDispatch); return ptrDispatch; }
The final code for sending an email will look like this:
BOOL SendMemo() { CComDispatchDriver ptrSession = CreateObject(CLSID_NotesSession); _variant_t serverName(OLESTR("")), fileName(OLESTR("")); _variant_t varMailDb; HRESULT hr = ptrSession.Invoke2(OLESTR("GetDatabase"), &serverName, &fileName, &varMailDb); if(FAILED(hr)) { return FALSE; } CComDispatchDriver ptrMailDb = (IUnknown*)varMailDb; _variant_t varIsOpen; ptrMailDb.GetPropertyByName(OLESTR("IsOpen"), &varIsOpen); if(!varIsOpen) { ptrMailDb.Invoke0(OLESTR("OpenMail")); } _variant_t varDoc; hr = ptrMailDb.Invoke0(OLESTR("CreateDocument"), &varDoc); if(FAILED(hr)) { return FALSE; } CComDispatchDriver ptrDoc = (IUnknown*)varDoc; _variant_t varSubject(OLESTR("Test subject")); ptrDoc.PutPropertyByName(OLESTR("Subject"), &varSubject); _variant_t varRecipient(OLESTR("notescoder@gmail.com")); ptrDoc.PutPropertyByName(OLESTR("SendTo"), &varRecipient); _variant_t varBodyText(OLESTR("Body text")); ptrDoc.PutPropertyByName(OLESTR("Body"), &varBodyText); _variant_t varTrue(-1l, VT_BOOL); ptrDoc.PutPropertyByName(OLESTR("SaveMessageOnSend"), &varTrue); _variant_t varNow(COleDateTime::GetCurrentTime()); ptrDoc.PutPropertyByName(OLESTR("PostedDate"), &varNow); _variant_t varZero(0); hr = ptrDoc.Invoke2(OLESTR("Send"), &varZero, &varRecipient); return SUCCEEDED(hr); }
Try to create a simple console application to test the code above. Don't forget to include ATL classes:
#include <comdef.h> #include <atlbase.h> #include <atlcom.h> #include <atlctl.h> #include <atlcomtime.h> using namespace ATL;
In the next article we will do the same with visual effects in Lotus Notes client.