File: OptionsCollection.cs
Web Access
Project: src\src\Workspaces\CoreTestUtilities\Microsoft.CodeAnalysis.Workspaces.Test.Utilities.csproj (Microsoft.CodeAnalysis.Workspaces.Test.Utilities)
// 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.
 
using System.Collections;
using System.Collections.Generic;
using System.Diagnostics;
using Microsoft.CodeAnalysis.CodeStyle;
using Microsoft.CodeAnalysis.Options;
using Microsoft.CodeAnalysis.Diagnostics.Analyzers.NamingStyles;
 
#if !CODE_STYLE
using System.Collections.Immutable;
using Microsoft.CodeAnalysis.Diagnostics;
using Microsoft.CodeAnalysis.Test.Utilities;
using System.Linq;
using Xunit;
#endif
 
#if !NETCOREAPP
using System;
using Roslyn.Utilities;
using System.Text;
#endif
 
namespace Microsoft.CodeAnalysis.Editor.UnitTests.CodeActions
{
    internal sealed class OptionsCollection(string languageName) : IReadOnlyCollection<KeyValuePair<OptionKey2, object?>>, IOptionsReader
    {
        private readonly Dictionary<OptionKey2, object?> _options = [];
 
        public string LanguageName => languageName;
        public string DefaultExtension => languageName == LanguageNames.CSharp ? "cs" : "vb";
 
        public int Count => _options.Count;
 
        public void Add<T>(OptionKey2 optionKey, T value)
        {
            // Can only add internally defined option whose storage is not mapped to another option:
            Debug.Assert(optionKey.Option is IOption2 { Definition.StorageMapping: null });
            _options.Add(optionKey, value);
        }
 
        public void Set<T>(Option2<T> option, T value)
            => _options[new OptionKey2(option)] = value;
 
        public void Add<T>(Option2<T> option, T value)
            => Add(new OptionKey2(option), value);
 
        public void Add<T>(Option2<CodeStyleOption2<T>> option, T value)
            => Add(option, value, option.DefaultValue.Notification);
 
        public void Add<T>(Option2<CodeStyleOption2<T>> option, T value, NotificationOption2 notification)
            => Add(new OptionKey2(option), new CodeStyleOption2<T>(value, notification));
 
        public void Add<T>(PerLanguageOption2<T> option, T value)
            => Add(new OptionKey2(option, languageName), value);
 
        public void Add<T>(PerLanguageOption2<CodeStyleOption2<T>> option, T value)
            => Add(option, value, option.DefaultValue.Notification);
 
        public void Add<T>(PerLanguageOption2<CodeStyleOption2<T>> option, T value, NotificationOption2 notification)
            => Add(new OptionKey2(option, languageName), new CodeStyleOption2<T>(value, notification));
 
        // 📝 This can be removed if/when collection initializers support AddRange.
        public void Add(OptionsCollection? options)
            => AddRange(options);
 
        public void AddRange(IEnumerable<KeyValuePair<OptionKey2, object?>>? options)
        {
            if (options == null)
                return;
 
            foreach (var (key, value) in options)
                _options.Add(key, value);
        }
 
        public IEnumerator<KeyValuePair<OptionKey2, object?>> GetEnumerator()
            => _options.GetEnumerator();
 
        IEnumerator IEnumerable.GetEnumerator()
            => GetEnumerator();
 
#if !CODE_STYLE
        public OptionSet ToOptionSet()
            => new TestOptionSet(_options.ToImmutableDictionary(entry => new OptionKey(entry.Key.Option, entry.Key.Language), entry => entry.Value));
 
        public void SetGlobalOptions(IGlobalOptionService globalOptions)
        {
            foreach (var (optionKey, value) in _options)
            {
                globalOptions.SetGlobalOption(optionKey, value);
            }
        }
 
        public StructuredAnalyzerConfigOptions ToAnalyzerConfigOptions()
        {
            Assert.All(this, o => Assert.True(o.Key.Option.Definition.IsEditorConfigOption));
 
            var builder = ImmutableDictionary.CreateBuilder<string, string>(AnalyzerConfigOptions.KeyComparer);
 
            foreach (var (key, value) in this)
            {
                if (value is NamingStylePreferences namingPreferences)
                {
                    NamingStylePreferencesEditorConfigSerializer.WriteNamingStylePreferencesToEditorConfig(
                        namingPreferences.SymbolSpecifications,
                        namingPreferences.NamingStyles,
                        namingPreferences.NamingRules,
                        LanguageName,
                        entryWriter: builder.Add,
                        triviaWriter: null);
                }
                else
                {
                    builder.Add(key.Option.Definition.ConfigName, key.Option.Definition.Serializer.Serialize(value));
                }
            }
 
            return StructuredAnalyzerConfigOptions.Create(new DictionaryAnalyzerConfigOptions(builder.ToImmutable()));
        }
#endif
 
        public bool TryGetOption<T>(OptionKey2 optionKey, out T value)
        {
            if (_options.TryGetValue(optionKey, out var objValue))
            {
                value = (T)objValue!;
                return true;
            }
 
            value = default!;
            return false;
        }
    }
}