Compartir a través de


Agregar un comando al menú contextual

Puede agregar comandos de menú a su lenguaje específico de dominio (DSL) para que sus usuarios puedan realizar tareas específicas de su DSL. Los comandos aparecen en el menú contextual cuando los usuarios hacen clic con el botón secundario en el diagrama. Puede definir un comando para que solo aparezca en el menú en circunstancias específicas. Por ejemplo, puede hacer que el comando sea visible solo cuando el usuario haga clic en tipos específicos de elementos, o en elementos con unos estados determinados.

En resumen, los pasos se realizan en el proyecto DslPackage de la siguiente manera:

  1. Declarar el comando en Commands.vsct.

  2. Actualizar el número de versión del paquete en Package.tt. Tiene que hacerlo siempre que cambie Commands.vsct

  3. Escribir métodos en la clase CommandSet para que el comando sea visible y definir lo que quiere que haga el comando.

Nota

También puede modificar el comportamiento de algunos comandos existentes, como Cortar, Pegar, Seleccionar todo e Imprimir invalidando los métodos en CommandSet.cs. Para obtener más información, vea Procedimientos para modificar un comando de menú estándar.

Definición de un comando mediante MEF

Managed Extension Framework (MEF) proporciona un método alternativo para definir comandos de menú en el menú del diagrama. Su principal finalidad es habilitar un DSL para que usted u otras personas puedan ampliarlo. Los usuarios pueden elegir instalar solo el DSL o pueden instalar el DSL y las extensiones. Sin embargo, una vez hecho el trabajo inicial de habilitar MEF en el DSL, MEF reduce el trabajo de definir los comandos del menú contextual.

Use el método de este tema si:

  1. Quiere definir comandos de menú en otros menús que no sea el contextual.

  2. Quiere definir agrupaciones específicas de comandos en el menú.

  3. No quiere permitir que otros amplíen el DSL con sus propios comandos.

  4. Solo quiere definir un comando.

    En otros casos, considere usar el método MEF para definir comandos. Para obtener más información, vea Extensión del DSL mediante MEF.

Declarar el comando en Commands.vsct

Los comandos de menú se declaran en DslPackage\Commands.vsct. Estas definiciones especifican las etiquetas de los elementos de menú y dónde aparecen en los menús.

El archivo que edita, Commands.vsct, importa definiciones de varios archivos .h, que se encuentran en el directorio ruta de acceso de instalación del SDK de Visual Studio\VisualStudioIntegration\Common\Inc. También incluye GeneratedVsct.vsct, que se genera a partir de la definición de DSL.

Para obtener más información sobre los archivos .vsct, vea Archivos de tabla de comandos de Visual Studio (.Vsct).

