File: CodeModel\Collections\BasesCollection.cs
Web Access
Project: src\src\VisualStudio\Core\Impl\Microsoft.VisualStudio.LanguageServices.Implementation.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.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Runtime.InteropServices;
using Microsoft.CodeAnalysis;
using Microsoft.VisualStudio.LanguageServices.Implementation.CodeModel.Interop;
using Microsoft.VisualStudio.LanguageServices.Implementation.Interop;
using Microsoft.VisualStudio.LanguageServices.Implementation.Utilities;
using Roslyn.Utilities;
 
namespace Microsoft.VisualStudio.LanguageServices.Implementation.CodeModel.Collections;
 
[ComVisible(true)]
[ComDefaultInterface(typeof(ICodeElements))]
public sealed class BasesCollection : AbstractCodeElementCollection
{
    private readonly bool _interfaces;
 
    internal static EnvDTE.CodeElements Create(
        CodeModelState state,
        object parent,
        FileCodeModel fileCodeModel,
        SyntaxNodeKey nodeKey,
        bool interfaces)
    {
        var collection = new BasesCollection(state, parent, fileCodeModel, nodeKey, interfaces);
        return (EnvDTE.CodeElements)ComAggregate.CreateAggregatedObject(collection);
    }
 
    private readonly ComHandle<EnvDTE.FileCodeModel, FileCodeModel> _fileCodeModel;
    private readonly SyntaxNodeKey _nodeKey;
 
    private BasesCollection(
        CodeModelState state,
        object parent,
        FileCodeModel fileCodeModel,
        SyntaxNodeKey nodeKey,
        bool interfaces)
        : base(state, parent)
    {
        Debug.Assert(fileCodeModel != null);
 
        _fileCodeModel = new ComHandle<EnvDTE.FileCodeModel, FileCodeModel>(fileCodeModel);
        _nodeKey = nodeKey;
        _interfaces = interfaces;
    }
 
    private FileCodeModel FileCodeModel
    {
        get { return _fileCodeModel.Object; }
    }
 
    private SyntaxNode LookupNode()
        => FileCodeModel.LookupNode(_nodeKey);
 
    private ITypeSymbol LookupSymbol()
    {
        var node = LookupNode();
        var semanticModel = FileCodeModel.GetSemanticModel();
        return semanticModel.GetDeclaredSymbol(node) as ITypeSymbol;
    }
 
    private IEnumerable<ITypeSymbol> GetBaseTypes()
    {
        var symbol = LookupSymbol();
        if (symbol == null)
        {
            throw Exceptions.ThrowEFail();
        }
 
        if (symbol.TypeKind == TypeKind.Interface || _interfaces)
        {
            return symbol.Interfaces;
        }
        else
        {
            return [symbol.BaseType];
        }
    }
 
    protected override bool TryGetItemByIndex(int index, out EnvDTE.CodeElement element)
    {
        var baseTypes = GetBaseTypes();
        if (index < baseTypes.Count())
        {
            var child = baseTypes.ElementAt(index);
            var projectId = FileCodeModel.GetProjectId();
            element = CodeModelService.CreateCodeType(this.State, projectId, child);
            return true;
        }
 
        element = null;
        return false;
    }
 
    protected override bool TryGetItemByName(string name, out EnvDTE.CodeElement element)
    {
        if (name != null)
        {
            // When searching by name it may or may not be the fully qualified named,
            // but we need the fully qualified name for comparison.
            var node = LookupNode();
            var semanticModel = FileCodeModel.GetSemanticModel();
 
            name = CodeModelService.GetFullyQualifiedName(name, node.SpanStart, semanticModel);
        }
 
        foreach (var child in GetBaseTypes())
        {
            var childName = child.GetEscapedFullName();
            if (childName == name)
            {
                var projectId = FileCodeModel.GetProjectId();
                element = CodeModelService.CreateCodeType(this.State, projectId, child);
                return true;
            }
        }
 
        element = null;
        return false;
    }
 
    public override int Count
    {
        get
        {
            var symbol = LookupSymbol();
            if (symbol == null)
            {
                throw Exceptions.ThrowEFail();
            }
 
            if (symbol.TypeKind == TypeKind.Interface || _interfaces)
            {
                return symbol.Interfaces.Length;
            }
            else
            {
                return 1;
            }
        }
    }
}