// 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.Generic; using System.Threading.Tasks; namespace System.Reactive.Linq { public partial class AsyncObservable { public static IAsyncObservable DistinctUntilChanged(IAsyncObservable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Create(source, static (source, observer) => source.SubscribeSafeAsync(AsyncObserver.DistinctUntilChanged(observer))); } public static IAsyncObservable DistinctUntilChanged(IAsyncObservable source, IEqualityComparer comparer) { if (source == null) throw new ArgumentNullException(nameof(source)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); return Create( source, comparer, static (source, comparer, observer) => source.SubscribeSafeAsync(AsyncObserver.DistinctUntilChanged(observer, comparer))); } public static IAsyncObservable DistinctUntilChanged(IAsyncObservable source, Func keySelector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); return Create( source, keySelector, static (source, keySelector, observer) => source.SubscribeSafeAsync(AsyncObserver.DistinctUntilChanged(observer, keySelector))); } public static IAsyncObservable DistinctUntilChanged(IAsyncObservable source, Func> keySelector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); return Create( source, keySelector, static (source, keySelector, observer) => source.SubscribeSafeAsync(AsyncObserver.DistinctUntilChanged(observer, keySelector))); } public static IAsyncObservable DistinctUntilChanged(IAsyncObservable source, Func keySelector, IEqualityComparer comparer) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); return Create( source, (keySelector, comparer), static (source, state, observer) => source.SubscribeSafeAsync(AsyncObserver.DistinctUntilChanged(observer, state.keySelector, state.comparer))); } public static IAsyncObservable DistinctUntilChanged(IAsyncObservable source, Func> keySelector, IEqualityComparer comparer) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); return Create( source, (keySelector, comparer), static (source, state, observer) => source.SubscribeSafeAsync(AsyncObserver.DistinctUntilChanged(observer, state.keySelector, state.comparer))); } } public partial class AsyncObserver { public static IAsyncObserver DistinctUntilChanged(IAsyncObserver observer) { if (observer == null) throw new ArgumentNullException(nameof(observer)); return DistinctUntilChanged(observer, x => x, EqualityComparer.Default); } public static IAsyncObserver DistinctUntilChanged(IAsyncObserver observer, IEqualityComparer comparer) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); return DistinctUntilChanged(observer, x => x, comparer); } public static IAsyncObserver DistinctUntilChanged(IAsyncObserver observer, Func keySelector) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); return DistinctUntilChanged(observer, keySelector, EqualityComparer.Default); } public static IAsyncObserver DistinctUntilChanged(IAsyncObserver observer, Func> keySelector) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); return DistinctUntilChanged(observer, keySelector, EqualityComparer.Default); } public static IAsyncObserver DistinctUntilChanged(IAsyncObserver observer, Func keySelector, IEqualityComparer comparer) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); var hasCurrentKey = false; var currentKey = default(TKey); return Create( async x => { var key = default(TKey); try { key = keySelector(x); } catch (Exception ex) { await observer.OnErrorAsync(ex).ConfigureAwait(false); return; } var equals = default(bool); if (hasCurrentKey) { try { equals = comparer.Equals(currentKey, key); } catch (Exception ex) { await observer.OnErrorAsync(ex).ConfigureAwait(false); return; } } if (!hasCurrentKey || !equals) { hasCurrentKey = true; currentKey = key; await observer.OnNextAsync(x).ConfigureAwait(false); } }, observer.OnErrorAsync, observer.OnCompletedAsync ); } public static IAsyncObserver DistinctUntilChanged(IAsyncObserver observer, Func> keySelector, IEqualityComparer comparer) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); var hasCurrentKey = false; var currentKey = default(TKey); return Create( async x => { var key = default(TKey); try { key = await keySelector(x).ConfigureAwait(false); } catch (Exception ex) { await observer.OnErrorAsync(ex).ConfigureAwait(false); return; } var equals = default(bool); if (hasCurrentKey) { try { equals = comparer.Equals(currentKey, key); } catch (Exception ex) { await observer.OnErrorAsync(ex).ConfigureAwait(false); return; } } if (!hasCurrentKey || !equals) { hasCurrentKey = true; currentKey = key; await observer.OnNextAsync(x).ConfigureAwait(false); } }, observer.OnErrorAsync, observer.OnCompletedAsync ); } } }