Para agregar el comando

  1. En el Explorador de soluciones, en el proyecto DslPackage, abra Commands.vsct.

  2. En el elemento Commands, defina uno o varios botones y un grupo. Un botón es un elemento del menú. Un grupo es una sección del menú. Para definir estos elementos, agregue lo siguiente:

    <!-- Define a group - a section in the menu -->
    <Groups>
      <Group guid="guidCustomMenuCmdSet" id="grpidMyMenuGroup" priority="0x0100">
        <!-- These symbols are defined in GeneratedVSCT.vsct -->
        <Parent guid="guidCmdSet" id="menuidContext" />
      </Group>
    </Groups>
    <!-- Define a button - a menu item - inside the Group -->
    <Buttons>
      <Button guid="guidCustomMenuCmdSet" id="cmdidMyContextMenuCommand"
        priority="0x0100" type="Button">
        <Parent guid="guidCustomMenuCmdSet" id="grpidMyMenuGroup"/>
        <!-- If you do not want to place the command in your own Group,
             use Parent guid="guidCmdSet" id="grpidContextMain".
             These symbols are defined in GeneratedVSCT.vsct -->
        <CommandFlag>DynamicVisibility</CommandFlag>
        <Strings>
          <ButtonText>My Context Menu Command</ButtonText>
        </Strings>
      </Button>
    </Buttons>
    

    Nota

    Cada botón o grupo se identifica por un GUID y un identificador entero. Puede crear varios grupos y botones con el mismo GUID. Sin embargo, sus identificadores deben ser diferentes. Los nombres de GUID y los nombres de id. se traducen a los GUID e id. reales en el nodo <Symbols>.

  3. Agregue una restricción de visibilidad para el comando de manera que se cargue solo en el contexto de su lenguaje específico de dominio. Para obtener más información, vea Elemento VisibilityConstraints.

    Para ello, agregue los siguientes elementos en el elemento CommandTable, después del elemento Commands.

    <VisibilityConstraints>
      <!-- Ensures the command is only loaded for this DSL -->
      <VisibilityItem guid="guidCustomMenuCmdSet" id="cmdidMyContextMenuCommand"
        context="guidEditor"/>
    </VisibilityConstraints>
    
  4. Defina los nombres que usó para los GUID y los id. Para ello, agregue un elemento Symbols en el elemento CommandTable, después del elemento Commands.

    <Symbols>
      <!-- Substitute a unique GUID for the placeholder: -->
      <GuidSymbol name="guidCustomMenuCmdSet"
        value="{00000000-0000-0000-0000-000000000000}" >
        <IDSymbol name="grpidMyMenuGroup" value="0x01001"/>
        <IDSymbol name="cmdidMyContextMenuCommand" value="0x00001"/>
      </GuidSymbol>
    </Symbols>
    
  5. Reemplace {000...000} por un GUID que identifique los grupos y elementos de menú. Para obtener un nuevo GUID, use la herramienta Crear GUID en el menú Herramientas.

    Nota

    Para agregar más grupos o elementos de menú, puede usar el mismo GUID. Sin embargo, debe usar nuevos valores para IDSymbols.

  6. En el código que copió de este procedimiento, reemplace cada aparición de las siguientes cadenas por sus propias cadenas:

    • grpidMyMenuGroup

    • cmdidMyContextMenuCommand

    • guidCustomMenuCmdSet

    • My Context Menu Command

Actualizar la versión del paquete en Package.tt

Siempre que se agregue o se cambie un comando, actualice el parámetro version del ProvideMenuResourceAttribute que se aplica a la clase de paquete antes de publicar la nueva versión de su lenguaje específico de dominio.

Como la clase de paquete se define en un archivo generado, actualice el atributo en el archivo de plantilla de texto que genera el archivo Package.cs.

Para actualizar el archivo Package.tt

  1. En el Explorador de soluciones, en el proyecto DslPackage, en la carpeta GeneratedCode, abra el archivo Package.tt.

  2. Busque el atributo ProvideMenuResource.

  3. Incremente el parámetro version del atributo, que es el segundo parámetro. Si quiere, puede escribir explícitamente el nombre del parámetro para recordarle su finalidad. Por ejemplo:

    [VSShell::ProvideMenuResource("1000.ctmenu", version: 2 )]

Definir el comportamiento del comando

Su DSL ya tiene algunos comandos que se implementan en una clase parcial que se declara en DslPackage\GeneratedCode\CommandSet.cs. Para agregar nuevos comandos, debe extender esta clase creando un nuevo archivo que contenga una declaración parcial de la misma clase. El nombre de la clase suele ser <suNombreDeDsl>CommandSet. Resulta útil empezar por comprobar el nombre de la clase e inspeccionar su contenido.

La clase de conjunto de comandos deriva de CommandSet.

