forked from dnSpy/dnSpy
-
-
Notifications
You must be signed in to change notification settings - Fork 530
/
Copy pathLoadAllDependenciesCommand.cs
120 lines (104 loc) · 4.93 KB
/
LoadAllDependenciesCommand.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
using System;
using System.Collections.Generic;
using System.ComponentModel.Composition;
using System.Diagnostics;
using System.IO;
using System.Linq;
using System.Text;
using System.Windows.Threading;
using dnlib.DotNet;
using dnSpy.AsmEditor.Commands;
using dnSpy.AsmEditor.Properties;
using dnSpy.AsmEditor.UndoRedo;
using dnSpy.Contracts.AsmEditor.Compiler;
using dnSpy.Contracts.Documents;
using dnSpy.Contracts.Documents.Tabs;
using dnSpy.Contracts.Documents.TreeView;
using dnSpy.Contracts.Images;
using dnSpy.Contracts.Menus;
using dnSpy.Contracts.MVVM;
using dnSpy.Contracts.Utilities;
namespace dnSpy.AsmEditor.Compiler {
[DebuggerDisplay("{Description}")]
sealed class LoadAllDependenciesCommand : EditCodeCommandBase {
[ExportMenuItem(Header = "res:LoadAllDependenciesCommand", Group = MenuConstants.GROUP_CTX_DOCUMENTS_ASMED_ILED, Order = 29.999)]
sealed class DocumentsCommand : DocumentsContextMenuHandler {
readonly Lazy<IUndoCommandService> undoCommandService;
readonly Lazy<IAddUpdatedNodesHelperProvider> addUpdatedNodesHelperProvider;
readonly IAppService appService;
[ImportingConstructor]
DocumentsCommand(Lazy<IUndoCommandService> undoCommandService, Lazy<IAddUpdatedNodesHelperProvider> addUpdatedNodesHelperProvider, IAppService appService) {
this.undoCommandService = undoCommandService;
this.addUpdatedNodesHelperProvider = addUpdatedNodesHelperProvider;
this.appService = appService;
}
public override bool IsVisible(AsmEditorContext context) => LoadAllDependenciesCommand.CanExecute(context.Nodes);
public override void Execute(AsmEditorContext context) => LoadAllDependenciesCommand.Execute(addUpdatedNodesHelperProvider, undoCommandService, appService, context.Nodes);
}
[ExportMenuItem(OwnerGuid = MenuConstants.APP_MENU_EDIT_GUID, Header = "res:LoadAllDependenciesCommand", Group = MenuConstants.GROUP_APP_MENU_EDIT_ASMED_SETTINGS, Order = 59.999)]
sealed class EditMenuCommand : EditMenuHandler {
readonly Lazy<IUndoCommandService> undoCommandService;
readonly Lazy<IAddUpdatedNodesHelperProvider> addUpdatedNodesHelperProvider;
readonly IAppService appService;
[ImportingConstructor]
EditMenuCommand(Lazy<IUndoCommandService> undoCommandService, Lazy<IAddUpdatedNodesHelperProvider> addUpdatedNodesHelperProvider, IAppService appService)
: base(appService.DocumentTreeView) {
this.undoCommandService = undoCommandService;
this.addUpdatedNodesHelperProvider = addUpdatedNodesHelperProvider;
this.appService = appService;
}
public override bool IsVisible(AsmEditorContext context) => LoadAllDependenciesCommand.CanExecute(context.Nodes);
public override void Execute(AsmEditorContext context) => LoadAllDependenciesCommand.Execute(addUpdatedNodesHelperProvider, undoCommandService, appService, context.Nodes);
}
static bool CanExecute(DocumentTreeNodeData[] nodes) => nodes.Length == 1 && GetModuleNode(nodes[0]) is not null;
static ModuleDocumentNode? GetModuleNode(DocumentTreeNodeData node) {
if (node is AssemblyDocumentNode asmNode) {
asmNode.TreeNode.EnsureChildrenLoaded();
return asmNode.TreeNode.DataChildren.FirstOrDefault() as ModuleDocumentNode;
}
else
return node.GetModuleNode();
}
static void Execute(Lazy<IAddUpdatedNodesHelperProvider> addUpdatedNodesHelperProvider, Lazy<IUndoCommandService> undoCommandService, IAppService appService, DocumentTreeNodeData[] nodes) {
if (!CanExecute(nodes))
return;
var modNode = GetModuleNode(nodes[0]);
Debug2.Assert(modNode is not null);
if (modNode is null)
return;
var module = modNode.Document.ModuleDef;
Debug2.Assert(module is not null);
if (module is null)
throw new InvalidOperationException();
var loadeds = new HashSet<ModuleDef>();
LoadAllDependencies(modNode.Context.DocumentTreeView, module, loadeds);
}
static void LoadAllDependencies(IDocumentTreeView documentTreeView, ModuleDef module, HashSet<ModuleDef> loadeds) {
if (loadeds.Contains(module))
return;
loadeds.Add(module);
foreach (var assemblyRef in module.GetAssemblyRefs()) {
var asm = documentTreeView.DocumentService.Resolve(assemblyRef, module);
if (asm?.ModuleDef is null)
continue;
if (FrameworkFileUtils.IsFrameworkAssembly("", assemblyRef.Name))
continue;
LoadAllDependencies(documentTreeView, asm.ModuleDef, loadeds);
}
}
readonly struct ModuleResult {
public IAssembly? Assembly { get; }
public byte[] RawBytes { get; }
public DebugFileResult DebugFile { get; }
public ModuleResult(IAssembly? assembly, byte[] bytes, DebugFileResult debugFile) {
Assembly = assembly;
RawBytes = bytes;
DebugFile = debugFile;
}
}
LoadAllDependenciesCommand(Lazy<IAddUpdatedNodesHelperProvider> addUpdatedNodesHelperProvider, ModuleDocumentNode modNode, ModuleImporter importer)
: base(addUpdatedNodesHelperProvider, modNode, importer) {
}
public override string Description => dnSpy_AsmEditor_Resources.LoadAllDependenciesCommand;
}
}