从wpf应用程序打开并且Outlook正在运行时,如何获取Outlook选择名称对话框以进行聚焦

时间:2012-12-07 07:33:49

标签: c# wpf outlook vsto

我正在创建一个发送Outlook预约的wpf应用程序。在此应用程序中,我打开Outlook选择名称对话框以选择约会的收件人。以下是我的代码:

Outlook.SelectNamesDialog selectNameDialog =
           outlookApp.Session.GetSelectNamesDialog();
        selectNameDialog.SetDefaultDisplayMode(
            Outlook.OlDefaultSelectNamesDisplayMode.olDefaultMeeting);

        foreach (var recipient in recipientsList)
        {
            if (string.IsNullOrEmpty(recipient))
                continue;
            Outlook.Recipient confRoom =
                selectNameDialog.Recipients.Add(recipient);
            // Explicitly specify Recipient.Type.
            confRoom.Type = (int)Outlook.OlMeetingRecipientType.olRequired;
        }

        selectNameDialog.Recipients.ResolveAll();

        selectNameDialog.Display();

我的问题是,当我打开选择名称对话框时,如果outlook未运行,它可以正常工作。但是如果outlook正在运行并且我从我的应用程序单击打开此对话框,它将在我的应用程序后面和Outlook窗口中打开。即使outlook正在运行,我也需要在我的应用程序之上显示它。任何帮助将非常感谢将此对话框放在所有人面前。提前谢谢。

1 个答案:

答案 0 :(得分:1)

您可以尝试获取Outlook流程并将其窗口向前推进。没有.NET钩子这样做,你需要使用本机Win32 DLL。

    [Flags()]
    private enum SetWindowPosFlags : uint
    {
        SynchronousWindowPosition = 0x4000,
        DeferErase = 0x2000,
        DrawFrame = 0x0020,
        FrameChanged = 0x0020,
        HideWindow = 0x0080,
        DoNotActivate = 0x0010,
        DoNotCopyBits = 0x0100,
        IgnoreMove = 0x0002,
        DoNotChangeOwnerZOrder = 0x0200,
        DoNotRedraw = 0x0008,
        DoNotReposition = 0x0200,
        DoNotSendChangingEvent = 0x0400,
        IgnoreResize = 0x0001,
        IgnoreZOrder = 0x0004,
        ShowWindow = 0x0040,
    }

    [DllImport("user32.dll")]
    [return: MarshalAs(UnmanagedType.Bool)]
    static extern bool SetForegroundWindow(IntPtr hWnd);

    [DllImport("user32.dll", SetLastError = true)]
    static extern bool BringWindowToTop(IntPtr hWnd);

    static readonly IntPtr HWND_TOP = new IntPtr(0);
    static readonly IntPtr HWND_TOPMOST = new IntPtr(-1);

    [DllImport("user32.dll")]
    [return: MarshalAs(UnmanagedType.Bool)]
    static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, SetWindowPosFlags uFlags);

    [DllImport("user32.dll")]
    static extern IntPtr SetFocus(IntPtr hWnd);

    int bring_window_to_front_mode = 1; // Various option here, try them out
    Boolean TopMost = false;

        System.Diagnostics.Process[] prcs = System.Diagnostics.Process.GetProcessesByName("XBMCLauncher");
        foreach (var proc in prcs)
        {
            Log.LogLine("Main Window Handle {0}", p.MainWindowHandle.ToInt32());
            switch (bring_window_to_front_mode)
            {
                case 1:
                    if (TopMost)
                    {
                        Log.LogLine("SetWindowPos TopMost {0}", p.MainWindowHandle.ToInt32());
                        SetWindowPos(p.MainWindowHandle, HWND_TOPMOST, 50, 50, 500, 500, SetWindowPosFlags.IgnoreMove | SetWindowPosFlags.IgnoreResize);
                    }
                    else
                    {
                        Log.LogLine("SetWindowPos {0}", p.MainWindowHandle.ToInt32());
                        SetWindowPos(p.MainWindowHandle, HWND_TOP, 50, 50, 500, 500, SetWindowPosFlags.IgnoreMove | SetWindowPosFlags.IgnoreResize);
                    }
                    break;
                case 2:
                    Log.LogLine("BringWindowToTop {0}", p.MainWindowHandle.ToInt32());
                    BringWindowToTop(p.MainWindowHandle);
                    break;
                case 3:
                    Log.LogLine("SetForegroundWindow {0}", p.MainWindowHandle.ToInt32());
                    SetForegroundWindow(p.MainWindowHandle);
                    break;
                case 4:
                    Log.LogLine("SetFocus {0}", p.MainWindowHandle.ToInt32());
                    SetFocus(p.MainWindowHandle);
                    break;
            }
        }

当然是Try / Catch ......