Extensión de la clase CommandSet

  1. En el Explorador de soluciones, en el proyecto DslPackage, abra la carpeta GeneratedCode, busque en CommandSet.tt y abra su archivo generado CommandSet.cs. Anote el espacio de nombres y el nombre de la primera clase que se define en él. Por ejemplo, puede que vea:

    namespace Company.Language1

    { ... internal partial class Language1CommandSet : ...

  2. En DslPackage, cree una carpeta llamada Custom Code. En esta carpeta, cree un archivo de clase llamado CommandSet.cs.

  3. En el nuevo archivo, escriba una declaración parcial que tenga el mismo espacio de nombres y nombre que la clase parcial generada. Por ejemplo:

    namespace Company.Language1 /* Make sure this is correct */

    { internal partial class Language1CommandSet { ...

    Nota

    Si usó la plantilla de clase para crear el archivo, debe corregir tanto el espacio de nombres como el nombre de la clase.

Normalmente, el código del conjunto de comandos tendrá que exportar los siguientes espacios de nombres:

using System;
using System.Collections.Generic;
using System.ComponentModel.Design;
using System.Linq;
using Microsoft.VisualStudio.Modeling;
using Microsoft.VisualStudio.Modeling.Diagrams;
using Microsoft.VisualStudio.Modeling.Shell;

Ajuste el espacio de nombres y el nombre de la clase para que coincidan con los del archivo CommandSet.cs generado:

namespace Company.Language1 /* Make sure this is correct */
{
  // Same class as the generated class.
  internal partial class Language1CommandSet
  {

Tiene que definir dos métodos, uno para determinar cuándo será visible el comando en el menú al que se accede haciendo clic con el botón secundario (contextual) y el otro para ejecutar el comando. Estos métodos no son invalidaciones; se registran en una lista de comandos.

Definir cuándo será visible el comando

Para cada comando, defina un método OnStatus... que determine si el comando aparecerá en el menú y si estará habilitado o atenuado. Establezca las propiedades Visible y Enabled de MenuCommand, tal como se muestra en el ejemplo siguiente. Se llama a este método para construir el menú contextual cada vez que el usuario hace clic con el botón secundario en el diagrama, por lo que debe trabajar rápidamente.

En este ejemplo, el comando solo está visible cuando el usuario ha seleccionado un tipo de forma determinado, y solo se habilita cuando al menos uno de los elementos seleccionados está en un estado determinado. El ejemplo se basa en la plantilla de diagrama de clases de DSL, y ClassShape y ModelClass son tipos que se definen en el DSL:

private void OnStatusMyContextMenuCommand(object sender, EventArgs e)
{
  MenuCommand command = sender as MenuCommand;
  command.Visible = command.Enabled = false;
  foreach (object selectedObject in this.CurrentSelection)
  {
    ClassShape shape = selectedObject as ClassShape;
    if (shape != null)
    {
      // Visibility depends on what is selected.
      command.Visible = true;
      ModelClass element = shape.ModelElement as ModelClass;
      // Enabled depends on state of selection.
      if (element != null && element.Comments.Count == 0)
      {
        command.Enabled = true;
        return; // seen enough
} } } }

Los siguientes fragmentos suelen resultar útiles en los métodos OnStatus:

  • this.CurrentSelection. La forma en la que el usuario hizo clic con el botón secundario se incluye siempre en esta lista. Si el usuario hace clic en una parte en blanco del diagrama, el diagrama es el único miembro de la lista.

  • this.IsDiagramSelected() - true si el usuario hizo clic en una parte en blanco del diagrama.

  • this.IsCurrentDiagramEmpty()

  • this.IsSingleSelection(): el usuario no seleccionó varios objetos.

  • this.SingleSelection: forma o diagrama en los que el usuario hizo clic con el botón secundario.

  • shape.ModelElement as MyLanguageElement: elemento de modelo que representa una forma.

Como regla general, haga que la propiedad Visible dependa de lo que se ha seleccionado y haga que la propiedad Enabled dependa del estado de los elementos seleccionados.

Un método OnStatus no debería cambiar el estado del almacén.

Definir la acción del comando

Para cada comando, defina un método OnMenu... que realice la acción necesaria cuando el usuario haga clic en el comando de menú.

Si realiza cambios en los elementos de modelo, debe hacerlo dentro de una transacción. Para obtener más información, vea Procedimientos para modificar un comando de menú estándar.

En este ejemplo, ClassShape, ModelClass y Comment son tipos que se definen en el DSL, y derivan de la plantilla de diagrama de clases de DSL.

private void OnMenuMyContextMenuCommand(object sender, EventArgs e)
{
  MenuCommand command = sender as MenuCommand;
  Store store = this.CurrentDocData.Store;
  // Changes to elements and shapes must be performed in a Transaction.
  using (Transaction transaction =
       store.TransactionManager.BeginTransaction("My command"))
  {
    foreach (object selectedObject in this.CurrentSelection)
    {
      // ClassShape is defined in my DSL.
      ClassShape shape = selectedObject as ClassShape;
      if (shape != null)
      {
        // ModelClass is defined in my DSL.
        ModelClass element = shape.ModelElement as ModelClass;
        if (element != null)
        {
          // Do required action here - for example:

          // Create a new element. Comment is defined in my DSL.
          Comment newComment = new Comment(element.Partition);
          // Every element must be the target of an embedding link.
          element.ModelRoot.Comments.Add(newComment);
          // Set properties of new element.
          newComment.Text = "This is a comment";
          // Create a reference link to existing object.
          element.Comments.Add(newComment);
        }
      }
    }
    transaction.Commit(); // Don't forget this!
  }
}

Para obtener más información sobre cómo navegar de un objeto a otro en el modelo y cómo crear objetos y vínculos, vea Procedimientos para modificar un comando de menú estándar.

Registrar el comando

Repita en C# las declaraciones de los valores de los GUID e identificadores que realizó en la sección Symbols de CommandSet.vsct:

private Guid guidCustomMenuCmdSet =
    new Guid("00000000-0000-0000-0000-000000000000");
private const int grpidMyMenuGroup = 0x01001;
private const int cmdidMyContextMenuCommand = 1;

Use el mismo valor de GUID que insertó en Commands.vsct.

Nota

Si cambia la sección Symbols del archivo VSCT, debe cambiar también estas declaraciones para que coincidan. También debe incrementar el número de versión en Package.tt

Registre los comandos de menú como parte de este conjunto de comandos. Se llama a GetMenuCommands() una vez cuando el diagrama se inicializa:

protected override IList<MenuCommand> GetMenuCommands()
{
  // Get the list of generated commands.
  IList<MenuCommand> commands = base.GetMenuCommands();
  // Add a custom command:
  DynamicStatusMenuCommand myContextMenuCommand =
    new DynamicStatusMenuCommand(
      new EventHandler(OnStatusMyContextMenuCommand),
      new EventHandler(OnMenuMyContextMenuCommand),
      new CommandID(guidCustomMenuCmdSet, cmdidMyContextMenuCommand));
  commands.Add(myContextMenuCommand);
  // Add more commands here.
  return commands;
}

Probar el comando

Compile y ejecute el DSL en una instancia experimental de Visual Studio. El comando debe aparecer en el menú contextual en las situaciones que haya especificado.

Para probar el comando

  1. En la barra de herramientas del Explorador de soluciones, haga clic en Transformar todas las plantillas.

  2. Presione F5 para volver a compilar la solución y comience la depuración del lenguaje específico de dominio en la compilación experimental.

  3. En la compilación experimental, abra un diagrama de muestra.

  4. Haga clic con el botón secundario en varios elementos del diagrama para comprobar que el comando aparece correctamente habilitado o deshabilitado, y que se muestra o se oculta correctamente según el elemento seleccionado.

Solución de problemas

El comando no aparece en el menú:

  • El comando solo aparecerá en las instancias de depuración de Visual Studio, hasta que instale el paquete de DSL. Para obtener más información, vea Implementación de soluciones de lenguaje específico de dominio.

  • Asegúrese de que la muestra experimental tiene la extensión de nombre de archivo correcta para este DSL. Para comprobar la extensión de nombre de archivo, abra DslDefinition.dsl en la instancia principal de Visual Studio. Después, en DSL Explorer (Explorador de DSL), haga clic en el nodo Editor y en Properties (Propiedades). En la ventana Properties (Propiedades), haga clic en la propiedad FileExtension.

  • ¿Incrementó el número de versión del paquete?

  • Establezca un punto de interrupción al principio del método OnStatus. Debería interrumpirse al hacer clic con el botón secundario en cualquier parte del diagrama.

No se llama al método OnStatus:

  • Asegúrese de que los GUID e identificadores de su código de CommandSet coinciden con los de la sección Symbols de Commands.vsct.

  • En Commands.vsct, asegúrese de que, en cada nodo primario, el GUID y el identificador identifican correctamente el grupo primario.

  • En un símbolo del sistema de Visual Studio, escriba devenv /rootsuffix exp /setup. Después, reinicie la instancia de depuración de Visual Studio.

  • Recorra el método OnStatus para comprobar que el valor de command.Visible y command.Enabled es True.

Se muestra un texto incorrecto o el comando aparece en el lugar incorrecto:

  • Asegúrese de que la combinación de GUID e identificador sea única para este comando.

  • Asegúrese de que ha desinstalado las versiones anteriores del paquete.

Nota

El componente Transformación de plantilla de texto se instala de forma automática como parte de la carga de trabajo Desarrollo de extensiones de Visual Studio. También lo puede instalar desde la pestaña Componentes individuales del Instalador de Visual Studio, en la categoría SDK, bibliotecas y marcos. Instale el componente SDK de modelado desde la pestaña Componentes individuales.