Database Design 3Hands-on Excercises

Hands-on 8: Automation using Word

To work with Office documents (such as Word or Excel) using VBA code you need to use Automation. This allows you to work with the object models of the other applications.

In this exercise you will explore automation with Word.

Set a reference to Word’s Object Library

The first stage in writing Automation code is to set a reference to the object model of the application you want to work with.

To set the Word object library reference:

Open the VBA Editor and click Tools > References... to display the References dialog.

Make sure a reference is set to the Word 11 object library. You might need to scroll the list to find this. If it is not checked already, click the check box.

Fig.12.1: Setting reference to Word’s Object Library

View the Word object model using the Object Browser

Open the VBA Editor and press F2 to open the Object Browser. Select Word from the libraries list.

The Classes pane lists the objects, collections and enumerations that are available. Selecting an item from this list displays the associated properties, methods and events in the Members pane. When you select one of these, the syntax will be displayed in the status bar at the bottom of the screen.

Create a new standard module

You will be writing several functions to automate Word from Access. It would be useful if these were available to any forms in your application. You will store all of these together in a standard module.

To make the same Word application available to all the functions in the module we will declare it as a Private variable in the General Declarations section at the start of the module. The Word application is an Object variable.

Option Compare Database

Option Explicit

Private objWord As Object

Note: The two Option lines should be inserted when you open a new module. To ensure that the Option Explicit statement (that forces you to declare variables before using them) is always inserted click Tools > Options and check the require variable declarations check box in the Editor tab.

Create a Word instance

You will first write a function that will open word. The application needs to be opened before you can work with it. Any function that needs to use Word can call this function.

Public Function OpenWord()

On Error GoTo Err_OpenWord

Set objWord = GetObject(, "Word.Application")

objWord.Application.Visible = True

Exit_OpenWord:

Exit Function

Err_OpenWord:

If Err = 429 Then 'word is not running

Set objWord = CreateObject("Word.Application")

Resume Next

Else

MsgBox "Error no. " & Err.Number

Resume Exit_OpenWord

End If

End Function

This code initially uses the GetObject function to open Word. GetObject opens an instance of Word if it is already running. It is best to use this first as otherwise you will have two versions of Word open. This is not necessarily a problem with the large amounts of memory available today but it is better practice to use the existing open version if possible.

If GetObject is called when Word is not running it generates error 429. The error handling code traps this error and calls the CreateObject function to open a new version of Word. If there is an error other than this, an error message is displayed to the user and the function exits.

Use F8 to step through the code both with a running version of Word and with Word closed. You will see that there is a different path through the function in each case.

