module Control.Functor.HT where

import qualified Data.Tuple.HT as Tuple
import Data.Tuple.HT (fst3, snd3, thd3)

import qualified Prelude as P
import Prelude (Functor, fmap, flip, const, (.), ($), fst, snd)


void :: Functor f => f a -> f ()
void :: forall (f :: * -> *) a. Functor f => f a -> f ()
void = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (forall a b. a -> b -> a
const ())

map :: Functor f => (a -> b) -> f a -> f b
map :: forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
map = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap

for :: Functor f => f a -> (a -> b) -> f b
for :: forall (f :: * -> *) a b. Functor f => f a -> (a -> b) -> f b
for = forall a b c. (a -> b -> c) -> b -> a -> c
flip forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap


{- |
Caution:
Every pair member has a reference to the argument of 'unzip'.
Depending on the consumption pattern this may cause a memory leak.
For lists, I think, you should generally prefer 'List.unzip'.
-}
unzip :: Functor f => f (a, b) -> (f a, f b)
unzip :: forall (f :: * -> *) a b. Functor f => f (a, b) -> (f a, f b)
unzip f (a, b)
x = (forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall a b. (a, b) -> a
fst f (a, b)
x, forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall a b. (a, b) -> b
snd f (a, b)
x)

{- |
Caution: See 'unzip'.
-}
unzip3 :: Functor f => f (a, b, c) -> (f a, f b, f c)
unzip3 :: forall (f :: * -> *) a b c.
Functor f =>
f (a, b, c) -> (f a, f b, f c)
unzip3 f (a, b, c)
x = (forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall a b c. (a, b, c) -> a
fst3 f (a, b, c)
x, forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall a b c. (a, b, c) -> b
snd3 f (a, b, c)
x, forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall a b c. (a, b, c) -> c
thd3 f (a, b, c)
x)


{- |
Caution: See 'unzip'.
-}
uncurry :: Functor f => (f a -> f b -> g) -> f (a, b) -> g
uncurry :: forall (f :: * -> *) a b g.
Functor f =>
(f a -> f b -> g) -> f (a, b) -> g
uncurry f a -> f b -> g
f = forall a b c. (a -> b -> c) -> (a, b) -> c
P.uncurry f a -> f b -> g
f forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (f :: * -> *) a b. Functor f => f (a, b) -> (f a, f b)
unzip

{- |
Caution: See 'unzip'.
-}
uncurry3 :: Functor f => (f a -> f b -> f c -> g) -> f (a, b, c) -> g
uncurry3 :: forall (f :: * -> *) a b c g.
Functor f =>
(f a -> f b -> f c -> g) -> f (a, b, c) -> g
uncurry3 f a -> f b -> f c -> g
f = forall a b c d. (a -> b -> c -> d) -> (a, b, c) -> d
Tuple.uncurry3 f a -> f b -> f c -> g
f forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (f :: * -> *) a b c.
Functor f =>
f (a, b, c) -> (f a, f b, f c)
unzip3


mapFst :: Functor f => (a -> f c) -> (a, b) -> f (c, b)
mapFst :: forall (f :: * -> *) a c b.
Functor f =>
(a -> f c) -> (a, b) -> f (c, b)
mapFst a -> f c
f ~(a
a,b
b) = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (forall a b c. (a -> b -> c) -> b -> a -> c
flip (,) b
b) forall a b. (a -> b) -> a -> b
$ a -> f c
f a
a

mapSnd :: Functor f => (b -> f c) -> (a, b) -> f (a, c)
mapSnd :: forall (f :: * -> *) b c a.
Functor f =>
(b -> f c) -> (a, b) -> f (a, c)
mapSnd b -> f c
f ~(a
a,b
b) = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap ((,) a
a) forall a b. (a -> b) -> a -> b
$ b -> f c
f b
b


mapFst3 :: Functor f => (a -> f d) -> (a,b,c) -> f (d,b,c)
mapFst3 :: forall (f :: * -> *) a d b c.
Functor f =>
(a -> f d) -> (a, b, c) -> f (d, b, c)
mapFst3 a -> f d
f ~(a
a,b
b,c
c) = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (\d
x -> (d
x,b
b,c
c)) forall a b. (a -> b) -> a -> b
$ a -> f d
f a
a

mapSnd3 :: Functor f => (b -> f d) -> (a,b,c) -> f (a,d,c)
mapSnd3 :: forall (f :: * -> *) b d a c.
Functor f =>
(b -> f d) -> (a, b, c) -> f (a, d, c)
mapSnd3 b -> f d
f ~(a
a,b
b,c
c) = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (\d
x -> (a
a,d
x,c
c)) forall a b. (a -> b) -> a -> b
$ b -> f d
f b
b

mapThd3 :: Functor f => (c -> f d) -> (a,b,c) -> f (a,b,d)
mapThd3 :: forall (f :: * -> *) c d a b.
Functor f =>
(c -> f d) -> (a, b, c) -> f (a, b, d)
mapThd3 c -> f d
f ~(a
a,b
b,c
c) = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap ((,,) a
a b
b) forall a b. (a -> b) -> a -> b
$ c -> f d
f c
c


{- |
Generalization of 'Data.List.HT.outerProduct'.
-}
outerProduct ::
   (Functor f, Functor g) =>
   (a -> b -> c) -> f a -> g b -> f (g c)
outerProduct :: forall (f :: * -> *) (g :: * -> *) a b c.
(Functor f, Functor g) =>
(a -> b -> c) -> f a -> g b -> f (g c)
outerProduct a -> b -> c
f f a
xs g b
ys = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (forall a b c. (a -> b -> c) -> b -> a -> c
flip forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap g b
ys forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> b -> c
f) f a
xs