/* 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.Collections.Generic; using System.Threading.Tasks; using dnlib.DotNet; using dnSpy.Contracts.AsmEditor.Compiler; using dnSpy.Contracts.Decompiler; namespace dnSpy.AsmEditor.Compiler { sealed class EditMethodCodeVM : EditCodeVM { readonly MethodDef methodToEdit; readonly MethodSourceStatement? methodSourceStatement; sealed class EditMethodDecompileCodeState : DecompileCodeState { public ReferenceDecompilerOutput MainOutput { get; } public StringBuilderDecompilerOutput HiddenOutput { get; } public EditMethodDecompileCodeState(object referenceToEdit, MethodSourceStatement? methodSourceStatement) { MainOutput = new ReferenceDecompilerOutput(referenceToEdit, methodSourceStatement); HiddenOutput = new StringBuilderDecompilerOutput(); } } public EditMethodCodeVM(EditCodeVMOptions options, MethodDef methodToEdit, IList statementsInMethodToEdit) : base(options, methodToEdit.DeclaringType) { this.methodToEdit = methodToEdit; methodSourceStatement = statementsInMethodToEdit.Count == 0 ? (MethodSourceStatement?)null : statementsInMethodToEdit[0]; StartDecompile(); } protected override DecompileCodeState CreateDecompileCodeState() => new EditMethodDecompileCodeState(methodToEdit, methodSourceStatement); protected override Task DecompileAsync(DecompileCodeState decompileCodeState) { var state = (EditMethodDecompileCodeState)decompileCodeState; state.CancellationToken.ThrowIfCancellationRequested(); var type = methodToEdit.DeclaringType; while (type.DeclaringType is not null) type = type.DeclaringType; DecompileTypeMethods options; state.DecompilationContext.CalculateILSpans = true; options = new DecompileTypeMethods(state.MainOutput, state.DecompilationContext, type); options.Methods.Add(methodToEdit); options.DecompileHidden = false; decompiler.Decompile(DecompilationType.TypeMethods, options); state.CancellationToken.ThrowIfCancellationRequested(); state.DecompilationContext.CalculateILSpans = false; options = new DecompileTypeMethods(state.HiddenOutput, state.DecompilationContext, type); options.Methods.Add(methodToEdit); options.DecompileHidden = true; decompiler.Decompile(DecompilationType.TypeMethods, options); state.CancellationToken.ThrowIfCancellationRequested(); var result = new DecompileAsyncResult(); result.AddDocument(MainCodeName, state.MainOutput.ToString(), state.MainOutput.Span); result.AddDocument(MainGeneratedCodeName, state.HiddenOutput.ToString(), null); return Task.FromResult(result); } protected override void Import(ModuleImporter importer, CompilationResult result) => importer.Import(result.RawFile!, result.DebugFile, methodToEdit); } }