~ubuntu-branches/ubuntu/trusty/monodevelop/trusty-proposed

« back to all changes in this revision

Viewing changes to external/nrefactory/ICSharpCode.NRefactory/TypeSystem/InheritanceHelper.cs

  • Committer: Package Import Robot
  • Author(s): Jo Shields
  • Date: 2013-05-12 09:46:03 UTC
  • mto: This revision was merged to the branch mainline in revision 29.
  • Revision ID: package-import@ubuntu.com-20130512094603-mad323bzcxvmcam0
Tags: upstream-4.0.5+dfsg
Import upstream version 4.0.5+dfsg

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
// Copyright (c) 2010-2013 AlphaSierraPapa for the SharpDevelop Team
 
2
// 
 
3
// Permission is hereby granted, free of charge, to any person obtaining a copy of this
 
4
// software and associated documentation files (the "Software"), to deal in the Software
 
5
// without restriction, including without limitation the rights to use, copy, modify, merge,
 
6
// publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons
 
7
// to whom the Software is furnished to do so, subject to the following conditions:
 
8
// 
 
9
// The above copyright notice and this permission notice shall be included in all copies or
 
10
// substantial portions of the Software.
 
11
// 
 
12
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
 
13
// INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
 
14
// PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE
 
15
// FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
 
16
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 
17
// DEALINGS IN THE SOFTWARE.
 
18
 
 
19
using System;
 
20
using System.Collections.Generic;
 
21
using System.Linq;
 
22
using ICSharpCode.NRefactory.TypeSystem.Implementation;
 
23
 
 
24
namespace ICSharpCode.NRefactory.TypeSystem
 
