Browse Source

[CodeActions] Add MoveToOuterScopeAction.

newNRvisualizers
Simon Lindgren 13 years ago
parent
commit
3ec073f293
  1. 1
      ICSharpCode.NRefactory.CSharp/ICSharpCode.NRefactory.CSharp.csproj
  2. 155
      ICSharpCode.NRefactory.CSharp/Refactoring/CodeActions/MoveToOuterScopeAction.cs
  3. 114
      ICSharpCode.NRefactory.Tests/CSharp/CodeActions/MoveToOuterScopeTests.cs
  4. 1
      ICSharpCode.NRefactory.Tests/ICSharpCode.NRefactory.Tests.csproj

1
ICSharpCode.NRefactory.CSharp/ICSharpCode.NRefactory.CSharp.csproj

@ -386,6 +386,7 @@ @@ -386,6 +386,7 @@
<Compile Include="Refactoring\CodeActions\ConvertToInitializer\StatementsToInitializerConverter.cs" />
<Compile Include="Refactoring\CodeActions\ConvertToInitializer\InitializerPath.cs" />
<Compile Include="Refactoring\CodeIssues\CallToStaticMemberViaDerivedTypeIssue.cs" />
<Compile Include="Refactoring\CodeActions\MoveToOuterScopeAction.cs" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\ICSharpCode.NRefactory\ICSharpCode.NRefactory.csproj">

155
ICSharpCode.NRefactory.CSharp/Refactoring/CodeActions/MoveToOuterScopeAction.cs

@ -0,0 +1,155 @@ @@ -0,0 +1,155 @@
//
// MoveToOuterScopeAction.cs
//
// Author:
// Simon Lindgren <simon.n.lindgren@gmail.com>
//
// Copyright (c) 2012 Simon Lindgren
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
using System.Collections.Generic;
using System;
using ICSharpCode.NRefactory.CSharp.Resolver;
using System.Linq;
using ICSharpCode.NRefactory.Semantics;
using System.Threading;
namespace ICSharpCode.NRefactory.CSharp.Refactoring
{
public class MoveToOuterScopeAction : ICodeActionProvider
{
#region ICodeActionProvider implementation
public IEnumerable<CodeAction> GetActions(RefactoringContext context)
{
var variableDeclaration = context.GetNode<VariableDeclarationStatement>();
if (variableDeclaration == null)
yield break;
var entryNode = FindCurrentScopeEntryNode(variableDeclaration);
if (entryNode == null)
yield break;
var selectedInitializer = context.GetNode<VariableInitializer>();
if (selectedInitializer != null) {
if (HasDependency(context, entryNode, selectedInitializer)) {
yield return MoveDeclarationAction(context, entryNode, variableDeclaration, selectedInitializer);
} else {
yield return MoveInitializerAction(context, entryNode, variableDeclaration, selectedInitializer);
}
} else {
yield return new CodeAction(context.TranslateString("Move declaration to outer scope"), script => {
script.Remove(variableDeclaration);
script.InsertBefore(entryNode, variableDeclaration.Clone());
});
}
}
static CodeAction MoveInitializerAction(RefactoringContext context, AstNode insertAnchor,
VariableDeclarationStatement declaration, VariableInitializer initializer)
{
var type = declaration.Type.Clone();
var name = initializer.Name;
return new CodeAction(context.TranslateString("Move initializer to outer scope"), script => {
if (declaration.Variables.Count != 1) {
var innerDeclaration = RemoveInitializer(declaration, initializer);
script.InsertBefore(declaration, innerDeclaration);
}
script.Remove(declaration);
var outerDeclaration = new VariableDeclarationStatement(type, name, initializer.Initializer.Clone());
script.InsertBefore(insertAnchor, outerDeclaration);
});
}
static CodeAction MoveDeclarationAction(RefactoringContext context, AstNode insertAnchor,
VariableDeclarationStatement declarationStatement, VariableInitializer initializer)
{
var type = declarationStatement.Type.Clone();
var name = initializer.Name;
return new CodeAction(context.TranslateString("Move declaration to outer scope"), script => {
script.InsertBefore(declarationStatement, new ExpressionStatement() {
Expression = new AssignmentExpression(new IdentifierExpression(name), initializer.Initializer.Clone())
});
script.Remove(declarationStatement);
script.InsertBefore(insertAnchor, new VariableDeclarationStatement(type, name, Expression.Null));
});
}
bool HasDependency(RefactoringContext context, AstNode firstSearchNode, AstNode targetNode)
{
var referenceFinder = new FindReferences();
var identifiers = targetNode.Descendants
.Where(n => n is IdentifierExpression)
.Select<AstNode, IdentifierExpression>(node => (IdentifierExpression)node);
foreach (var identifier in identifiers) {
var resolveResult = context.Resolve(identifier);
var localResolveResult = resolveResult as LocalResolveResult;
if (localResolveResult == null)
continue;
bool referenceFound = false;
var variable = localResolveResult.Variable;
var compilationUnit = context.RootNode as CompilationUnit;
referenceFinder.FindLocalReferences(localResolveResult.Variable, context.ParsedFile, compilationUnit,
context.Compilation, (node, nodeResolveResult) => {
if (node.StartLocation > firstSearchNode.StartLocation && node.EndLocation < targetNode.StartLocation)
referenceFound = true;
}, CancellationToken.None);
if (referenceFound)
return true;
}
return false;
}
static VariableDeclarationStatement RemoveInitializer(VariableDeclarationStatement variableDeclarationStatement, VariableInitializer selectedVariableInitializer)
{
var newVariableDeclarationStatement = new VariableDeclarationStatement() {
Type = variableDeclarationStatement.Type.Clone()
};
foreach (var variableInitializer in variableDeclarationStatement.Variables) {
if (variableInitializer != selectedVariableInitializer) {
newVariableDeclarationStatement.AddChild((VariableInitializer)variableInitializer.Clone(), Roles.Variable);
}
}
return newVariableDeclarationStatement;
}
IList<Type> scopeEntryPoints = new List<Type>() {
typeof (IfElseStatement),
typeof (WhileStatement),
typeof (ForeachStatement),
typeof (ForStatement),
typeof (DoWhileStatement),
typeof (UsingStatement)
};
List<Type> scopeContainers = new List<Type>() {
typeof (MethodDeclaration),
typeof (Accessor)
};
AstNode FindCurrentScopeEntryNode(AstNode startNode)
{
var startNodeType = startNode.GetType();
if (scopeEntryPoints.Contains(startNodeType))
return startNode;
if (scopeContainers.Contains(startNodeType))
return null;
return FindCurrentScopeEntryNode(startNode.Parent);
}
#endregion
}
}

