Hinzufügen eines Befehls zum Kontextmenü
Sie können Ihrer domänenspezifischen Sprache (DSL) Menübefehle hinzufügen, damit Benutzer für die DSL typische Aufgaben ausführen können. Die Befehle werden im Kontextmenü angezeigt, wenn Benutzer mit der rechten Maustaste auf das Diagramm klicken. Sie können einen Befehl so definieren, dass er nur unter bestimmten Bedingungen im Menü angezeigt wird. Sie können beispielsweise angeben, dass der Befehl nur sichtbar ist, wenn der Benutzer auf bestimmte Typen von Elementen oder auf Elemente in bestimmten Zuständen klickt.
Zusammenfassend werden die folgenden Schritte im DslPackage-Projekt ausgeführt:
Aktualisieren der Paketversionsnummer in „Package.tt“ Dies müssen Sie nach jeder Änderung von Commands.vsct tun.
Schreiben von Methoden in der CommandSet-Klasse, um den Befehl sichtbar zu machen und dessen Aktionen zu definieren.
Hinweis
Sie können auch das Verhalten einiger vorhandener Befehle wie Ausschneiden, Einfügen, Alle auswählen und Drucken ändern, indem Sie die entsprechenden Methoden in "CommandSet.cs" überschreiben. Weitere Informationen finden Sie unter Vorgehensweise: Ändern eines Standardmenübefehls.
Definieren eines Befehls mit MEF
Managed Extension Framework (MEF) bietet eine alternative Möglichkeit, Menübefehle im Diagrammmenü zu definieren. Sein Hauptzweck besteht darin, Ihnen oder anderen Personen die Erweiterung einer DSL zu ermöglichen. Benutzer können entweder nur die DSL oder die DSL und Erweiterungen installieren. Nach dem anfänglichen Aufwand der Aktivierung von MEF für die DSL vereinfacht MEF jedoch das Definieren von Kontextmenübefehlen.
Verwenden Sie das Verfahren in diesem Thema in folgenden Fällen:
Sie möchten Menübefehle in anderen Menüs als dem Rechtsklick-Kontextmenü definieren.
Sie möchten bestimmte Gruppierungen von Befehlen im Menü definieren.
Sie möchten nicht, dass andere Personen die DSL mit ihren eigenen Befehlen erweitern können.
Sie möchten nur einen einzelnen Befehl definieren.
In anderen Fällen können Sie die MEF-Methode zum Definieren von Befehlen in Betracht ziehen. Weitere Informationen finden Sie unter Erweitern von DSL mittels MEF.
Deklarieren des Befehls in Commands.Vsct
Menübefehle werden in "DslPackage\Commands.vsct" deklariert. Diese Definitionen definieren die Bezeichnungen der Menüelemente und ihre Position in den Menüs.
Die von Ihnen bearbeitete Datei „Commands.vsct“ importiert Definitionen aus mehreren H-Dateien, die sich im Verzeichnis „Visual Studio SDK-Installationspfad\VisualStudioIntegration\Common\Inc“ befinden. Darin enthalten ist auch die Datei „GeneratedVsct.vsct“, die aus Ihrer DSL-Definition generiert wird.
Weitere Informationen zu VSCT-Dateien finden Sie unter VSCT-Dateien (Visual Studio Command Table).
So fügen Sie den Befehl hinzu
Öffnen Sie im Projektmappen-Explorer unter dem Projekt DslPackage die Datei „Commands.vsct“.
Definieren Sie im
Commands
-Element mindestens eine Schaltfläche und eine Gruppe. Eine Schaltfläche ist ein Element im Menü. Eine Gruppe stellt einen Abschnitt im Menü dar. Fügen Sie folgende Elemente hinzu, um diese Objekte zu definieren:<!-- 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>
Hinweis
Jede Schaltfläche oder Gruppe wird durch eine GUID und eine ganzzahlige ID identifiziert. Sie können mehrere Gruppen und Schaltflächen mit derselben GUID erstellen. Alle Gruppen und Schaltflächen müssen jedoch unterschiedliche IDs aufweisen. Die GUID- und ID-Namen werden in tatsächliche GUIDs und numerische IDs im
<Symbols>
-Knoten übersetzt.Fügen Sie eine Sichtbarkeitseinschränkung für den Befehl hinzu, sodass er nur im Kontext der domänenspezifischen Sprache geladen wird. Weitere Informationen finden Sie unter VisibilityConstraints-Element.
Fügen Sie hierzu im
CommandTable
-Element nach demCommands
-Element die folgenden Elemente hinzu.<VisibilityConstraints> <!-- Ensures the command is only loaded for this DSL --> <VisibilityItem guid="guidCustomMenuCmdSet" id="cmdidMyContextMenuCommand" context="guidEditor"/> </VisibilityConstraints>
Definieren Sie die Namen, die Sie für die GUIDs und IDs verwendet haben. Fügen Sie hierzu im
Symbols
-Element nach demCommandTable
-Element einCommands
-Element hinzu.<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>
Ersetzen Sie
{000...000}
durch eine GUID, die Ihre Gruppen und Menüelemente identifiziert. Verwenden Sie im Menü Tools das Tool GUID erstellen, um eine neue GUID abzurufen.Hinweis
Wenn Sie mehrere Gruppen oder Menüelemente hinzufügen, können Sie dieselbe GUID verwenden. Sie müssen aber neue Werte für
IDSymbols
verwenden.Ersetzen Sie in dem Code, den Sie aus diesem Verfahren kopiert haben, jedes Vorkommen der folgenden Zeichenfolgen durch Ihre eigenen Zeichenfolgen:
grpidMyMenuGroup
cmdidMyContextMenuCommand
guidCustomMenuCmdSet
My Context Menu Command
Aktualisieren der Paketversion in Package.tt
Immer wenn Sie einen Befehl hinzufügen oder ändern, müssen Sie den version
-Parameter des Attributs ProvideMenuResourceAttribute aktualisieren, das auf die Paketklasse angewendet wird, bevor Sie die neue Version der domänenspezifischen Sprache veröffentlichen.
Da die Paketklasse in einer generierten Datei definiert wird, müssen Sie das Attribut in der Textvorlagendatei aktualisieren, aus der die Datei Package.cs generiert wird.
So aktualisieren Sie die Datei Package.tt
Öffnen Sie im Projektmappen-Explorer im Projekt DslPackage im Ordner GeneratedCode die Datei „Package.tt“.
Suchen Sie das
ProvideMenuResource
-Attribut.Erhöhen Sie den
version
-Parameter des Attributs. Es ist der zweite Parameter. Auf Wunsch können Sie den Parameternamen explizit angeben, um sich an den Zweck zu erinnern. Beispiel:[VSShell::ProvideMenuResource("1000.ctmenu", version: 2 )]
Definieren des Verhaltens des Befehls
Die DSL umfasst bereits einige Befehle. Diese sind in einer partiellen Klasse implementiert, die in "DslPackage\GeneratedCode\CommandSet.cs" deklariert ist. Um neue Befehle hinzuzufügen, müssen Sie diese Klasse erweitern. Dazu erstellen Sie eine neue Datei mit einer partiellen Deklaration derselben Klasse. Der Name der Klasse ist üblicherweise <IhrDslName>CommandSet
. Überprüfen Sie am besten zunächst den Namen der Klasse und ihren Inhalt.
Die Befehlssatzklasse wird von CommandSet abgeleitet.
Erweitern der CommandSet-Klasse
Öffnen Sie im Projektmappen-Explorer im Projekt "DslPackage" den Ordner "GeneratedCode". Suchen Sie unter "CommandSet.tt" die generierte Datei "CommandSet.cs", und öffnen Sie sie. Notieren Sie sich den Namespace und den Namen der ersten dort definierten Klasse. Es wird beispielsweise Folgendes angezeigt:
namespace Company.Language1
{ ... internal partial class Language1CommandSet : ...
Erstellen Sie in DslPackage einen Ordner mit dem Namen Custom Code. Erstellen Sie in diesem Ordner eine neue Klassendatei namens
CommandSet.cs
.Schreiben Sie in die neue Datei eine partielle Deklaration mit demselben Namespace und Namen wie die generierte partielle Klasse. Beispiel:
namespace Company.Language1 /* Make sure this is correct */
{ internal partial class Language1CommandSet { ...
Hinweis
Wenn Sie die neue Datei mithilfe der Klassenvorlage erstellt haben, müssen Sie den Namespace und den Klassennamen korrigieren.
Ihr Befehlssatzcode muss in der Regel folgende Namespaces importieren:
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;
Passen Sie den Namespace und den Klassennamen an, sodass sie mit denen in der generierten Datei CommandSet.cs übereinstimmen:
namespace Company.Language1 /* Make sure this is correct */
{
// Same class as the generated class.
internal partial class Language1CommandSet
{
Sie müssen zwei Methoden definieren: Mit der einen bestimmen Sie, wann der Befehl im Kontextmenü angezeigt wird, mit der anderen wird der Befehl ausgeführt. Diese Methoden sind keine Überschreibungen, sondern Sie registrieren sie in einer Liste von Befehlen.
Definieren, wann der Befehl sichtbar ist
Definieren Sie für jeden Befehl eine OnStatus...
-Methode, die bestimmt, ob der Befehl im Menü angezeigt wird und ob er aktiviert oder abgeblendet wird. Legen Sie die Eigenschaften Visible
und Enabled
von MenuCommand
wie im folgenden Beispiel fest. Diese Methode wird jedes Mal, wenn der Benutzer mit der rechten Maustaste auf das Diagramm klickt, aufgerufen, um das Kontextmenü zu erstellen. Daher muss sie schnell ausführbar sein.
Im Beispiel ist der Befehl nur sichtbar, wenn der Benutzer einen bestimmten Formtyp ausgewählt hat. Und er wird nur aktiviert, wenn sich mindestens eines der ausgewählten Elemente in einem bestimmten Zustand befindet. Das Beispiel basiert auf der Vorlage "Klassendiagramm-DSL", und ClassShape und ModelClass sind in der DSL definierte Typen:
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
} } } }
Die folgenden Fragmente sind häufig in OnStatus-Methoden nützlich:
this.CurrentSelection
. Die Form, auf die der Benutzer mit der rechten Maustaste geklickt hat, wird immer in diese Liste aufgenommen. Wenn der Benutzer auf einen leeren Bereich des Diagramms klickt, ist das Diagramm das einzige Mitglied der Liste.this.IsDiagramSelected()
-true
, wenn Benutzer*innen auf einen leeren Bereich des Diagramms geklickt haben.this.IsCurrentDiagramEmpty()
this.IsSingleSelection()
: Benutzer*innen haben nur ein einzelnes Objekt ausgewählt.this.SingleSelection
: Die Form oder das Diagramm, auf die Benutzer*innen mit der rechten Maustaste geklickt haben.shape.ModelElement as MyLanguageElement
: Das Modellelement, das durch eine Form dargestellt wird
Generell sollten Sie die Visible
-Eigenschaft davon abhängig machen, was ausgewählt ist, und die Enabled
-Eigenschaft vom Zustand der ausgewählten Elemente abhängig machen.
Eine OnStatus-Methode sollte nicht den Zustand des Speichers ändern.
Definieren der Aktionen des Befehls
Definieren Sie für jeden Befehl eine OnMenu...
-Methode, die die erforderliche Aktion ausführt, wenn der Benutzer auf den Menübefehl klickt.
Falls Sie Modellelemente ändern, müssen Sie dies in einer Transaktion tun. Weitere Informationen finden Sie unter Vorgehensweise: Ändern eines Standardmenübefehls.
Im Beispiel sind ClassShape
, ModelClass
und Comment
Typen, die in der DSL definiert sind, die von der Klassendiagramm-DSL-Vorlage abgeleitet ist.
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!
}
}
Weitere Informationen zur Navigation zwischen Objekten im Modell sowie zur Erstellung von Objekten und Links finden Sie unter Vorgehensweise: Ändern eines Standardmenübefehls.
Registrieren des Befehls
Wiederholen Sie in C# die Deklarationen der GUID- und ID-Werte, die Sie im Abschnitt "Symbols" von CommandSet.vsct ausgeführt haben:
private Guid guidCustomMenuCmdSet =
new Guid("00000000-0000-0000-0000-000000000000");
private const int grpidMyMenuGroup = 0x01001;
private const int cmdidMyContextMenuCommand = 1;
Verwenden Sie den gleichen GUID-Wert, den Sie in Commands.vsct eingefügt haben.
Hinweis
Wenn Sie den Abschnitt "Symbols" in der VSCT-Datei ändern, müssen Sie auch diese Deklarationen entsprechend ändern. Außerdem sollten Sie die Versionsnummer in Package.tt erhöhen.
Registrieren Sie Ihre Menübefehle im Rahmen dieses Befehlssatzes. GetMenuCommands()
wird einmalig bei der Initialisierung des Diagramms aufgerufen:
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;
}
Testen des Befehls
Erstellen Sie die DSL, und führen Sie sie in einer experimentellen Instanz von Visual Studio aus. Der Befehl sollte in den von Ihnen angegebenen Situationen im Kontextmenü erscheinen.
So führen Sie den Befehl aus
Klicken Sie in der Symbolleiste im Projektmappen-Explorer auf Alle Vorlagen transformieren.
Drücken Sie F5, um die Projektmappe neu zu erstellen, und beginnen Sie mit dem Debuggen der domänenspezifischen Sprache im experimentellen Build.
Öffnen Sie im experimentellen Build ein Beispieldiagramm.
Klicken Sie mit der rechten Maustaste auf verschiedene Elemente im Diagramm, um zu überprüfen, ob der Befehl richtig aktiviert bzw. deaktiviert und je nach ausgewähltem Element angezeigt oder ausgeblendet wird.
Problembehandlung
Der Befehl wird nicht im Menü angezeigt:
Der Befehl wird nur in Debuginstanzen von Visual Studio angezeigt, bis Sie das DSL-Paket installiert haben. Weitere Informationen finden Sie unter Deploying Domain-Specific Language Solutions (Bereitstellen von Projektmappen für eine domänenspezifische Sprache).
Stellen Sie sicher, dass Ihr experimentelles Beispiel die richtige Dateinamenerweiterung für diese DSL hat. Öffnen Sie "DslDefinition.dsl" in der Hauptinstanz von Visual Studio, um die Dateinamenerweiterung zu überprüfen. Klicken Sie dann in DSL Explorer mit der rechten Maustaste auf den Editor-Knoten, und klicken Sie auf "Eigenschaften". Untersuchen Sie im Eigenschaftenfenster die Eigenschaft "FileExtension".
Haben Sie die Paketversionsnummer erhöht?
Setzen Sie am Anfang der OnStatus-Methode einen Haltepunkt. Er sollte die Verarbeitung anhalten, wenn Sie mit der rechten Maustaste auf einen Bereich des Diagramms klicken.
Die OnStatus-Methode wird nicht aufgerufen:
Vergewissern Sie sich, dass die GUIDs und IDs in Ihrem CommandSet-Code mit denen im Abschnitt "Symbols" von Commands.vsct übereinstimmen.
Überprüfen Sie in Commands.vsct, ob die GUID und ID in jedem Parent-Knoten die richtige übergeordnete Gruppe identifizieren.
Geben Sie an einer Visual Studio-Eingabeaufforderung Folgendes ein: devenv /rootsuffix exp /setup. Starten Sie dann die Debuginstanz von Visual Studio neu.
Durchlaufen Sie die OnStatus-Methode schrittweise, um zu überprüfen, ob "command.Visible" und "command.Enabled" auf True festgelegt sind.
Falscher Menütext wird angezeigt, oder der Befehl erscheint an der falschen Stelle:
Stellen Sie sicher, dass die Kombination von GUID und ID für diesen Befehl eindeutig ist.
Stellen Sie sicher, dass frühere Versionen des Pakets deinstalliert wurden.
Zugehöriger Inhalt
- Schreiben von Code zum Anpassen einer domänenspezifischen Sprache
- Vorgehensweise: Ändern eines Standardmenübefehls
- MSI- und VSIX-Bereitstellung einer DSL
- Beispielcode: Leitungsdiagramme
Hinweis
Die Komponente Textvorlagentransformation wird automatisch als Teil der Workload Visual Studio-Erweiterungsentwicklung installiert. Sie können die Installation auch über die Registerkarte Einzelne Komponenten des Visual Studio-Installers unter der Kategorie SDKs, Bibliotheken und Frameworks durchführen. Installieren Sie die Komponente Modellierungs-SDK auf der Registerkarte Einzelne Komponenten.