Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Codeactions with options/ui. #1406

Merged
merged 58 commits into from
Jul 10, 2019
Merged
Show file tree
Hide file tree
Changes from 18 commits
Commits
Show all changes
58 commits
Select commit Hold shift + click to select a range
eea9dfc
Initial version of actions with options.
savpek Feb 20, 2019
cbf1877
Fixes.
savpek Feb 20, 2019
70d3dc9
Fixes for extract interface.
savpek Feb 21, 2019
9d95a01
First test.
savpek Feb 21, 2019
84224b5
More tests.
savpek Feb 21, 2019
5437959
Tweaks and comments.
savpek Feb 21, 2019
761f657
Original global.json
savpek Feb 21, 2019
274e3c7
Merge branch 'master' into feature/codeactions-with-options
savpek Feb 23, 2019
ca9ecb3
Blacklisting.
savpek Feb 23, 2019
a7c21ff
Merge branch 'feature/codeactions-with-options' of https://github.com…
savpek Feb 23, 2019
8a298e5
Added test to keep track is official support added.
savpek Feb 24, 2019
c773575
Tests for blacklisting.
savpek Feb 24, 2019
f17aeeb
small comment tweak.
savpek Feb 24, 2019
9dd4cef
Merge branch 'master' into feature/codeactions-with-options
savpek Feb 27, 2019
2a199b1
Merge branch 'master' into feature/codeactions-with-options
savpek Mar 2, 2019
fca2bb4
Merge branch 'master' into feature/codeactions-with-options
savpek Mar 13, 2019
17eb6d9
Merge branch 'master' into feature/codeactions-with-options
savpek Mar 16, 2019
ac93122
Merge branch 'master' into feature/codeactions-with-options
savpek Mar 17, 2019
4659787
Moved to dispatch proxy.
savpek Mar 20, 2019
596cbe6
Merge from upstream/master
savpek Mar 20, 2019
5690dc1
Fixes for extract interface test.
savpek Mar 20, 2019
401e66e
Removed debug line which was forgotten.
savpek Mar 20, 2019
4a65088
Merge branch 'master' into feature/codeactions-with-options
savpek Mar 21, 2019
9efd76d
Merge branch 'master' into feature/codeactions-with-options
savpek Mar 22, 2019
f1a6061
Few review fixes.
savpek Mar 22, 2019
a06b47b
Merge branch 'feature/codeactions-with-options' of https://github.com…
savpek Mar 22, 2019
f136bf1
Refactored action facts to base class.
savpek Mar 22, 2019
a04812a
Merge branch 'master' into feature/codeactions-with-options
savpek Apr 1, 2019
0332ef6
Removed multiple reflection calls.
savpek Apr 3, 2019
1f47071
Merge branch 'feature/codeactions-with-options' of https://github.com…
savpek Apr 3, 2019
e60dc68
Merge branch 'master' into feature/codeactions-with-options
savpek Apr 3, 2019
1aeab24
Merge branch 'master' into feature/codeactions-with-options
savpek Apr 8, 2019
963c708
Attempt to fix test with net 472 hash code genenerator function.
savpek Apr 8, 2019
515f691
Additional fix.
savpek Apr 8, 2019
c6a3683
Trigger build.
savpek Apr 14, 2019
57543fc
Testfix to support multiple hashcode implementations per frameworks.
savpek Apr 14, 2019
2afb64a
Merge branch 'master' into feature/codeactions-with-options
savpek Apr 14, 2019
9949e56
Merge branch 'master' into feature/codeactions-with-options
savpek Apr 15, 2019
1138ac8
Merge branch 'master' into feature/codeactions-with-options
savpek Apr 18, 2019
79e0390
Merge branch 'master' into feature/codeactions-with-options
savpek Apr 19, 2019
7099e22
Merge branch 'master' into feature/codeactions-with-options
savpek May 1, 2019
26aaac9
Merge branch 'master' into feature/codeactions-with-options
savpek May 8, 2019
9b9ac63
Merge branch 'master' into feature/codeactions-with-options
savpek May 20, 2019
0471c2e
Merge branch 'master' into feature/codeactions-with-options
savpek May 21, 2019
dab7a8c
Merge branch 'master' into feature/codeactions-with-options
savpek May 23, 2019
03c31f6
Merge branch 'master' into feature/codeactions-with-options
savpek May 26, 2019
a463572
Merge branch 'master' into feature/codeactions-with-options
savpek May 31, 2019
bbc31ad
Merge branch 'master' into feature/codeactions-with-options
filipw Jun 11, 2019
2c13068
Merge branch 'master' into feature/codeactions-with-options
savpek Jun 20, 2019
90c7d93
Merge.
savpek Jun 23, 2019
fbf7d36
Testfix.
savpek Jun 23, 2019
49c01c1
Merge branch 'master' into feature/codeactions-with-options
savpek Jun 25, 2019
281ad00
Merge branch 'master' into feature/codeactions-with-options
savpek Jun 27, 2019
22e1789
Merge branch 'master' into feature/codeactions-with-options
david-driscoll Jun 27, 2019
672a012
Merge branch 'master' into feature/codeactions-with-options
savpek Jun 28, 2019
c8085c0
Merge branch 'master' into feature/codeactions-with-options
filipw Jul 5, 2019
67277c1
Merge branch 'master' into feature/codeactions-with-options
savpek Jul 7, 2019
01a5aca
Merge branch 'master' into feature/codeactions-with-options
savpek Jul 10, 2019
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion src/OmniSharp.Host/WorkspaceInitializer.cs
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ public static void Initialize(IServiceProvider serviceProvider, CompositionHost
var projectEventForwarder = compositionHost.GetExport<ProjectEventForwarder>();
projectEventForwarder.Initialize();
var projectSystems = compositionHost.GetExports<IProjectSystem>();

foreach (var projectSystem in projectSystems)
{
try
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -77,13 +77,19 @@ protected async Task<IEnumerable<AvailableCodeAction>> GetAvailableCodeActions(I
await CollectCodeFixesActions(document, span, codeActions);
await CollectRefactoringActions(document, span, codeActions);

// Be sure to filter out any code actions that inherit from CodeActionWithOptions.
// This isn't a great solution and might need changing later, but every Roslyn code action
// derived from this type tries to display a dialog. For now, this is a reasonable solution.
var availableActions = ConvertToAvailableCodeAction(codeActions)
.Where(a => !a.CodeAction.GetType().GetTypeInfo().IsSubclassOf(typeof(CodeActionWithOptions)));
var availableActions = ConvertToAvailableCodeAction(codeActions);

return availableActions;
return FilterBlacklistedCodeActions(availableActions);
}

private static IEnumerable<AvailableCodeAction> FilterBlacklistedCodeActions(IEnumerable<AvailableCodeAction> codeActions)
{
// Most of actions with UI works fine with defaults, however there's few exceptions:
return codeActions.Where(x =>
x.CodeAction.Title != "Generate new type..." && // Blacklisted because doesn't give additional value over non UI generate type (when defaults used.)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

let's do the blacklisting based on the original Roslyn type name instead of the title text

x.CodeAction.Title != "Change signature..." && // Blacklisted because cannot be used without proper UI.
x.CodeAction.Title != "Pull members up to base type..." // Blacklisted because doesn't give additional value over non UI generate type (when defaults used.)
);
}

private TextSpan GetTextSpan(ICodeActionRequest request, SourceText sourceText)
Expand Down
2 changes: 2 additions & 0 deletions src/OmniSharp.Roslyn/HostServicesAggregator.cs
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,8 @@ public HostServicesAggregator(
}
}

builder.Add(typeof(PickMemberWorkspaceService).Assembly);

_assemblies = builder.ToImmutableArray();
}

