/*
Copyright (C) 2014-2019 de4dot@gmail.com
This file is part of dnSpy
dnSpy is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
dnSpy is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with dnSpy. If not, see .
*/
using System;
using System.Collections.Generic;
using System.ComponentModel.Composition;
using System.Diagnostics;
using System.Windows.Input;
using dnSpy.AsmEditor.Commands;
using dnSpy.AsmEditor.MethodBody;
using dnSpy.AsmEditor.Properties;
using dnSpy.AsmEditor.UndoRedo;
using dnSpy.Contracts.App;
using dnSpy.Contracts.AsmEditor.Compiler;
using dnSpy.Contracts.Controls;
using dnSpy.Contracts.Decompiler;
using dnSpy.Contracts.Documents.Tabs;
using dnSpy.Contracts.Documents.Tabs.DocViewer;
using dnSpy.Contracts.Documents.TreeView;
using dnSpy.Contracts.Extension;
using dnSpy.Contracts.Images;
using dnSpy.Contracts.Menus;
using dnSpy.Decompiler.Utils;
namespace dnSpy.AsmEditor.Compiler {
[ExportAutoLoaded]
sealed class CommandLoader : IAutoLoaded {
static readonly RoutedCommand EditBodyCommand = new RoutedCommand("EditBodyCommand", typeof(CommandLoader));
[ImportingConstructor]
CommandLoader(IWpfCommandService wpfCommandService, EditBodyCommand editBodyCmd) {
var cmds = wpfCommandService.GetCommands(ControlConstants.GUID_DOCUMENTVIEWER_UICONTEXT);
ICommand editBodyCmd2 = editBodyCmd;
cmds.Add(EditBodyCommand,
(s, e) => editBodyCmd2.Execute(null),
(s, e) => e.CanExecute = editBodyCmd2.CanExecute(null),
ModifierKeys.Control | ModifierKeys.Shift, Key.E);
}
}
[DebuggerDisplay("{Description}")]
sealed class EditMethodBodyCodeCommand : EditCodeCommandBase {
[ExportMenuItem(Group = MenuConstants.GROUP_CTX_DOCUMENTS_ASMED_ILED, Order = 10)]
sealed class DocumentsCommand : DocumentsContextMenuHandler {
readonly Lazy undoCommandService;
readonly Lazy addUpdatedNodesHelperProvider;
readonly IAppService appService;
readonly EditCodeVMCreator editCodeVMCreator;
[ImportingConstructor]
DocumentsCommand(Lazy undoCommandService, Lazy addUpdatedNodesHelperProvider, IAppService appService, EditCodeVMCreator editCodeVMCreator) {
this.undoCommandService = undoCommandService;
this.addUpdatedNodesHelperProvider = addUpdatedNodesHelperProvider;
this.appService = appService;
this.editCodeVMCreator = editCodeVMCreator;
}
public override ImageReference? GetIcon(AsmEditorContext context) => editCodeVMCreator.GetIcon(CompilationKind.EditMethod);
public override string? GetHeader(AsmEditorContext context) => editCodeVMCreator.GetHeader(CompilationKind.EditMethod);
public override bool IsVisible(AsmEditorContext context) => EditMethodBodyCodeCommand.CanExecute(editCodeVMCreator, context.Nodes);
public override void Execute(AsmEditorContext context) => EditMethodBodyCodeCommand.Execute(editCodeVMCreator, addUpdatedNodesHelperProvider, undoCommandService, appService, context.Nodes);
}
[ExportMenuItem(OwnerGuid = MenuConstants.APP_MENU_EDIT_GUID, Group = MenuConstants.GROUP_APP_MENU_EDIT_ASMED_SETTINGS, Order = 40)]
sealed class EditMenuCommand : EditMenuHandler {
readonly Lazy undoCommandService;
readonly Lazy addUpdatedNodesHelperProvider;
readonly IAppService appService;
readonly EditCodeVMCreator editCodeVMCreator;
[ImportingConstructor]
EditMenuCommand(Lazy undoCommandService, Lazy addUpdatedNodesHelperProvider, IAppService appService, EditCodeVMCreator editCodeVMCreator)
: base(appService.DocumentTreeView) {
this.undoCommandService = undoCommandService;
this.addUpdatedNodesHelperProvider = addUpdatedNodesHelperProvider;
this.appService = appService;
this.editCodeVMCreator = editCodeVMCreator;
}
public override ImageReference? GetIcon(AsmEditorContext context) => editCodeVMCreator.GetIcon(CompilationKind.EditMethod);
public override string? GetHeader(AsmEditorContext context) => editCodeVMCreator.GetHeader(CompilationKind.EditMethod);
public override bool IsVisible(AsmEditorContext context) => EditMethodBodyCodeCommand.CanExecute(editCodeVMCreator, context.Nodes);
public override void Execute(AsmEditorContext context) => EditMethodBodyCodeCommand.Execute(editCodeVMCreator, addUpdatedNodesHelperProvider, undoCommandService, appService, context.Nodes);
}
[ExportMenuItem(Group = MenuConstants.GROUP_CTX_DOCVIEWER_ASMED_ILED, Order = 10)]
sealed class CodeCommand : CodeContextMenuHandler {
readonly Lazy undoCommandService;
readonly Lazy addUpdatedNodesHelperProvider;
readonly IAppService appService;
readonly EditCodeVMCreator editCodeVMCreator;
[ImportingConstructor]
CodeCommand(Lazy undoCommandService, Lazy addUpdatedNodesHelperProvider, IAppService appService, EditCodeVMCreator editCodeVMCreator)
: base(appService.DocumentTreeView) {
this.undoCommandService = undoCommandService;
this.addUpdatedNodesHelperProvider = addUpdatedNodesHelperProvider;
this.appService = appService;
this.editCodeVMCreator = editCodeVMCreator;
}
public override ImageReference? GetIcon(CodeContext context) => editCodeVMCreator.GetIcon(CompilationKind.EditMethod);
public override string? GetHeader(CodeContext context) => editCodeVMCreator.GetHeader(CompilationKind.EditMethod);
public override bool IsEnabled(CodeContext context) => !EditBodyCommand.IsVisibleInternal(editCodeVMCreator, context.MenuItemContext) && context.IsDefinition && EditMethodBodyCodeCommand.CanExecute(editCodeVMCreator, context.Nodes);
public override void Execute(CodeContext context) => EditMethodBodyCodeCommand.Execute(editCodeVMCreator, addUpdatedNodesHelperProvider, undoCommandService, appService, context.Nodes);
}
static bool CanExecute(EditCodeVMCreator editCodeVMCreator, DocumentTreeNodeData[] nodes) =>
editCodeVMCreator.CanCreate(CompilationKind.EditMethod) && nodes.Length == 1 && nodes[0] is MethodNode;
internal static void Execute(EditCodeVMCreator editCodeVMCreator, Lazy addUpdatedNodesHelperProvider, Lazy undoCommandService, IAppService appService, DocumentTreeNodeData[] nodes, IList? statements = null) {
if (!CanExecute(editCodeVMCreator, nodes))
return;
var methodNode = (MethodNode)nodes[0];
var modNode = methodNode.GetModuleNode();
Debug2.Assert(modNode is not null);
if (modNode is null)
throw new InvalidOperationException();
var module = modNode.Document.ModuleDef;
Debug2.Assert(module is not null);
if (module is null)
throw new InvalidOperationException();
using (var vm = editCodeVMCreator.CreateEditMethodCode(methodNode.MethodDef, statements ?? Array.Empty())) {
var win = new EditCodeDlg();
win.DataContext = vm;
win.Owner = appService.MainWindow;
win.Title = $"{win.Title} - {methodNode.ToString()}";
if (win.ShowDialog() != true)
return;
Debug2.Assert(vm.Result is not null);
undoCommandService.Value.Add(new EditMethodBodyCodeCommand(addUpdatedNodesHelperProvider, modNode, vm.Result));
}
}
EditMethodBodyCodeCommand(Lazy addUpdatedNodesHelperProvider, ModuleDocumentNode modNode, ModuleImporter importer)
: base(addUpdatedNodesHelperProvider, modNode, importer) {
}
public override string Description => dnSpy_AsmEditor_Resources.EditMethodCode;
}
[Export, ExportMenuItem(InputGestureText = "res:ShortCutKeyCtrlShiftE", Group = MenuConstants.GROUP_CTX_DOCVIEWER_ASMED_ILED, Order = 0)]
sealed class EditBodyCommand : MenuItemBase, ICommand {
readonly Lazy undoCommandService;
readonly Lazy addUpdatedNodesHelperProvider;
readonly IAppService appService;
readonly EditCodeVMCreator editCodeVMCreator;
[ImportingConstructor]
EditBodyCommand(Lazy undoCommandService, Lazy addUpdatedNodesHelperProvider, IAppService appService, EditCodeVMCreator editCodeVMCreator) {
this.undoCommandService = undoCommandService;
this.addUpdatedNodesHelperProvider = addUpdatedNodesHelperProvider;
this.appService = appService;
this.editCodeVMCreator = editCodeVMCreator;
}
public override ImageReference? GetIcon(IMenuItemContext context) => editCodeVMCreator.GetIcon(CompilationKind.EditMethod);
public override string? GetHeader(IMenuItemContext context) => editCodeVMCreator.GetHeader(CompilationKind.EditMethod);
public override bool IsVisible(IMenuItemContext context) => IsVisibleInternal(editCodeVMCreator, context);
internal static bool IsVisibleInternal(EditCodeVMCreator editCodeVMCreator, IMenuItemContext? context) => IsVisible(editCodeVMCreator, BodyCommandUtils.GetStatements(context, FindByTextPositionOptions.OuterMostStatement));
static bool IsVisible(EditCodeVMCreator editCodeVMCreator, IList? list) =>
editCodeVMCreator.CanCreate(CompilationKind.EditMethod) &&
list is not null &&
list.Count != 0 &&
list[0].Method.Body is not null &&
list[0].Method.Body.Instructions.Count > 0;
public override void Execute(IMenuItemContext context) => Execute(BodyCommandUtils.GetStatements(context, FindByTextPositionOptions.OuterMostStatement));
void Execute(IList? list) {
if (list is null)
return;
var method = list[0].Method;
if (StateMachineHelpers.TryGetKickoffMethod(method, out var containingMethod))
method = containingMethod;
var methodNode = appService.DocumentTreeView.FindNode(method);
if (methodNode is null) {
MsgBox.Instance.Show(string.Format(dnSpy_AsmEditor_Resources.Error_CouldNotFindMethod, method));
return;
}
EditMethodBodyCodeCommand.Execute(editCodeVMCreator, addUpdatedNodesHelperProvider, undoCommandService, appService, new DocumentTreeNodeData[] { methodNode }, list);
}
event EventHandler? ICommand.CanExecuteChanged {
add => CommandManager.RequerySuggested += value;
remove => CommandManager.RequerySuggested -= value;
}
IList? GetStatements() {
var documentViewer = appService.DocumentTabService.ActiveTab.TryGetDocumentViewer();
if (documentViewer is null)
return null;
if (!documentViewer.UIObject.IsKeyboardFocusWithin)
return null;
return BodyCommandUtils.GetStatements(documentViewer, documentViewer.Caret.Position.BufferPosition.Position, FindByTextPositionOptions.OuterMostStatement);
}
void ICommand.Execute(object? parameter) => Execute(GetStatements());
bool ICommand.CanExecute(object? parameter) => IsVisible(editCodeVMCreator, GetStatements());
}
}