简体   繁体   English

通过C#运行Excel宏:从另一个工作簿运行宏?

[英]Running an Excel Macro via C#: Run a macro from one workbook on another?

I am looking to run a macro, let's call it Macro01 from WorkSheet01 on WorkSheet02. 我想运行一个宏,让我们从WorkSheet02上的WorkSheet01中调用它。

Using Microsoft.Office.Interop.Excel Namespace I have opened a WorkSheet01. 使用Microsoft.Office.Interop.Excel命名空间我打开了一个WorkSheet01。

public void Main_CodedStep()
    {
        // Object for missing (or optional) arguments.
        object oMissing = System.Reflection.Missing.Value;

        // Create an instance of Microsoft Excel
        Excel.ApplicationClass oExcel = new Excel.ApplicationClass();

        // Make it visible
        oExcel.Visible = true;

        // Open Worksheet01.xlsm
        Excel.Workbooks oBooks = oExcel.Workbooks;
        Excel._Workbook oBook = null;
        oBook = oBooks.Open("C:\\Users\\Admin\\Documents\\Worksheet01.xlsm", oMissing, oMissing,
            oMissing, oMissing, oMissing, oMissing, oMissing, oMissing, 
            oMissing, oMissing, oMissing, oMissing, oMissing, oMissing);
    }

I then use an automated script to pull a report. 然后我使用自动脚本来提取报告。 This report is opened via IE's download prompt and not the Interop. 此报告是通过IE的下载提示而不是Interop打开的。

The problem comes when I try to run the macro via C# (I made another new Excel.ApplicationClass(); only so it compiled, I believe this is one of my missteps.) 当我尝试通过C#运行宏时出现了问题(我创建了另一个新的Excel.ApplicationClass();只有这样才编译,我相信这是我的失误之一。)

public void FirstMacro_CodedStep()
    {
        // Create an instance of Microsoft Excel
        Excel.ApplicationClass oExcel = new Excel.ApplicationClass();
        Console.WriteLine("ApplicationClass: " + oExcel);

        // Run the macro, "First_Macro"
        RunMacro(oExcel, new Object[]{"Worksheet01.xlsm!First_Macro"});

        //Garbage collection
        GC.Collect();
    }

    private void RunMacro(object oApp, object[] oRunArgs)
    {
        oApp.GetType().InvokeMember("Run", System.Reflection.BindingFlags.Default | System.Reflection.BindingFlags.InvokeMethod, null, oApp, oRunArgs);
    }

When this method runs it runs the macro from Worksheet01 on Worksheet01 instead of Worksheet02. 当此方法运行时,它从Worksheet01上的Worksheet01而不是Worksheet02运行宏。 Also it was looking for the worksheet in My Documents so I moved it over to see what would happen. 它也在寻找我的文档中的工作表,所以我把它移到了看看会发生什么。

Recap: 概括:

  1. Open Worksheet01 打开Worksheet01
  2. Via scripting get and open a report (Worksheet02) from MSIE 通过脚本获取并打开来自MSIE的报告(Worksheet02)
  3. Run Macro01 from Worksheet01 on Worksheet02 从Worksheet02上的Worksheet01运行Macro01

Resources: 资源:

http://support.microsoft.com/kb/306683 http://support.microsoft.com/kb/306683

http://msdn.microsoft.com/en-us/library/microsoft.office.interop.excel.aspx http://msdn.microsoft.com/en-us/library/microsoft.office.interop.excel.aspx

For those who would like to try it add this to your using directives: 对于那些想要尝试的人,请将其添加到using指令中:

using System.Reflection;
using Microsoft.Office.Core; //Added to Project Settings' References from C:\Program Files (x86)\Microsoft Visual Studio 10.0\Visual Studio Tools for Office\PIA\Office14 - "office"
using Excel = Microsoft.Office.Interop.Excel; //Added to Project Settings' References from C:\Program Files (x86)\Microsoft Visual Studio 10.0\Visual Studio Tools for Office\PIA\Office14 - "Microsoft.Office.Interop.Excel"

I found a solution that I'd like to share. 我找到了一个我想分享的解决方案。 First, I removed the bit where I opened Worksheet01. 首先,我删除了我打开Worksheet01的位。 I then had my automated script save the .CSV to My Documents. 然后,我有自动脚本将.CSV保存到我的文档。 I then used the code I had to open Worksheet01 to open the downloaded file. 然后我使用我必须打开Worksheet01的代码来打开下载的文件。 The key thing at this point is that Worksheet01 is in the Documents folder with Worksheet02. 此时的关键是Worksheet01位于Documents文件夹的Documents文件夹中。 Lastly I used the code to run the macro from Worksheet01, which runs on Worksheet02. 最后,我使用代码从Worksheet01运行宏,它在Worksheet02上运行。

    public void WebTest_CodedStep()
    {
        // Object for missing (or optional) arguments.
        object oMissing = System.Reflection.Missing.Value;

        // Create an instance of Microsoft Excel
        Excel.ApplicationClass oExcel = new Excel.ApplicationClass();

        // Make it visible
        oExcel.Visible = true;

        // Define Workbooks
        Excel.Workbooks oBooks = oExcel.Workbooks;
        Excel._Workbook oBook = null;

        // Get the file path
        string path = System.Environment.GetFolderPath(Environment.SpecialFolder.MyDocuments);
        path = path + "\\Worksheet02.csv";

        //Open the file, using the 'path' variable
        oBook = oBooks.Open(path, oMissing, oMissing, oMissing, oMissing, oMissing, oMissing, oMissing, oMissing,  oMissing, oMissing, oMissing, oMissing, oMissing, oMissing);

        // Run the macro, "First_Macro"
        RunMacro(oExcel, new Object[]{"Worksheet01.xlsm!First_Macro"});

        // Quit Excel and clean up.
        oBook.Close(false, oMissing, oMissing);
        System.Runtime.InteropServices.Marshal.ReleaseComObject (oBook);
        oBook = null;
        System.Runtime.InteropServices.Marshal.ReleaseComObject (oBooks);
        oBooks = null;
        oExcel.Quit();
        System.Runtime.InteropServices.Marshal.ReleaseComObject (oExcel);
        oExcel = null;

        //Garbage collection
        GC.Collect();
    }

    private void RunMacro(object oApp, object[] oRunArgs)
    {
        oApp.GetType().InvokeMember("Run", System.Reflection.BindingFlags.Default | System.Reflection.BindingFlags.InvokeMethod, null, oApp, oRunArgs);
    }

I ran this C# VSTO code to invoke a VBA Macro, this is the syntax I use: 我运行这个C#VSTO代码来调用VBA宏,这是我使用的语法:

this.Application.Run("mymacro");

Edit: 编辑:

Macros are Workbook wide, perhaps you need to make Sheet2 the active worksheet before running the macro, eg: 宏是工作簿范围的,也许您需要在运行宏之前使Sheet2成为活动工作表,例如:

foreach (Worksheet worksheet in workbook.Sheets.ComLinq<Worksheet>())
{
    if (worksheet.Name == "Sheet2") worksheet.Activate();
}

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM