// Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the Apache 2.0 License. // See the LICENSE file in the project root for more information. using System.Collections.Generic; namespace System.Reactive.Linq.ObservableImpl { internal sealed class Distinct : Producer { private readonly IObservable _source; private readonly Func _keySelector; private readonly IEqualityComparer _comparer; public Distinct(IObservable source, Func keySelector, IEqualityComparer comparer) { _source = source; _keySelector = keySelector; _comparer = comparer; } protected override IDisposable Run(IObserver observer, IDisposable cancel, Action setSink) { var sink = new _(this, observer, cancel); setSink(sink); return _source.SubscribeSafe(sink); } class _ : Sink, IObserver { private readonly Distinct _parent; private HashSet _hashSet; public _(Distinct parent, IObserver observer, IDisposable cancel) : base(observer, cancel) { _parent = parent; _hashSet = new HashSet(_parent._comparer); } public void OnNext(TSource value) { var key = default(TKey); var hasAdded = false; try { key = _parent._keySelector(value); hasAdded = _hashSet.Add(key); } catch (Exception exception) { base._observer.OnError(exception); base.Dispose(); return; } if (hasAdded) base._observer.OnNext(value); } public void OnError(Exception error) { base._observer.OnError(error); base.Dispose(); } public void OnCompleted() { base._observer.OnCompleted(); base.Dispose(); } } } }