Expand Down
1 change: 1 addition & 0 deletions src/OmniSharp.Roslyn/OmniSharp.Roslyn.csproj
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
<PackageReference Include="Microsoft.Extensions.Caching.Memory" />
<PackageReference Include="Microsoft.CodeAnalysis.Common" />
<PackageReference Include="System.ComponentModel.Composition" />
<PackageReference Include="Castle.Core" Version="4.3.1" />
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We probably want to avoid a dependency for Castle.Core?

</ItemGroup>

</Project>
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
using System;
using System.Composition;
using System.Reflection;
using Microsoft.CodeAnalysis.Host.Mef;

namespace OmniSharp
{
[MetadataAttribute]
[AttributeUsage(AttributeTargets.Class)]
public class ExportWorkspaceServiceFactoryWithAssemblyQualifiedNameAttribute : ExportAttribute
{
public string ServiceType { get; }
public string Layer { get; }

// Theres builtin public attribute for this, but since we target internal types
// this is needed to build service. MEF doesn't care is it internal or not.
public ExportWorkspaceServiceFactoryWithAssemblyQualifiedNameAttribute(string typeAssembly, string typeName, string layer = ServiceLayer.Host)
: base(typeof(IWorkspaceServiceFactory))
{
var type = Assembly.Load(typeAssembly).GetType(typeName)
?? throw new InvalidOperationException($"Could not resolve '{typeName} from '{typeAssembly}'");

Console.WriteLine($"Resolved to type: {type.AssemblyQualifiedName}");
this.ServiceType = type.AssemblyQualifiedName;
this.Layer = layer ?? throw new ArgumentNullException(nameof(layer));
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
using System;
using System.Collections.Generic;
using System.Collections.Immutable;
using System.Linq;
using System.Reflection;
using System.Threading.Tasks;
using Castle.DynamicProxy;
using Microsoft.CodeAnalysis;

namespace OmniSharp
{
public class ExtractInterfaceWorkspaceService : IInterceptor
{
public void Intercept(IInvocation invocation)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To avoid a dependency on Castle.Core, you can have a look in MetadataHelper on how this could be done.

{
// IExtractInterfaceOptionsService and extract interface results are internal types -> workaround with proxy.
// This service simply passes all members through as selected and doesn't try show UI.
// When roslyn exposes this interface and members -> remove this workaround.

var resultTypeInternal = Assembly.Load("Microsoft.CodeAnalysis.Features").GetType("Microsoft.CodeAnalysis.ExtractInterface.ExtractInterfaceOptionsResult");
var enumType = resultTypeInternal.GetNestedTypes().Single(x => x.Name == "ExtractLocation");

var toSameFileEnumValue = Enum.Parse(enumType, "SameFile");

var interfaceName = invocation.Arguments[3] ?? throw new InvalidOperationException($"{nameof(ExtractInterfaceWorkspaceService)} default interface name was null.");

var resultObject = Activator.CreateInstance(resultTypeInternal, new object[] {
false, // isCancelled
((List<ISymbol>)invocation.Arguments[2]).ToImmutableArray(), // InterfaceMembers selected -> select all.
interfaceName,
$"{interfaceName}.cs",
toSameFileEnumValue
});

var fromResultMethod = typeof(Task).GetMethod("FromResult").MakeGenericMethod(resultTypeInternal).Invoke(null, new[] { resultObject });
invocation.ReturnValue = fromResultMethod;
}

private static Type MakeGenerictTaskTypeFromResult(Type resultTypeInternal)
{
var taskType = typeof(Task<>);
Type[] typeArgs = { resultTypeInternal };
var taskGenericType = taskType.MakeGenericType(typeArgs);
return taskGenericType;
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
using System.Composition;
using System.Reflection;
using Castle.DynamicProxy;
using Microsoft.CodeAnalysis.Host;
using Microsoft.CodeAnalysis.Host.Mef;

namespace OmniSharp
{
[Shared]
[ExportWorkspaceServiceFactoryWithAssemblyQualifiedName("Microsoft.CodeAnalysis.Features", "Microsoft.CodeAnalysis.ExtractInterface.IExtractInterfaceOptionsService")]
public class ExtractInterfaceWorkspaceServiceFactory : IWorkspaceServiceFactory
{
public IWorkspaceService CreateService(HostWorkspaceServices workspaceServices)
{
// Generates proxy class to get around issue that IExtractInterfaceOptionsService is internal at this point.
ProxyGenerator generator = new ProxyGenerator();
var internalType = Assembly.Load("Microsoft.CodeAnalysis.Features").GetType("Microsoft.CodeAnalysis.ExtractInterface.IExtractInterfaceOptionsService");
return (IWorkspaceService)generator.CreateInterfaceProxyWithoutTarget(internalType, new[] { typeof(IWorkspaceService)}, new ExtractInterfaceWorkspaceService());
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
using System;
using System.Reflection;
using Castle.DynamicProxy;

namespace OmniSharp
{
public class PickMemberWorkspaceService : IInterceptor
{
public void Intercept(IInvocation invocation)
{
// IPickMember and PickMemberResults are internal types -> workaround with proxy.
// This service simply passes all members through as selected and doesn't try show UI.
// When roslyn exposes this interface and members -> remove this workaround.
var resultTypeInternal = Assembly.Load("Microsoft.CodeAnalysis.Features").GetType("Microsoft.CodeAnalysis.PickMembers.PickMembersResult");
var resultInstance = Activator.CreateInstance(resultTypeInternal, new object[] { invocation.Arguments[1], invocation.Arguments[2] });
invocation.ReturnValue = resultInstance;
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
using System.Composition;
using System.Reflection;
using Castle.DynamicProxy;
using Microsoft.CodeAnalysis.Host;
using Microsoft.CodeAnalysis.Host.Mef;

namespace OmniSharp
{
[Shared]
[ExportWorkspaceServiceFactoryWithAssemblyQualifiedName("Microsoft.CodeAnalysis.Features", "Microsoft.CodeAnalysis.PickMembers.IPickMembersService")]
public class PickMemberWorkspaceServiceFactory : IWorkspaceServiceFactory
{
public IWorkspaceService CreateService(HostWorkspaceServices workspaceServices)
{
// Generates proxy class to get around issue that IPickMembersService is internal at this point.
ProxyGenerator generator = new ProxyGenerator();
var internalType = Assembly.Load("Microsoft.CodeAnalysis.Features").GetType("Microsoft.CodeAnalysis.PickMembers.IPickMembersService");
return (IWorkspaceService)generator.CreateInterfaceProxyWithoutTarget(internalType, new[] { typeof(IWorkspaceService)}, new PickMemberWorkspaceService());
}
}
}
Loading