Как получить доступ к существующему экземпляру Microsoft Word, используя позднюю привязку - PullRequest
5 голосов
/ 05 февраля 2010

Я занимаюсь разработкой кода на C #, где я буду взаимодействовать с Microsoft Word. Я хочу иметь возможность повторно использовать существующий экземпляр или создать его в качестве альтернативы.

Имея в виду, что я хочу сделать все это, используя LATE BINDING ... можно с уверенностью сказать, что я выяснил, как заставить вещи работать при создании нового экземпляра ... я просто вызываю Activator.CreateInstance и т.д ... 1003 *

Проблема, с которой я сталкиваюсь, заключается в том, как повторно использовать существующий экземпляр, например, Word уже открыт, и я хочу использовать этот экземпляр.

Есть ли Activator.UseExistingInstance? или что-то подобное ??

Спасибо!

Ответы [ 2 ]

11 голосов
/ 05 февраля 2010

Вы ищете Marshal.GetActiveObject.

object word;
try
{
    word = System.Runtime.InteropServices.Marshal.GetActiveObject("Word.Application");
}
catch (COMException)
{
    Type type = Type.GetTypeFromProgID("Word.Application");
    word = System.Activator.CreateInstance(type);
}
7 голосов
/ 05 февраля 2010

Возможно, вы захотите взглянуть на функцию AccessibleObjectFromWindow api, определенную в Oleacc.dll. Эндрю Уайтчепел имеет несколько статей о том, как его использовать. На основании его статей я написал ответ на очень похожий вопрос (об Excel, а не Word), который вы можете найти здесь:

Как использовать позднюю привязку для получения экземпляра Excel?

Там вы найдете пример, как подключиться к уже запущенному экземпляру Excel, а затем автоматизировать этот экземпляр с использованием позднего связывания.

Обновление:

Вот краткий пример, адаптированный к Word:

using System;
using System.Reflection;
using System.Runtime.InteropServices;
using System.Text;

namespace WordLateBindingSample
{
    [ComImport, InterfaceType(ComInterfaceType.InterfaceIsIUnknown), Guid("00020400-0000-0000-C000-000000000046")]
    public interface IDispatch
    {
    }

    class Program
    {
        [DllImport("user32.dll", SetLastError = true)]
        static extern IntPtr FindWindow(string lpClassName, string lpWindowName);

        [DllImport("Oleacc.dll")]
        static extern int AccessibleObjectFromWindow(int hwnd, uint dwObjectID, byte[] riid, out IDispatch ptr);

        public delegate bool EnumChildCallback(int hwnd, ref int lParam);

        [DllImport("User32.dll")]
        public static extern bool EnumChildWindows(int hWndParent, EnumChildCallback lpEnumFunc, ref int lParam);

        [DllImport("User32.dll")]
        public static extern int GetClassName(int hWnd, StringBuilder lpClassName, int nMaxCount);

        public static bool EnumChildProc(int hwndChild, ref int lParam)
        {
            StringBuilder buf = new StringBuilder(128);
            GetClassName(hwndChild, buf, 128);
            if (buf.ToString() == "_WwG")
            {
                lParam = hwndChild;
                return false;
            }
            return true;
        }

        static void Main(string[] args)
        {
            // Use the window class name ("OpusApp") to retrieve a handle to Word's main window.
            // Alternatively you can get the window handle via the process id:
            // int hwnd = (int)Process.GetProcessById(wordPid).MainWindowHandle;
            //
            int hwnd = (int)FindWindow("OpusApp", null);

            if (hwnd != 0)
            {
                int hwndChild = 0;

                // Search the accessible child window (it has class name "_WwG") 
                // as described in http://msdn.microsoft.com/en-us/library/dd317978%28VS.85%29.aspx
                //
                EnumChildCallback cb = new EnumChildCallback(EnumChildProc);
                EnumChildWindows(hwnd, cb, ref hwndChild);

                if (hwndChild != 0)
                {
                    // We call AccessibleObjectFromWindow, passing the constant OBJID_NATIVEOM (defined in winuser.h) 
                    // and IID_IDispatch - we want an IDispatch pointer into the native object model.
                    //
                    const uint OBJID_NATIVEOM = 0xFFFFFFF0;
                    Guid IID_IDispatch = new Guid("{00020400-0000-0000-C000-000000000046}");
                    IDispatch ptr;

                    int hr = AccessibleObjectFromWindow(hwndChild, OBJID_NATIVEOM, IID_IDispatch.ToByteArray(), out ptr);

                    if (hr >= 0)
                    {
                        object wordApp = ptr.GetType().InvokeMember("Application", BindingFlags.GetProperty, null, ptr, null);

                        object version = wordApp.GetType().InvokeMember("Version", BindingFlags.GetField | BindingFlags.InvokeMethod | BindingFlags.GetProperty, null, wordApp, null);
                        Console.WriteLine(string.Format("Word version is: {0}", version));
                    }
                }
            }
        }
    }
}
...