Walkthrough: Office Programming in C#

C# offers features that improve Microsoft Office programming. Helpful C# features include named and optional arguments and return values of type dynamic. In COM programming, you can omit the ref keyword and gain access to indexed properties.

Both languages enable embedding of type information, which allows deployment of assemblies that interact with COM components without deploying primary interop assemblies (PIAs) to the user's computer. For more information, see Walkthrough: Embedding Types from Managed Assemblies.

This walkthrough demonstrates these features in the context of Office programming, but many of these features are also useful in general programming. In the walkthrough, you use an Excel Add-in application to create an Excel workbook. Next, you create a Word document that contains a link to the workbook. Finally, you see how to enable and disable the PIA dependency.

Important

VSTO (Visual Studio Tools for Office) relies on the .NET Framework. COM add-ins can also be written with the .NET Framework. Office Add-ins cannot be created with .NET Core and .NET 5+, the latest versions of .NET. This is because .NET Core/.NET 5+ cannot work together with .NET Framework in the same process and may lead to add-in load failures. You can continue to use .NET Framework to write VSTO and COM add-ins for Office. Microsoft will not be updating VSTO or the COM add-in platform to use .NET Core or .NET 5+. You can take advantage of .NET Core and .NET 5+, including ASP.NET Core, to create the server side of Office Web Add-ins.

Prerequisites

You must have Microsoft Office Excel and Microsoft Office Word installed on your computer to complete this walkthrough.

Note

Your computer might show different names or locations for some of the Visual Studio user interface elements in the following instructions. The Visual Studio edition that you have and the settings that you use determine these elements. For more information, see Personalizing the IDE.

Set up an Excel Add-in application

  1. Start Visual Studio.
  2. On the File menu, point to New, and then select Project.
  3. In the Installed Templates pane, expand C#, expand Office, and then select the version year of the Office product.
  4. In the Templates pane, select Excel <version> Add-in.
  5. Look at the top of the Templates pane to make sure that .NET Framework 4, or a later version, appears in the Target Framework box.
  6. Type a name for your project in the Name box, if you want to.
  7. Select OK.
  8. The new project appears in Solution Explorer.

Add references

  1. In Solution Explorer, right-click your project's name and then select Add Reference. The Add Reference dialog box appears.
  2. On the Assemblies tab, select Microsoft.Office.Interop.Excel, version <version>.0.0.0 (for a key to the Office product version numbers, see Microsoft Versions), in the Component Name list, and then hold down the CTRL key and select Microsoft.Office.Interop.Word, version <version>.0.0.0. If you don't see the assemblies, you may need to install them (see How to: Install Office Primary Interop Assemblies).
  3. Select OK.

Add necessary Imports statements or using directives

In Solution Explorer, right-click the ThisAddIn.cs file and then select View Code. Add the following using directives (C#) to the top of the code file if they aren't already present.

using System.Collections.Generic;
using Excel = Microsoft.Office.Interop.Excel;
using Word = Microsoft.Office.Interop.Word;

Create a list of bank accounts

In Solution Explorer, right-click your project's name, select Add, and then select Class. Name the class Account.cs. Select Add. Replace the definition of the Account class with the following code. The class definitions use automatically implemented properties.

class Account
{
    public int ID { get; set; }
    public double Balance { get; set; }
}

To create a bankAccounts list that contains two accounts, add the following code to the ThisAddIn_Startup method in ThisAddIn.cs. The list declarations use collection initializers.

var bankAccounts = new List<Account>
{
    new Account
    {
        ID = 345,
        Balance = 541.27
    },
    new Account
    {
        ID = 123,
        Balance = -127.44
    }
};

Export data to Excel

In the same file, add the following method to the ThisAddIn class. The method sets up an Excel workbook and exports data to it.

void DisplayInExcel(IEnumerable<Account> accounts,
           Action<Account, Excel.Range> DisplayFunc)
{
    var excelApp = this.Application;
    // Add a new Excel workbook.
    excelApp.Workbooks.Add();
    excelApp.Visible = true;
    excelApp.Range["A1"].Value = "ID";
    excelApp.Range["B1"].Value = "Balance";
    excelApp.Range["A2"].Select();

    foreach (var ac in accounts)
    {
        DisplayFunc(ac, excelApp.ActiveCell);
        excelApp.ActiveCell.Offset[1, 0].Select();
    }
    // Copy the results to the Clipboard.
    excelApp.Range["A1:B3"].Copy();
}
  • Method Add has an optional parameter for specifying a particular template. Optional parameters enable you to omit the argument for that parameter if you want to use the parameter's default value. Because the previous example has no arguments, Add uses the default template and creates a new workbook. The equivalent statement in earlier versions of C# requires a placeholder argument: excelApp.Workbooks.Add(Type.Missing). For more information, see Named and Optional Arguments.
  • The Range and Offset properties of the Range object use the indexed properties feature. This feature enables you to consume these properties from COM types by using the following typical C# syntax. Indexed properties also enable you to use the Value property of the Range object, eliminating the need to use the Value2 property. The Value property is indexed, but the index is optional. Optional arguments and indexed properties work together in the following example.
// Visual C# 2010 provides indexed properties for COM programming.
excelApp.Range["A1"].Value = "ID";
excelApp.ActiveCell.Offset[1, 0].Select();

You can't create indexed properties of your own. The feature only supports consumption of existing indexed properties.

Add the following code at the end of DisplayInExcel to adjust the column widths to fit the content.

excelApp.Columns[1].AutoFit();
excelApp.Columns[2].AutoFit();

These additions demonstrate another feature in C#: treating Object values returned from COM hosts such as Office as if they have type dynamic. COM objects are treated as dynamic automatically when Embed Interop Types has its default value, True, or, equivalently, when you reference the assembly with the EmbedInteropTypes compiler option. For more information about embedding interop types, see procedures "To find the PIA reference" and "To restore the PIA dependency" later in this article. For more information about dynamic, see dynamic or Using Type dynamic.

Invoke DisplayInExcel

Add the following code at the end of the ThisAddIn_StartUp method. The call to DisplayInExcel contains two arguments. The first argument is the name of the list of accounts processed. The second argument is a multiline lambda expression defining how to process the data. The ID and balance values for each account are displayed in adjacent cells, and the row is displayed in red if the balance is less than zero. For more information, see Lambda Expressions.

DisplayInExcel(bankAccounts, (account, cell) =>
// This multiline lambda expression sets custom processing rules
// for the bankAccounts.
{
    cell.Value = account.ID;
    cell.Offset[0, 1].Value = account.Balance;
    if (account.Balance < 0)
    {
        cell.Interior.Color = 255;
        cell.Offset[0, 1].Interior.Color = 255;
    }
});

To run the program, press F5. An Excel worksheet appears that contains the data from the accounts.

Add a Word document

Add the following code at the end of the ThisAddIn_StartUp method to create a Word document that contains a link to the Excel workbook.

var wordApp = new Word.Application();
wordApp.Visible = true;
wordApp.Documents.Add();
wordApp.Selection.PasteSpecial(Link: true, DisplayAsIcon: true);

This code demonstrates several of the features in C#: the ability to omit the ref keyword in COM programming, named arguments, and optional arguments.The PasteSpecial method has seven parameters, all of which are optional reference parameters. Named and optional arguments enable you to designate the parameters you want to access by name and to send arguments to only those parameters. In this example, arguments indicate creating a link to the workbook on the Clipboard (parameter Link) and displaying that the link in the Word document as an icon (parameter DisplayAsIcon). C# also enables you to omit the ref keyword for these arguments.

Run the application

Press F5 to run the application. Excel starts and displays a table that contains the information from the two accounts in bankAccounts. Then a Word document appears that contains a link to the Excel table.

Clean up the completed project

In Visual Studio, select Clean Solution on the Build menu. Otherwise, the add-in runs every time that you open Excel on your computer.

Find the PIA reference

  1. Run the application again, but don't select Clean Solution.
  2. Select the Start. Locate Microsoft Visual Studio <version> and open a developer command prompt.
  3. Type ildasm in the Developer Command Prompt for Visual Studio window, and then press Enter. The IL DASM window appears.
  4. On the File menu in the IL DASM window, select File > Open. Double-click Visual Studio <version>, and then double-click Projects. Open the folder for your project, and look in the bin/Debug folder for your project name.dll. Double-click your project name.dll. A new window displays your project's attributes, in addition to references to other modules and assemblies. The assembly includes the namespaces Microsoft.Office.Interop.Excel and Microsoft.Office.Interop.Word. By default in Visual Studio, the compiler imports the types you need from a referenced PIA into your assembly. For more information, see How to: View Assembly Contents.
  5. Double-click the MANIFEST icon. A window appears that contains a list of assemblies that contain items referenced by the project. Microsoft.Office.Interop.Excel and Microsoft.Office.Interop.Word aren't in the list. Because you imported the types your project needs into your assembly, you aren't required to install references to a PIA. Importing the types into your assembly makes deployment easier. The PIAs don't have to be present on the user's computer. An application doesn't require deployment of a specific version of a PIA. Applications can work with multiple versions of Office, provided that the necessary APIs exist in all versions. Because deployment of PIAs is no longer necessary, you can create an application in advanced scenarios that works with multiple versions of Office, including earlier versions. Your code can't use any APIs that aren't available in the version of Office you're working with. It isn't always clear whether a particular API was available in an earlier version. Working with earlier versions of Office isn't recommended.
  6. Close the manifest window and the assembly window.

Restore the PIA dependency

  1. In Solution Explorer, select the Show All Files button. Expand the References folder and select Microsoft.Office.Interop.Excel. Press F4 to display the Properties window.
  2. In the Properties window, change the Embed Interop Types property from True to False.
  3. Repeat steps 1 and 2 in this procedure for Microsoft.Office.Interop.Word.
  4. In C#, comment out the two calls to Autofit at the end of the DisplayInExcel method.
  5. Press F5 to verify that the project still runs correctly.
  6. Repeat steps 1-3 from the previous procedure to open the assembly window. Notice that Microsoft.Office.Interop.Word and Microsoft.Office.Interop.Excel are no longer in the list of embedded assemblies.
  7. Double-click the MANIFEST icon and scroll through the list of referenced assemblies. Both Microsoft.Office.Interop.Word and Microsoft.Office.Interop.Excel are in the list. Because the application references the Excel and Word PIAs, and the Embed Interop Types property is False, both assemblies must exist on the end user's computer.
  8. In Visual Studio, select Clean Solution on the Build menu to clean up the completed project.

See also