Enter this code into your module and test it (press F5 with the cursor somewhere inside the function to run the code directly from the editor. Word should open with no document displayed.

Create a new blank document in Word

Public Function OpenNewWordDoc()

' Uses OpenWord function to open Word

' Opens new blank document

On Error GoTo Err_OpenNewWordDoc

Dim docWord As Word.Document

OpenWord

Set docWord = objWord.Documents.Add

Exit_OpenNewWordDoc:

Exit Function

Err_OpenNewWordDoc:

MsgBox "Error no. " & Err.Number

Resume Exit_OpenNewWordDoc

End Function

Close the document and Word

Of course, you will actually want to use the document you have opened in some way but before we look at how you might do that we will write the code to close the document and exit Word.

Adding the two lines below to your code will close the document and release the object reference.

docWord.Close

Set docWord = Nothing

To close the Word application you need to use the Quit method:

objWord.Quit

The complete function up to this point with the new code highlighted is shown below:

Public Function OpenNewWordDoc()

' Uses OpenWord function to open Word

' Opens new blank document

On Error GoTo Err_OpenNewWordDoc

Dim docWord As Word.Document

OpenWord

Set docWord = objWord.Documents.Add

docWord.Close

Set docWord = Nothing

objWord.Quit

Set objWord = Nothing

Exit_OpenNewWordDoc:

Exit Function

Err_OpenNewWordDoc:

MsgBox "Error no. " & Err.Number

Resume Exit_OpenNewWordDoc

End Function

Again, use F8 to step through the function to see how it works. You should see that the code starts Word, opens a new blank document and then closes it and exits Word.

Although Word is successfully closed there is one problem with the approach we have taken here. If the user already had a version of Word running our code used that instance rather than opening a new one. If the user had unsaved documents open in Word they have now been closed which might have resulted in lost work. To avoid this happening we need to add code so that the function will only close Word if it also opened it. To do this we will use a flag – a Boolean variable that will be set to true if the function has opened Word and False otherwise.

As we are using the OpenWord function to open Word, we need to set the flag in this function and then test its value in any other functions that are using Word. As the flag needs to be available to all the functions in the module we will declare it as a private variable in the declarations section at the start of the module in the same way as we declared the Word variable earlier.

Option Compare Database

Option Explicit

Private objWord As Object

Private blOpenedWord As Boolean

Then add the following code to the error handling section of the OpenWord function:

Public Function OpenWord()

...

Err_OpenWord:

If Err = 429 Then 'word is not running

Set objWord = CreateObject("Word.Application")

blOpenedWord = True

Resume Next

...

Now when the function opens a new copy of Word, the flag will be set to true. The code that closes Word can now check the value of this flag and only close Word if it was opened by our function. Add this code to the OpenNewWordDoc function to test the value of the flag before closing Word.

Public Function OpenNewWordDoc()

...

OpenWord

Set docWord = objWord.Documents.Add

docWord.Close

Set docWord = Nothing

If blOpenedWord = True Then

objWord.Quit

End If

Set objWord = Nothing

...

Again test the function by stepping through it both with an open and closed version of Word. You should notice that if Word is already open when you run the function the Quit method is not executed.

Inserting text

Now add some code to the OpenNewWordDoc function to insert some text into the document. To do this we will use the TypeText method of the Selection object. The Selection object is the text selected (or the position of the insertion point) in the currently active Word document. Once we have inserted a line of text we will use the TypeParagraph method of the Selection object to insert a new line character.

Insert this code into your OpenNewWordDoc function:

Set docWord = objWord.Documents.Add

' insert text into the Word document

objWord.Selection.TypeText "This document was created automatically by Access"

objWord.Selection.TypeParagraph

objWord.Selection.TypeText "It will now be closed and saved"

objWord.Selection.TypeParagraph
objWord.Activate

This code will insert two lines of text into your Word document. The last statement uses the Activate method to display Word on your screen.

Save and Close the document

Not surprisingly, we will use the Save and Close methods of the document object to do this. The Save method will open the Save As box if the document hasn’t been saved before so that the user can select a folder and enter a file name for the document. insert the following lines into your function:

docWord.Save

docWord.Close

Set docWord = Nothing

The final line of this code releases the memory used by the docWord variable.

Test the function. It should open a new Word document, insert the specified text, and then open the Save As dialog to allow you to save the file. Provided you do save the file everything is ok. However, if you click the Cancel button you generate error 4198 – Command Cancelled. You need to add appropriate error handling code to the function to deal with this. It is sufficient to add general purpose error handling unless you want to deal with error 4198 specifically.

The completed code for this function is given below:

Public Function OpenNewWordDoc()

' Uses OpenWord function to open Word

' Opens new blank document

On Error GoTo Err_OpenNewWordDoc

Dim docWord As Word.Document

OpenWord

Set docWord = objWord.Documents.Add

' objWord.Activate

' insert text into the Word document

objWord.Selection.TypeText "This document was created automatically by Access"

objWord.Selection.TypeParagraph

objWord.Selection.TypeText "It will now be closed and saved"

objWord.Selection.TypeParagraph

objWord.Activate

docWord.Save

docWord.Close

Set docWord = Nothing

If blOpenedWord = True Then

objWord.Quit

End If

Set objWord = Nothing

Exit_OpenNewWordDoc:

Exit Function

Err_OpenNewWordDoc:

MsgBox "Error no. " & Err.Number

Resume Exit_OpenNewWordDoc

End Function

Open an existing document

To open an exisitng document you use the Open method of the Documents object. The first argument to the method is the name of the file you wish to open. You need to use the full pathname for the file. The following code will open a file named empMemo.doc, stored in the D3 folder of the I: drive.

appWord.Documents.Open("I:\D3\empMemo.doc")

Note:

You can use the Path property of the current project to get the path for the folder where your database is located. Test this out by typing print CurrentProject.Path into the Immediate Window. Press Enter and you will see the drive and folder for your current database displayed. This method is used in the following code to open a file named empMemo.doc stored in the same folder as the database.

Add this function to your modWordUtilities module.

You might want to consider passing the file name as an argument to the function to make it more general purpose.

Public Function OpenWordDoc()

On Error GoTo Err_OpenWordDoc

Dim fname As String

Dim docWord As Word.Document

OpenWord

' file name needs to include path

' if file to open is in same folder as database can use Path attribute of current project

fname = CurrentProject.Path & "\empMemo.doc"

Set docWord = objWord.Documents.Open(fname)

Exit_OpenWordDoc:

Exit Function

Err_OpenWordDoc:

MsgBox "Error no. " & Err.Number

Resume Exit_OpenWordDoc

End Function

Create a Word Memo for all Yum employees

You will now use Word Automation to create a memo document to be sent to all Yum employees. Before you can do that you will need to create a query to list the employee names.

Create a query to list the employee names

To provide the list of employee names to be used by Word to create the memo we will use a query that will combine the employee first name and surname in a single field. Create this query as shown below and save it with a meaningful name.

Fig. 12.2: Employee name query to be used as data source for memo

The Word document

The Word document you will use for this exercise has already been created for you. Download it from WebCT and save it in the same folder as your database (this will simplify the code to open the file as you can then use the CurrentProject.Path as discussed above).

We are going to write text to specific locations in this document, identified by bookmarks. Two bookmarks have been set – one for the position of the date and the other for the position of the employee names. You can see the Bookmarks that are set in a Word document by clicking Bookmark from the Insert Menu.

If you click the Go To button the insertion point will be moved to the specified bookmark.

Insert the date and list of names

The following code will insert today’s date and the list of employee names at the appropriate points in the memo document. You need to make sure that you have created the employee names query and saved the memo document to the same folder as your database before running the code.

Public Function CreateWordMemo()

' Open memo in Word and insert text

On Error GoTo CreateWordMemo_Error

Dim dbs As Database

Dim rstEmployees As Recordset

Dim appWord As Word.Application

Dim docWord As Word.Document

Dim fname As String

Dim strDate As String

strDate = Date 'assign date to string variable so we can print it

OpenWord ' use function to open Word

'Open recordset based on employee name query

Set dbs = CurrentDb()

Set rstEmployees = dbs.OpenRecordset("qryEmpFullName")

' open the memo document

' assume memo document in same folder as database

fname = CurrentProject.Path & "\empMemo.doc"

Set docWord = objWord.Documents.Open(fname)

objWord.Visible = True

With objWord

.Selection.Goto wdGoToBookmark, Name:="DateToday" 'move to date bookmark

.Selection.TypeText " " & strDate 'insert date

.Selection.Goto wdGoToBookmark, Name:="MemoToLine" 'move to memo line bookmark

End With

'Loop through recordset returned by query, inserting name of each employee

Do Until rstEmployees.EOF

objWord.Selection.TypeText rstEmployees!Employee & ", "

rstEmployees.MoveNext

Loop

'reset variables to nothing and free up memory for other processes

Set rstEmployees = Nothing

Set dbs = Nothing

Set docWord = Nothing

Set objWord = Nothing

Exit_CreateWordMemo:

Exit Function

CreateWordMemo_Error: 'error trapping routine

MsgBox Err.Description

Resume Exit_CreateWordMemo

End Function

The comments embedded in the code should help you to understand how it works. You can test it from the VBA editor by pressing F8 with the insertion point somewhere inside the function.

Notice that although the code frees the variables, it does not close Word. This means that the document is open in Word for the user to save or edit as they choose. If you want to save the document it is best to use Save As or you will write duplicate text to the same document the next time you run the function.

Once you are happy that the function works as it should, create a utilities form and add a button to allow the user to create an employee memo.

Enhancements

At the moment only the date and list of employees are entered automatically by Access. The user then needs to enter the memo text directly into Word. One simple improvement that could be made is to add another Bookmark to the Word document at the position where the memo text needs to be entered. Then add a line of code to your function so that the document is ready for the user to enter text.

You could also consider passing the recordset as a parameter so that the user could select who the memo would be sent to. To do this you would pass a string containing the name of the query to be used.

A final modification would be to add a text box to your form to allow the user to enter the memo content. This would then be passed as a string to the function and printed to the appropriate bookmark by your code.

d3_ho8.docPage 1 of 9