114
ICSharpCode.NRefactory.Tests/CSharp/CodeActions/MoveToOuterScopeTests.cs

@ -0,0 +1,114 @@ @@ -0,0 +1,114 @@
//
// MoveToOuterScopeTests.cs
//
// Author:
// Simon Lindgren <simon.n.lindgren@gmail.com>
//
// Copyright (c) 2012 Simon Lindgren
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
using NUnit.Framework;
using ICSharpCode.NRefactory.CSharp.Refactoring;
namespace ICSharpCode.NRefactory.CSharp.CodeActions
{
[TestFixture]
public class MoveToOuterScopeTests : ContextActionTestBase
{
void TestStatements (string input, string output)
{
Test<MoveToOuterScopeAction>(@"
class A
{
void F()
{"
+ input +
@" }
}", @"
class A
{
void F()
{"
+ output +
@" }
}");
}
[Test]
public void SimpleCase()
{
TestStatements(@"
while (true) {
int $i = 2;
}
", @"
int i = 2;
while (true) {
}
");
}
[Test]
public void MovesOnlyTheCurrentVariableInitialization()
{
TestStatements(@"
while (true) {
int $i = 2, j = 3;
}
", @"
int i = 2;
while (true) {
int j = 3;
}
");
}
[Test]
public void MovesAllInitializersWhenOnType()
{
TestStatements(@"
while (true) {
i$nt i = 2, j = 3;
}
", @"
int i = 2, j = 3;
while (true) {
}
");
}
[Test]
public void OnlyMovesDeclarationWhenInitializerDependsOnOtherStatements()
{
TestStatements(@"
while (true) {
int i = 2;
int j$ = i;
}
", @"
int j;
while (true) {
int i = 2;
j = i;
}
");
}
}
}

1
ICSharpCode.NRefactory.Tests/ICSharpCode.NRefactory.Tests.csproj

@ -282,6 +282,7 @@ @@ -282,6 +282,7 @@
<Compile Include="CSharp\Refactoring\NamingHelperTests.cs" />
<Compile Include="CSharp\CodeActions\ConvertToInitializer\ConvertToInitializerTests.cs" />
<Compile Include="CSharp\CodeIssues\CallToStaticMemberViaDerivedTypeTests.cs" />
<Compile Include="CSharp\CodeActions\MoveToOuterScopeTests.cs" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\..\Mono.Cecil\Mono.Cecil.csproj">

Loading…
Cancel
Save