25
{
 
26
        /// <summary>
 
27
        /// Provides helper methods for inheritance.
 
28
        /// </summary>
 
29
        public static class InheritanceHelper
 
30
        {
 
31
                // TODO: maybe these should be extension methods?
 
32
                // or even part of the interface itself? (would allow for easy caching)
 
33
                
 
34
                #region GetBaseMember
 
35
                /// <summary>
 
36
                /// Gets the base member that has the same signature.
 
37
                /// </summary>
 
38
                public static IMember GetBaseMember(IMember member)
 
39
                {
 
40
                        return GetBaseMembers(member, false).FirstOrDefault();
 
41
                }
 
42
 
 
43
                /// <summary>
 
44
                /// Gets all base members that have the same signature.
 
45
                /// </summary>
 
46
                /// <returns>
 
47
                /// List of base members with the same signature. The member from the derived-most base class is returned first.
 
48
                /// </returns>
 
49
                public static IEnumerable<IMember> GetBaseMembers(IMember member, bool includeImplementedInterfaces)
 
50
                {
 
51
                        if (member == null)
 
52
                                throw new ArgumentNullException("member");
 
53
 
 
54
                        if (member.IsExplicitInterfaceImplementation && member.ImplementedInterfaceMembers.Count == 1) {
 
55
                                // C#-style explicit interface implementation
 
56
                                member = member.ImplementedInterfaceMembers[0];
 
57
                                yield return member;
 
58
                        }
 
59
                        
 
60
                        // TODO: can we get rid of this upcast?
 
61
                        SpecializedMember specializedMember = member as SpecializedMember;
 
62
                        member = member.MemberDefinition;
 
63
                        
 
64
                        IEnumerable<IType> allBaseTypes;
 
65
                        if (includeImplementedInterfaces) {
 
66
                                allBaseTypes = member.DeclaringTypeDefinition.GetAllBaseTypes();
 
67
                        } else {
 
68
                                allBaseTypes = member.DeclaringTypeDefinition.GetNonInterfaceBaseTypes();
 
69
                        }
 
70
                        foreach (IType baseType in allBaseTypes.Reverse()) {
 
71
                                if (baseType == member.DeclaringTypeDefinition)
 
72
                                        continue;
 
73
 
 
74
                                IEnumerable<IMember> baseMembers;
 
75
                                if (member.EntityType == EntityType.Accessor) {
 
76
                                        baseMembers = baseType.GetAccessors(m => m.Name == member.Name && !m.IsExplicitInterfaceImplementation, GetMemberOptions.IgnoreInheritedMembers);
 
77
                                } else {
 
78
                                        baseMembers = baseType.GetMembers(m => m.Name == member.Name && !m.IsExplicitInterfaceImplementation, GetMemberOptions.IgnoreInheritedMembers);
 
79
                                }
 
80
                                foreach (IMember baseMember in baseMembers) {
 
81
                                        if (SignatureComparer.Ordinal.Equals(member, baseMember)) {
 
82
                                                if (specializedMember != null)
 
83
                                                        yield return baseMember.Specialize(specializedMember.Substitution);
 
84
                                                else
 
85
                                                        yield return baseMember;
 
86
                                        }
 
87
                                }
 
88
                        }
 
89
                }
 
90
                #endregion
 
91
                
 
92
                #region GetDerivedMember
 
93
                /// <summary>
 
94
                /// Finds the member declared in 'derivedType' that has the same signature (could override) 'baseMember'.
 
95
                /// </summary>
 
96
                public static IMember GetDerivedMember(IMember baseMember, ITypeDefinition derivedType)
 
97
                {
 
98
                        if (baseMember == null)
 
99
                                throw new ArgumentNullException("baseMember");
 
100
                        if (derivedType == null)
 
101
                                throw new ArgumentNullException("derivedType");
 
102
                        
 
103
                        if (baseMember.Compilation != derivedType.Compilation)
 
104
                                throw new ArgumentException("baseMember and derivedType must be from the same compilation");
 
105
                        
 
106
                        baseMember = baseMember.MemberDefinition;
 
107
                        bool includeInterfaces = baseMember.DeclaringTypeDefinition.Kind == TypeKind.Interface;
 
108
                        IMethod method = baseMember as IMethod;
 
109
                        if (method != null) {
 
110
                                foreach (IMethod derivedMethod in derivedType.Methods) {
 
111
                                        if (derivedMethod.Name == method.Name && derivedMethod.Parameters.Count == method.Parameters.Count) {
 
112
                                                if (derivedMethod.TypeParameters.Count == method.TypeParameters.Count) {
 
113
                                                        // The method could override the base method:
 
114
                                                        if (GetBaseMembers(derivedMethod, includeInterfaces).Any(m => m.MemberDefinition == baseMember))
 
115
                                                                return derivedMethod;
 
116
                                                }
 
117
                                        }
 
118
                                }
 
119
                        }
 
120
                        IProperty property = baseMember as IProperty;
 
121
                        if (property != null) {
 
122
                                foreach (IProperty derivedProperty in derivedType.Properties) {
 
123
                                        if (derivedProperty.Name == property.Name && derivedProperty.Parameters.Count == property.Parameters.Count) {
 
124
                                                // The property could override the base property:
 
125
                                                if (GetBaseMembers(derivedProperty, includeInterfaces).Any(m => m.MemberDefinition == baseMember))
 
126
                                                        return derivedProperty;
 
127
                                        }
 
128
                                }
 
129
                        }
 
130
                        if (baseMember is IEvent) {
 
131
                                foreach (IEvent derivedEvent in derivedType.Events) {
 
132
                                        if (derivedEvent.Name == baseMember.Name)
 
133
                                                return derivedEvent;
 
134
                                }
 
135
                        }
 
136
                        if (baseMember is IField) {
 
137
                                foreach (IField derivedField in derivedType.Fields) {
 
138
                                        if (derivedField.Name == baseMember.Name)
 
139
                                                return derivedField;
 
140
                                }
 
141
                        }
 
142
                        return null;
 
143
                }
 
144
                #endregion
 
145
        }
 
146
}