VS 2015动态菜单命令不像宣传的那样工作

本文关键字:工作 动态 2015 菜单 命令 VS | 更新日期: 2023-09-27 18:12:50

我试图在VS 2015扩展中创建动态菜单项。我能做的最好的事情就是让占位符命令保持可见和活动状态。下面是我从VS向导开始创建的最简单的例子(很抱歉太长了,但这确实是我可以创建的最简单的例子):

Command1Package.cs:

using Microsoft.VisualStudio;
using Microsoft.VisualStudio.Shell;
using System;
using System.ComponentModel.Design;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.Runtime.InteropServices;
namespace minimal {
    [PackageRegistration(UseManagedResourcesOnly = true)]
    [InstalledProductRegistration("#110", "#112", "1.0", IconResourceID = 400)] // Info on this package for Help/About
    [ProvideMenuResource("Menus.ctmenu", 1)]
    [Guid(Command1Package.PackageGuidString)]
    [ProvideAutoLoad(VSConstants.UICONTEXT.NoSolution_string)]
    [SuppressMessage("StyleCop.CSharp.DocumentationRules", "SA1650:ElementDocumentationMustBeSpelledCorrectly", Justification = "pkgdef, VS and vsixmanifest are valid VS terms")]
    public sealed class Command1Package : Package {
        public const string PackageGuidString = "3e88287b-7b79-403d-ae8d-3329af218869";
        public Command1Package() {
            }
        #region Package Members
        protected override void Initialize() {
            Debug.WriteLine("minimal package instantiated");
            Command1.Initialize(this, GetService(typeof(IMenuCommandService)) as OleMenuCommandService);
            base.Initialize();
            }
        #endregion
        }
    }

Command1.cs:

using Microsoft.VisualStudio.Shell;
using System;
using System.ComponentModel.Design;
using System.Diagnostics;
namespace minimal {
    internal sealed class Command1 {
        public static readonly Guid CommandSet = new Guid("c1388361-6429-452c-8ba0-580d292ef0ca");
        private Command1() {
            }
        public static void Initialize(Package package, OleMenuCommandService mcs) {
            AddOneCommand(mcs, 0x0100, "Renamed placeholder command");  // the placeholder
            AddOneCommand(mcs, 0x0101, "Dynamic command 1"); // a dynamic command
            AddOneCommand(mcs, 0x0102, "Dynamic command 2");
            }
        internal static void AddOneCommand(OleMenuCommandService mcs, uint cmdid, string cmdname) {
            Debug.WriteLine("AddOneCommand" + cmdid.ToString("X4"));
            CommandID id = new CommandID(CommandSet, (int) cmdid);
            OleMenuCommand mc = new OleMenuCommand(new EventHandler(MenuItemCallback), id, cmdname);
            mc.BeforeQueryStatus += OnBeforeQueryStatus;
            mcs.AddCommand(mc);
            }
        private static void OnBeforeQueryStatus(object sender, EventArgs e) {
            OleMenuCommand mc = sender as OleMenuCommand;
            Debug.WriteLine("OnBeforeQueryStatus called for " + mc.CommandID.ToString());
            }
        private static void MenuItemCallback(object sender, EventArgs e) {
            OleMenuCommand mc = sender as OleMenuCommand;
            System.Windows.Forms.MessageBox.Show("MenuItemCallback called for " + mc.CommandID.ToString());
            }
        }
    }

Command1Package.vsct:

<?xml version="1.0" encoding="utf-8"?>
<CommandTable xmlns="http://schemas.microsoft.com/VisualStudio/2005-10-18/CommandTable" xmlns:xs="http://www.w3.org/2001/XMLSchema">
    <Extern href="stdidcmd.h"/>
    <Extern href="vsshlids.h"/>
    <Commands package="guidCommand1Package">
        <Groups>
            <Group guid="guidCommand1PackageCmdSet" id="MyMenuGroup" priority="0x0600">
                <Parent guid="guidSHLMainMenu" id="IDM_VS_MENU_TOOLS"/>
            </Group>
            <Group guid="guidCommand1PackageCmdSet" id="MyMenuSubgroup" priority="0x0100">
                <Parent guid="guidCommand1PackageCmdSet" id="SubMenu"/>
            </Group>
        </Groups>
        <Menus>
            <Menu guid="guidCommand1PackageCmdSet" id="SubMenu" priority="0x0100" type="Menu">
                <Parent guid="guidCommand1PackageCmdSet" id="MyMenuGroup"/>
                <Strings>
                    <ButtonText>Minimal commands</ButtonText>
                    <CommandName>MinimalCommands</CommandName>
                </Strings>
            </Menu>
        </Menus>
        <Buttons>
            <Button guid="guidCommand1PackageCmdSet" id="Command1Id" priority="0x0100" type="Button">
                <Parent guid="guidCommand1PackageCmdSet" id="MyMenuSubgroup" />
                <CommandFlag>DynamicItemStart</CommandFlag>
                <CommandFlag>TextChanges</CommandFlag>
                <CommandFlag>DynamicVisibility</CommandFlag>
                <Strings>
                    <ButtonText>Invoke Command1</ButtonText>
                    <CommandName>Command1</CommandName>
                </Strings>
            </Button>
        </Buttons>
    </Commands>
    <Symbols>
        <GuidSymbol name="guidCommand1Package" value="{3e88287b-7b79-403d-ae8d-3329af218869}" />
        <GuidSymbol name="guidCommand1PackageCmdSet" value="{c1388361-6429-452c-8ba0-580d292ef0ca}">
            <IDSymbol name="MyMenuGroup" value="0x1020" />
            <IDSymbol name="MyMenuSubgroup" value="0x1021"/>
            <IDSymbol name="SubMenu" value="0x200"/>
            <IDSymbol name="Command1Id" value="0x0100" />
        </GuidSymbol>
    </Symbols>
</CommandTable>

一些观察:

  1. 如果包上没有[ProvideAutoLoad]属性,框架不会实例化包,直到之后执行占位符命令,这是此时子菜单上唯一的项。似乎由VS向导生成的代码应该为你做这件事,因为这个属性(在许多其他属性中)没有在Package类的帮助条目中提到。我真的希望占位符不出现在菜单上。但是将它标记为不可见(在OnBeforeQueryStatus中)会使整个子菜单消失。我能做的最好的事情就是更改这个命令的名称。"如何:动态添加菜单项"的帮助条目没有提到这一点。(不要费心阅读VS 2015的条目,因为示例代码不必要地复杂,并且没有显示如何做动态添加菜单命令的一件简单的事情。VS 2012的例子更容易遵循。)
  2. Dang,但这在MFC中要容易得多!GetMenu(), GetSubMenu()等。我认为。net应该比传统的Win32编程更容易。
  3. 有没有更好的方法来添加动态项目到菜单?

VS 2015动态菜单命令不像宣传的那样工作

是的,它比在MFC中困难得多。

  1. [provideautolload]确实是必需的。

  2. 您不应该添加占位符命令。从

    开始

    AddOneCommand(mcs, 0x0100, "Dynamic command 1");

通常您将DefaultInvisible添加到DynamicItemStart命令并设置menuCommand。Text and menuCommand。在OnBeforeQueryStatus处理程序中可见