C#上下文菜单删除默认选择

C#上下文菜单删除默认选择,c#,mobile,compact-framework,contextmenu,C#,Mobile,Compact Framework,Contextmenu,好吧,这个问题似乎太简单了,但我已经浪费了足够多的时间来寻找如何做到这一点。我正在移动设备上使用CE 6.5,我有一个带有六个菜单项的ContextMenu。当菜单弹出时,列表中的第一项将自动突出显示。我想删除这个亮点,因为它让我的一些用户感到困惑,以为这是当前状态。我查看了ContextMenu及其所有变量和MenuItem,并没有发现如何删除第一项的自动突出显示。主菜单也是如此。我想答案是,不幸的是,你不能。今天下午,我花了很大的力气才弄到了菜单,但我就是弄不到操作系统允许我使用的有效菜单。

好吧,这个问题似乎太简单了,但我已经浪费了足够多的时间来寻找如何做到这一点。我正在移动设备上使用CE 6.5,我有一个带有六个菜单项的ContextMenu。当菜单弹出时,列表中的第一项将自动突出显示。我想删除这个亮点,因为它让我的一些用户感到困惑,以为这是当前状态。我查看了ContextMenu及其所有变量和MenuItem,并没有发现如何删除第一项的自动突出显示。主菜单也是如此。

我想答案是,不幸的是,你不能。今天下午,我花了很大的力气才弄到了菜单,但我就是弄不到操作系统允许我使用的有效菜单。如果你想继续追寻我的路径,代码如下,但我真的认为这是一条死胡同。在这一点上,如果你真的需要这个特性,我会考虑p/调用菜单的所有内容(创作、人口等)。
using System;
using System.Linq;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
using System.Reflection;
using System.Runtime.InteropServices;

using UINT = System.UInt32;
using HMENU = System.IntPtr;
using HBITMAP = System.IntPtr;
using DWORD = System.UInt32;
using LPTSTR = System.IntPtr;

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

            contextMenu.MenuItems.Add(new MenuItem() { Text = "Item A" });
            contextMenu.MenuItems.Add(new MenuItem() { Text = "Item B" });
            contextMenu.MenuItems.Add(new MenuItem() { Text = "Item C" });
            contextMenu.MenuItems.Add(new MenuItem() { Text = "Item D" });

            this.MouseDown += new MouseEventHandler(Form1_MouseDown);
            contextMenu.Popup += new EventHandler(contextMenu_Popup);
        }

        void contextMenu_Popup(object sender, EventArgs e)
        {
            var type = contextMenu.GetType();
            var members = type.GetMembers(
                          BindingFlags.NonPublic | BindingFlags.Instance);
            var menuMember = type.GetField("m_hmnu", 
                             BindingFlags.NonPublic | BindingFlags.Instance);
            var hMenu = (HMENU)menuMember.GetValue(contextMenu);

            var info = new MENUITEMINFO();
            info.cbSize = (uint)Marshal.SizeOf(info);
            info.fMask = MIIM_STATE;
            var result = GetMenuItemInfo(hMenu, 0, true, out info);
            if (!result)
            {
                var err = Marshal.GetLastWin32Error();
                if (err == 0x0579) MessageBox.Show("Invalid menu handle");
                return;
            }
            info.fMask = MIIM_STATE;
            info.fState &= (~MFS_HILITE);
            result = SetMenuItemInfo(hMenu, 0, true, ref info); 
        }

        void Form1_MouseDown(object sender, MouseEventArgs e)
        {
            contextMenu.Show(this, new Point(e.X, e.Y));
        }

        private const uint MIIM_STATE = 1;
        private const uint MFS_UNHILITE = 0;
        private const uint MFS_HILITE = 0x80;        

        //typedef struct tagMENUITEMINFO {
        //  UINT cbSize; 
        //  UINT fMask; 
        //  UINT fType; 
        //  UINT fState; 
        //  UINT wID; 
        //  HMENU hSubMenu; 
        //  HBITMAP hbmpChecked; 
        //  HBITMAP hbmpUnchecked; 
        //  DWORD dwItemData; 
        //  LPTSTR dwTypeData; 
        //  UINT cch; 
        //} MENUITEMINFO, FAR* LPMENUITEMINFO; 
        private struct MENUITEMINFO
        {
            public UINT cbSize;
            public UINT fMask;
            public UINT fType;
            public UINT fState;
            public UINT wID;
            public HMENU hSubMenu;
            public HBITMAP hbmpChecked;
            public HBITMAP hbmpUnchecked;
            public DWORD dwItemData;
            public LPTSTR dwTypeData;
            public UINT cch; 
        }

        //BOOL SetMenuItemInfo(
        //  HMENU hMenu,
        //  UINT uItem,
        //  BOOL fByPosition,
        //  LPCMENUITEMINFO lpmii
        //);
        [DllImport("coredll", SetLastError = true)]
        private static extern bool SetMenuItemInfo(HMENU hMenu, UINT uItem, 
                                   [MarshalAs(UnmanagedType.Bool)]bool fByPosition, 
                                   ref MENUITEMINFO lpmii);

        //BOOL GetMenuItemInfo(
        //  HMENU hMenu,
        //  UINT uItem,
        //  BOOL fByPosition,
        //  LPMENUITEMINFO lpmii
        //);
        [DllImport("coredll", SetLastError = true)]
        private static extern bool GetMenuItemInfo(HMENU hMenu, UINT uItem, 
                                   [MarshalAs(UnmanagedType.Bool)]bool fByPosition, 
                                   out MENUITEMINFO lpmii);

        //HMENU GetSubMenu(
        //  HMENU hMenu,
        //  int nPos
        //);
        [DllImport("coredll", SetLastError = true)]
        private static extern HMENU GetSubMenu(HMENU hMenu, int nPos);
    }
}
编辑


我知道我在某个地方有代码来做这一切。我们曾经出售一个商业PopupMenu控件,它封装了所有用于创建菜单的P/调用。控件的销售额很小,所以几年前我们就从产品线中撤出了它。我现在以开源的形式发布了它。

如果您能告诉我如何p/调用它,因为我对Windows CE/C#环境(Java coder by trade)还是一个新手,这将是一个很长的教训——远比这里的文章要多。这与我上面所做的是一样的,但是您必须P/Invoke来创建菜单和项,然后将菜单单击处理程序连接到托管委托中。这并不太难——我会给它打10分中的4分左右,但会有点长。