3

我试图在我的 WinForms UI 中显示来自不同进程的系统菜单(包含最小化、还原等)。我知道我需要像 GetSystemMenu 和 TrackPopupMenuEx 这样的互操作调用,但我没有让它工作。有人可以提供示例代码怎么做吗?

我找到了这个代码片段(用于 WPF): 打开另一个应用程序的系统菜单

我把它修改成这样:

    const uint TPM_LEFTBUTTON = 0x0000;
    const uint TPM_RETURNCMD = 0x0100;
    const uint WM_SYSCOMMAND = 0x0112;

    [DllImport("user32.dll")]
    static extern IntPtr GetSystemMenu(IntPtr hWnd, bool bRevert);

    [DllImport("user32.dll")]
    static extern uint TrackPopupMenuEx(IntPtr hmenu, uint fuFlags, int x, int y, IntPtr hwnd, IntPtr lptpm);

    [return: MarshalAs(UnmanagedType.Bool)]
    [DllImport("user32.dll", SetLastError = true)]
    static extern bool PostMessage(IntPtr hWnd, uint Msg, IntPtr wParam, IntPtr lParam);

    public void ShowContextMenu()
    {
        IntPtr wMenu = GetSystemMenu(ExternalWindowHandle, false);
        // Display the menu
        uint command = TrackPopupMenuEx(wMenu, TPM_LEFTBUTTON | TPM_RETURNCMD, 10, 10, ExternalWindowHandle, IntPtr.Zero);
        if (command == 0)
            return;
        PostMessage(ExternalWindowHandle, WM_SYSCOMMAND, new IntPtr(command), IntPtr.Zero);
    }

如问题标题中所述,我不想最小化系统托盘的窗口,我想在我选择的位置显示来自另一个进程(窗口)的系统菜单。几乎和windows任务栏一样。当您在任务栏上右键单击它时,任务栏(资源管理器)似乎能够显示系统菜单。

谢谢,斯特凡

4

1 回答 1

5

我有一个工作版本的代码,我也检查了 MSDN 库,我发现为了让 TrackPopupMenuEx 方法工作你传递的“ExternalWindowHandle”变量,即句柄代表的窗口需要在前台桌面。

MSDN 库说明如下:

“要显示通知图标的上下文菜单,当前窗口必须是应用程序调用 TrackPopupMenu 或 TrackPopupMenuEx 之前的前台窗口。否则,当用户在菜单外或创建菜单的窗口之外单击时,菜单不会消失(如果它是可见的)。如果当前窗口是子窗口,则必须将(顶级)父窗口设置为前景窗口。, http://msdn.microsoft.com/en-us/library/windows/desktop/ms648003(v=vs.85).aspx

所以这意味着它只会在你的窗口是活动窗口时工作,如果你在 Visual Studio 中调试它不会工作,因为窗口不是前台窗口,即 Visual Studio 不是你的应用程序。

请参阅随附的工作代码示例,请记住,它仅在应用程序窗口处于焦点/前景时才有效。即,当您在调试或使用另一个窗口时,TrackPopupMenuEx 将始终返回 0。

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace WindowsFormsApplication2
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        const uint TPM_LEFTBUTTON = 0x0000;
        const uint TPM_RETURNCMD = 0x0100;
        const uint WM_SYSCOMMAND = 0x0112;

        [DllImport("user32.dll")]
        static extern IntPtr GetSystemMenu(IntPtr hWnd, bool bRevert);

        [DllImport("user32.dll")]
        static extern uint TrackPopupMenuEx(IntPtr hmenu, uint fuFlags, int x, int y, IntPtr hwnd, IntPtr lptpm);

        [return: MarshalAs(UnmanagedType.Bool)]
        [DllImport("user32.dll", SetLastError = true)]
        static extern bool PostMessage(IntPtr hWnd, uint Msg, IntPtr wParam, IntPtr lParam);

        public static void ShowContextMenu(IntPtr appWindow, IntPtr myWindow, Point point)
        {
            IntPtr wMenu = GetSystemMenu(appWindow, false);
            // Display the menu
            uint command = TrackPopupMenuEx(wMenu,
                TPM_LEFTBUTTON | TPM_RETURNCMD, (int)point.X, (int)point.Y, myWindow, IntPtr.Zero);
            if (command == 0)
                return;

            PostMessage(appWindow, WM_SYSCOMMAND, new IntPtr(command), IntPtr.Zero);
        }

        private void button1_Click(object sender, EventArgs e)
        {
            ShowContextMenu(new IntPtr(<<put your target window handle here>>), this.Handle, new Point(0, 0));
        }
    }
}
于 2013-03-05T19:21:00.540 回答