{-# LANGUAGE Safe #-}
{-# LANGUAGE BangPatterns #-}
module Relude.Extra.Foldable
( foldlSC
, average
) where
import Relude
foldlSC :: forall t b a. Foldable t => (b -> a -> Either b b) -> b -> t a -> b
foldlSC :: forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> Either b b) -> b -> t a -> b
foldlSC b -> a -> Either b b
f = (t a -> b -> b) -> b -> t a -> b
forall a b c. (a -> b -> c) -> b -> a -> c
flip ((t a -> b -> b) -> b -> t a -> b)
-> (t a -> b -> b) -> b -> t a -> b
forall a b. (a -> b) -> a -> b
$ (a -> (b -> b) -> b -> b) -> (b -> b) -> t a -> b -> b
forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
foldr a -> (b -> b) -> b -> b
go b -> b
forall a. a -> a
id
where
go :: a -> (b -> b) -> b -> b
go :: a -> (b -> b) -> b -> b
go a
x b -> b
k b
z = case b -> a -> Either b b
f b
z a
x of
Left b
l -> b
l
Right b
r -> b -> b
k b
r
{-# INLINE foldlSC #-}
average :: forall a f . (Foldable f, Fractional a) => f a -> Maybe a
average :: forall a (f :: * -> *).
(Foldable f, Fractional a) =>
f a -> Maybe a
average f a
xs
| f a -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null f a
xs = Maybe a
forall a. Maybe a
Nothing
| Bool
otherwise = a -> Maybe a
forall a. a -> Maybe a
Just
(a -> Maybe a) -> (f a -> a) -> f a -> Maybe a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (a -> a -> a) -> (a, a) -> a
forall a b c. (a -> b -> c) -> (a, b) -> c
uncurry a -> a -> a
forall a. Fractional a => a -> a -> a
(/)
((a, a) -> a) -> (f a -> (a, a)) -> f a -> a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ((a, a) -> a -> (a, a)) -> (a, a) -> f a -> (a, a)
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl' (\(!a
total, !a
count) a
x -> (a
total a -> a -> a
forall a. Num a => a -> a -> a
+ a
x, a
count a -> a -> a
forall a. Num a => a -> a -> a
+ a
1)) (a
0,a
0)
(f a -> Maybe a) -> f a -> Maybe a
forall a b. (a -> b) -> a -> b
$ f a
xs
{-# INLINE average #-}