File: CodeModel\Collections\ExternalMemberCollection.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.Immutable;
using System.Runtime.InteropServices;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.PooledObjects;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Microsoft.VisualStudio.LanguageServices.Implementation.CodeModel.Interop;
using Microsoft.VisualStudio.LanguageServices.Implementation.Interop;
using Microsoft.VisualStudio.LanguageServices.Implementation.Utilities;
 
namespace Microsoft.VisualStudio.LanguageServices.Implementation.CodeModel.Collections;
 
[ComVisible(true)]
[ComDefaultInterface(typeof(ICodeElements))]
public sealed class ExternalMemberCollection : AbstractCodeElementCollection
{
    internal static EnvDTE.CodeElements Create(
        CodeModelState state,
        object parent,
        ProjectId projectId,
        ITypeSymbol typeSymbol)
    {
        var collection = new ExternalMemberCollection(state, parent, projectId, typeSymbol);
        return (EnvDTE.CodeElements)ComAggregate.CreateAggregatedObject(collection);
    }
 
    private readonly ProjectId _projectId;
    private readonly SymbolKey _typeSymbolId;
    private ImmutableArray<EnvDTE.CodeElement> _children;
 
    private ExternalMemberCollection(CodeModelState state, object parent, ProjectId projectId, ITypeSymbol typeSymbol)
        : base(state, parent)
    {
        _projectId = projectId;
        _typeSymbolId = typeSymbol.GetSymbolKey();
    }
 
    private ImmutableArray<EnvDTE.CodeElement> GetChildren()
    {
        if (_children == null)
        {
            var project = this.State.Workspace.CurrentSolution.GetProject(_projectId);
            if (project == null)
            {
                throw Exceptions.ThrowEFail();
            }
 
            if (_typeSymbolId.Resolve(project.GetCompilationAsync().Result).Symbol is not ITypeSymbol typeSymbol)
            {
                throw Exceptions.ThrowEFail();
            }
 
            var childrenBuilder = ArrayBuilder<EnvDTE.CodeElement>.GetInstance();
 
            foreach (var member in typeSymbol.GetMembers())
            {
                if (this.CodeModelService.IsValidExternalSymbol(member))
                {
                    childrenBuilder.Add(this.State.CodeModelService.CreateExternalCodeElement(this.State, _projectId, member));
                }
            }
 
            foreach (var typeMember in typeSymbol.GetTypeMembers())
            {
                childrenBuilder.Add(this.State.CodeModelService.CreateExternalCodeElement(this.State, _projectId, typeMember));
            }
 
            _children = childrenBuilder.ToImmutableAndFree();
        }
 
        return _children;
    }
 
    protected override bool TryGetItemByIndex(int index, out EnvDTE.CodeElement element)
    {
        var children = GetChildren();
        if (index < children.Length)
        {
            element = children[index];
            return true;
        }
 
        element = null;
        return false;
    }
 
    protected override bool TryGetItemByName(string name, out EnvDTE.CodeElement element)
    {
        var children = GetChildren();
        var index = children.IndexOf(e => e.Name == name);
 
        if (index < children.Length)
        {
            element = children[index];
            return true;
        }
 
        element = null;
        return false;
    }
 
    public override int Count
    {
        get { return GetChildren().Length; }
    }
}