envlang-csharp/Utils/Enumerable.cs
2020-08-17 02:11:47 +00:00

191 lines
5.3 KiB
C#

using System;
using System.Linq;
using System.Collections.Generic;
using Immutable;
public static class Collection {
public static void ForEach<T>(this IEnumerable<T> x, Action<T> f)
=> x.ToList().ForEach(f);
/*
public static ListI<Tuple<T,U>> Add<T,U>(this ListI<Tuple<T,U>> l, T x, U y)
=> l.Add(Tuple.Create(x,y));
*/
// System.Collections.Immutable requires NuGet and is not available on repl.it
public static List<T> Cons<T>(this List<T> l, T x) { l.Add(x); return l; }
// Circumvent bug with collection initializers, tuples and
// first-class functions by using repeated .Add()
// See https://repl.it/@suzannesoy/WarlikeWorstTraining#main.cs
public static List<Tuple<T,U>> Cons<T,U>(this List<Tuple<T,U>> l, T x, U y)
=> l.Cons(Tuple.Create(x,y));
public static List<Tuple<T,U,V>> Cons<T,U,V>(this List<Tuple<T,U,V>> l, T x, U y, V z)
=> l.Cons(Tuple.Create(x,y,z));
public static void Deconstruct<A, B>(this Tuple<A, B> t, out A a, out B b) {
a = t.Item1;
b = t.Item2;
}
public struct Item<T> {
public readonly T item;
public readonly long index;
public readonly bool first;
public readonly bool last;
public Item(T item, long index, bool first, bool last) {
this.item = item;
this.index = index;
this.first = first;
this.last = last;
}
}
public static IEnumerable<Item<T>> Indexed<T>(this IEnumerable<T> e) {
long i = 0L;
bool first = true;
// These default(…) are written below before being read
T prevX = default(T);
long prevI = default(long);
bool prevFirst = default(bool);
foreach (var x in e) {
if (!first) {
yield return new Item<T>(prevX, prevI, prevFirst, false);
}
prevX = x;
prevI = i;
prevFirst = first;
first = false;
i++;
}
if (!first) {
yield return new Item<T>(prevX, prevI, prevFirst, true);
}
}
public struct Peekable<T> : IEnumerator<T>, System.Collections.IEnumerator {
private IEnumerator<T> e;
private bool peeked;
private T previous;
public T Current { get => peeked ? previous : e.Current; }
object System.Collections.IEnumerator.Current {
get => this.Current;
}
public bool MoveNext() {
this.peeked = false;
this.previous = default(T); // guarded by peeked
return this.e.MoveNext();
}
public bool Peek() {
if (this.peeked) {
throw new Exception("Already peeked once");
} else {
this.previous = e.Current;
this.peeked = true;
return this.e.MoveNext();
}
}
public void Dispose() { e.Dispose(); }
public void Reset() { e.Reset(); }
public Peekable(IEnumerable<T> e) {
this.e = e.GetEnumerator();
this.peeked = false;
this.previous = default(T); // guarded by peeked
}
}
public static Peekable<T> Peekable<T>(this IEnumerable<T> e) {
return new Peekable<T>(e);
}
public static IEnumerable<T> SingleUseEnumerable<T>(this IEnumerator<T> e) {
while (e.MoveNext()) {
yield return e.Current;
}
}
public static IEnumerable<T> TakeUntil<T>(this IEnumerable<T> e, Func<T, bool> f)
=> e.TakeWhile(x => !f(x));
public static IEnumerable<T> Singleton<T>(this T x) {
yield return x;
}
public static string Join(this string separator, IEnumerable<string> strings)
=> String.Join(separator, strings);
public static Option<T> First<T>(this IEnumerable<T> ie) {
var e = ie.GetEnumerator();
if (e.MoveNext()) {
return e.Current.Some();
} else {
return Option.None<T>();
}
}
public static Option<T> First<T>(this IEnumerable<T> ie, Func<T, bool> predicate) {
var e = ie.GetEnumerator();
while (e.MoveNext()) {
if (predicate(e.Current)) {
return e.Current.Some();
}
}
return Option.None<T>();
}
public static Option<T> Single<T>(this IEnumerable<T> ie) {
var e = ie.GetEnumerator();
if (e.MoveNext()) {
var value = e.Current;
if (e.MoveNext()) {
return Option.None<T>();
} else {
return value.Some();
}
} else {
return Option.None<T>();
}
}
public static Option<V> GetValue<K, V>(this Dictionary<K, V> d, K key) {
V result = default(V);
if (d.TryGetValue(key, out result)) {
return result.Some();
} else {
return Option.None<V>();
}
}
public static V GetOrDefault<K, V>(this Dictionary<K, V> d, K key, V defaultValue) {
V result = default(V);
if (d.TryGetValue(key, out result)) {
return result;
} else {
return defaultValue;
}
}
public class DefaultDictionary<TKey, TValue> : Dictionary<TKey, TValue> {
public readonly TValue defaultValue;
//public readonly Dictionary<TKey, TValue> dictionary;
public DefaultDictionary(TValue defaultValue, Dictionary<TKey, TValue> dictionary) : base(dictionary) {
this.defaultValue = defaultValue;
//this.dictionary = dictionary;
}
public new TValue this[TKey key] {
get {
return this.GetOrDefault(key, defaultValue);
}
}
}
public static DefaultDictionary<UKey, UValue> ToDefaultDictionary<T, UKey, UValue>(this IEnumerable<T> e, UValue defaultValue, Func<T, UKey> key, Func<T, UValue> value)
=> new DefaultDictionary<UKey, UValue>(defaultValue, e.ToDictionary(key, value));
}