File: CodeModel\ExternalElements\ExternalCodeFunction.cs
Web Access
Project: ..\..\..\src\VisualStudio\Core\Impl\Microsoft.VisualStudio.LanguageServices.Implementation_zmmkbl53_wpftmp.csproj (Microsoft.VisualStudio.LanguageServices.Implementation)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
 
#nullable disable
 
using System.Linq;
using System.Runtime.InteropServices;
using Microsoft.CodeAnalysis;
using Microsoft.VisualStudio.LanguageServices.Implementation.CodeModel.Collections;
using Microsoft.VisualStudio.LanguageServices.Implementation.Interop;
using Microsoft.VisualStudio.LanguageServices.Implementation.Utilities;
 
namespace Microsoft.VisualStudio.LanguageServices.Implementation.CodeModel.ExternalElements
{
    [ComVisible(true)]
    [ComDefaultInterface(typeof(EnvDTE.CodeFunction))]
    public sealed class ExternalCodeFunction : AbstractExternalCodeMember, ICodeElementContainer<ExternalCodeParameter>, EnvDTE.CodeFunction, EnvDTE80.CodeFunction2
    {
        internal static EnvDTE.CodeFunction Create(CodeModelState state, ProjectId projectId, IMethodSymbol symbol)
        {
            var element = new ExternalCodeFunction(state, projectId, symbol);
            return (EnvDTE.CodeFunction)ComAggregate.CreateAggregatedObject(element);
        }
 
        private ExternalCodeFunction(CodeModelState state, ProjectId projectId, IMethodSymbol symbol)
            : base(state, projectId, symbol)
        {
        }
 
        private IMethodSymbol MethodSymbol
        {
            get { return (IMethodSymbol)LookupSymbol(); }
        }
 
        EnvDTE.CodeElements ICodeElementContainer<ExternalCodeParameter>.GetCollection()
            => this.Parameters;
 
        public override EnvDTE.vsCMElement Kind
        {
            get { return EnvDTE.vsCMElement.vsCMElementFunction; }
        }
 
        public EnvDTE.vsCMFunction FunctionKind
        {
            get
            {
                // TODO: Verify VB implementation
                switch (MethodSymbol.MethodKind)
                {
                    case MethodKind.Constructor:
                        return EnvDTE.vsCMFunction.vsCMFunctionConstructor;
                    case MethodKind.Destructor:
                        return EnvDTE.vsCMFunction.vsCMFunctionDestructor;
                    case MethodKind.UserDefinedOperator:
                    case MethodKind.Conversion:
                        return EnvDTE.vsCMFunction.vsCMFunctionOperator;
                    case MethodKind.Ordinary:
                        return EnvDTE.vsCMFunction.vsCMFunctionFunction;
                    default:
                        throw Exceptions.ThrowEFail();
                }
            }
        }
 
        public bool IsOverloaded
        {
            get
            {
                var symbol = (IMethodSymbol)LookupSymbol();
 
                // Only methods and constructors can be overloaded
                if (symbol.MethodKind is not MethodKind.Ordinary and
                    not MethodKind.Constructor)
                {
                    return false;
                }
 
                var methodsOfName = symbol.ContainingType.GetMembers(symbol.Name)
                                                         .Where(m => m.Kind == SymbolKind.Method);
 
                return methodsOfName.Count() > 1;
            }
        }
 
        public EnvDTE.CodeElements Overloads
        {
            get
            {
                return ExternalOverloadsCollection.Create(this.State, this, this.ProjectId);
            }
        }
 
        public EnvDTE.CodeTypeRef Type
        {
            get
            {
                // TODO: What if ReturnType is null?
                return CodeTypeRef.Create(this.State, this, this.ProjectId, MethodSymbol.ReturnType);
            }
 
            set
            {
                throw Exceptions.ThrowEFail();
            }
        }
 
        public bool IsGeneric
        {
            get { return MethodSymbol.IsGenericMethod; }
        }
 
        public EnvDTE80.vsCMOverrideKind OverrideKind
        {
            get
            {
                // TODO: Verify VB implementation
 
                var symbol = MethodSymbol;
                var result = EnvDTE80.vsCMOverrideKind.vsCMOverrideKindNone;
 
                if (symbol.IsAbstract)
                {
                    result = EnvDTE80.vsCMOverrideKind.vsCMOverrideKindAbstract;
                }
 
                if (symbol.IsVirtual)
                {
                    result = EnvDTE80.vsCMOverrideKind.vsCMOverrideKindVirtual;
                }
 
                if (symbol.IsOverride)
                {
                    result = EnvDTE80.vsCMOverrideKind.vsCMOverrideKindOverride;
                }
 
                if (symbol.HidesBaseMethodsByName)
                {
                    result = EnvDTE80.vsCMOverrideKind.vsCMOverrideKindNew;
                }
 
                if (symbol.IsSealed)
                {
                    result = EnvDTE80.vsCMOverrideKind.vsCMOverrideKindSealed;
                }
 
                return result;
            }
 
            set
            {
                throw Exceptions.ThrowEFail();
            }
        }
    }
}