// // ImplementInterfaceAction.cs // // Author: // Mike Krüger // // Copyright (c) 2012 Xamarin Inc. (http://xamarin.com) // // 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; using ICSharpCode.NRefactory.TypeSystem; using System.Threading; using System.Collections.Generic; using System.Linq; namespace ICSharpCode.NRefactory.CSharp.Refactoring { [ContextAction("Implement interface", Description = "Creates an interface implementation.")] public class ImplementInterfaceAction : ICodeActionProvider { public IEnumerable GetActions(RefactoringContext context) { var type = context.GetNode(); if (type == null || type.Role != Roles.BaseType) yield break; var state = context.GetResolverStateBefore(type); if (state.CurrentTypeDefinition == null) yield break; var resolveResult = context.Resolve(type); if (resolveResult.Type.Kind != TypeKind.Interface) yield break; var toImplement = CollectMembersToImplement(state.CurrentTypeDefinition, resolveResult.Type, false); if (toImplement.Count == 0) yield break; yield return new CodeAction(context.TranslateString("Implement interface"), script => { script.InsertWithCursor( context.TranslateString ("Implement Interface"), state.CurrentTypeDefinition, GenerateImplementation (context, toImplement) ); }); } public static IEnumerable GenerateImplementation(RefactoringContext context, IEnumerable> toImplement) { var nodes = new Dictionary>(); foreach (var member in toImplement) { if (!nodes.ContainsKey(member.Item1.DeclaringType)) nodes [member.Item1.DeclaringType] = new List(); nodes [member.Item1.DeclaringType].Add(GenerateMemberImplementation(context, member)); } foreach (var kv in nodes) { yield return new PreProcessorDirective( PreProcessorDirectiveType.Region, string.Format("{0} implementation", kv.Key.Name)); foreach (var member in kv.Value) yield return member; yield return new PreProcessorDirective( PreProcessorDirectiveType.Endregion ); } } static AstNode GenerateMemberImplementation(RefactoringContext context, Tuple member) { switch (member.Item1.EntityType) { case EntityType.Property: return null; case EntityType.Indexer: return null; case EntityType.Event: return null; case EntityType.Method: return GenerateMethod(context, (IMethod)member.Item1, member.Item2); default: throw new ArgumentOutOfRangeException(); } } static AstNode GenerateMethod(RefactoringContext context, IMethod method, bool explicitImplementation) { var result = new MethodDeclaration() { Name = method.Name, ReturnType = context.CreateShortType (method.ReturnType), Body = new BlockStatement() { new ThrowStatement(new ObjectCreateExpression(context.CreateShortType("System", "NotImplementedException"))) } }; if (!explicitImplementation) { result.Modifiers = Modifiers.Public; } else { result.PrivateImplementationType = context.CreateShortType(method.DeclaringType); } foreach (var typeParam in method.TypeParameters) { result.TypeParameters.Add(new TypeParameterDeclaration(typeParam.Name)); var constraint = new Constraint() { TypeParameter = new SimpleType(typeParam.Name) }; if (typeParam.HasDefaultConstructorConstraint) { constraint.BaseTypes.Add (new PrimitiveType("new")); } else if (typeParam.HasReferenceTypeConstraint) { constraint.BaseTypes.Add (new PrimitiveType("class")); } else if (typeParam.HasValueTypeConstraint) { constraint.BaseTypes.Add (new PrimitiveType("struct")); } foreach (var type in typeParam.DirectBaseTypes) { if (type.FullName == "System.Object") continue; if (type.FullName == "System.ValueType") continue; constraint.BaseTypes.Add (context.CreateShortType (type)); } if (constraint.BaseTypes.Count == 0) continue; result.Constraints.Add (constraint); } foreach (var p in method.Parameters) { ParameterModifier modifier; if (p.IsOut) { modifier = ParameterModifier.Out; } else if (p.IsRef) { modifier = ParameterModifier.Ref; } else if (p.IsParams) { modifier = ParameterModifier.Params; } else { modifier = ParameterModifier.None; } result.Parameters.Add(new ParameterDeclaration(context.CreateShortType(p.Type), p.Name, modifier)); } return result; } public static List> CollectMembersToImplement(ITypeDefinition implementingType, IType interfaceType, bool explicitly) { var def = interfaceType.GetDefinition(); List> toImplement = new List>(); bool alreadyImplemented; // Stub out non-implemented events defined by @iface foreach (var ev in interfaceType.GetEvents (e => !e.IsSynthetic && e.DeclaringTypeDefinition.ReflectionName == def.ReflectionName)) { bool needsExplicitly = explicitly; alreadyImplemented = implementingType.GetAllBaseTypeDefinitions().Any( x => x.Kind != TypeKind.Interface && x.Events.Any (y => y.Name == ev.Name) ); if (!alreadyImplemented) toImplement.Add(new Tuple(ev, needsExplicitly)); } Console.WriteLine("foo!!!!"); // Stub out non-implemented methods defined by @iface foreach (var method in interfaceType.GetMethods (d => !d.IsSynthetic /* && d.DeclaringTypeDefinition.ReflectionName == def.ReflectionName*/)) { bool needsExplicitly = explicitly; alreadyImplemented = false; foreach (var cmet in implementingType.GetMethods ()) { if (CompareMethods(method, cmet)) { if (!needsExplicitly && !cmet.ReturnType.Equals(method.ReturnType)) needsExplicitly = true; else alreadyImplemented |= !needsExplicitly /*|| cmet.InterfaceImplementations.Any (impl => impl.InterfaceType.Equals (interfaceType))*/; } } if (toImplement.Where (t => t.Item1 is IMethod).Any (t => CompareMethods (method, (IMethod)t.Item1))) needsExplicitly = true; if (!alreadyImplemented) toImplement.Add(new Tuple(method, needsExplicitly)); } // Stub out non-implemented properties defined by @iface foreach (var prop in interfaceType.GetProperties (p => !p.IsSynthetic && p.DeclaringTypeDefinition.ReflectionName == def.ReflectionName)) { bool needsExplicitly = explicitly; alreadyImplemented = false; foreach (var t in implementingType.GetAllBaseTypeDefinitions ()) { if (t.Kind == TypeKind.Interface) continue; foreach (IProperty cprop in t.Properties) { if (cprop.Name == prop.Name) { if (!needsExplicitly && !cprop.ReturnType.Equals(prop.ReturnType)) needsExplicitly = true; else alreadyImplemented |= !needsExplicitly/* || cprop.InterfaceImplementations.Any (impl => impl.InterfaceType.Resolve (ctx).Equals (interfaceType))*/; } } } if (!alreadyImplemented) toImplement.Add(new Tuple(prop, needsExplicitly)); } return toImplement; } static bool CompareMethods (IMethod interfaceMethod, IMethod typeMethod) { if (typeMethod.IsExplicitInterfaceImplementation) return typeMethod.ImplementedInterfaceMembers.Any (m => m.Equals (interfaceMethod)); return SignatureComparer.Ordinal.Equals (interfaceMethod, typeMethod); } } }