{-# LANGUAGE MultiParamTypeClasses, FlexibleContexts, TypeFamilies, CPP #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE DefaultSignatures #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeOperators #-}
{-# LANGUAGE UndecidableInstances #-}
{-# LANGUAGE DeriveGeneric #-}
module Data.AffineSpace
(
AffineSpace(..), (.-^), distanceSq, distance, alerp, affineCombo
) where
import Control.Applicative (liftA2)
import Data.Ratio
import Foreign.C.Types (CSChar, CInt, CShort, CLong, CLLong, CIntMax, CFloat, CDouble)
import Control.Arrow(first)
import Data.VectorSpace
import Data.Basis
import Data.VectorSpace.Generic
import qualified GHC.Generics as Gnrx
import GHC.Generics (Generic, (:*:)(..))
infixl 6 .+^, .-^
infix 6 .-.
class AdditiveGroup (Diff p) => AffineSpace p where
type Diff p
type Diff p = GenericDiff p
(.-.) :: p -> p -> Diff p
default (.-.) :: ( Generic p, Diff p ~ GenericDiff p, AffineSpace (VRep p) )
=> p -> p -> Diff p
p
p .-. p
q = Diff (VRep p) -> GenericDiff p
forall p. Diff (VRep p) -> GenericDiff p
GenericDiff
(Diff (VRep p) -> GenericDiff p) -> Diff (VRep p) -> GenericDiff p
forall a b. (a -> b) -> a -> b
$ (p -> VRep p
forall a x. Generic a => a -> Rep a x
Gnrx.from p
p VRep p -> VRep p -> Diff (VRep p)
forall p. AffineSpace p => p -> p -> Diff p
.-. (p -> VRep p
forall a x. Generic a => a -> Rep a x
Gnrx.from p
q :: VRep p))
(.+^) :: p -> Diff p -> p
default (.+^) :: ( Generic p, Diff p ~ GenericDiff p, AffineSpace (VRep p) )
=> p -> Diff p -> p
p
p .+^ GenericDiff q = VRep p -> p
forall a x. Generic a => Rep a x -> a
Gnrx.to (p -> VRep p
forall a x. Generic a => a -> Rep a x
Gnrx.from p
p VRep p -> Diff (VRep p) -> VRep p
forall p. AffineSpace p => p -> Diff p -> p
.+^ Diff (VRep p)
q :: VRep p)
(.-^) :: AffineSpace p => p -> Diff p -> p
p
p .-^ :: p -> Diff p -> p
.-^ Diff p
v = p
p p -> Diff p -> p
forall p. AffineSpace p => p -> Diff p -> p
.+^ Diff p -> Diff p
forall v. AdditiveGroup v => v -> v
negateV Diff p
v
distanceSq :: (AffineSpace p, v ~ Diff p, InnerSpace v) =>
p -> p -> Scalar v
distanceSq :: p -> p -> Scalar v
distanceSq = (((p -> v) -> p -> Scalar v) -> (p -> p -> v) -> p -> p -> Scalar v
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap(((p -> v) -> p -> Scalar v)
-> (p -> p -> v) -> p -> p -> Scalar v)
-> ((v -> Scalar v) -> (p -> v) -> p -> Scalar v)
-> (v -> Scalar v)
-> (p -> p -> v)
-> p
-> p
-> Scalar v
forall b c a. (b -> c) -> (a -> b) -> a -> c
.(v -> Scalar v) -> (p -> v) -> p -> Scalar v
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap) v -> Scalar v
forall v s. (InnerSpace v, s ~ Scalar v) => v -> s
magnitudeSq p -> p -> v
forall p. AffineSpace p => p -> p -> Diff p
(.-.)
distance :: (AffineSpace p, v ~ Diff p, InnerSpace v
, s ~ Scalar v, Floating (Scalar v))
=> p -> p -> s
distance :: p -> p -> s
distance = (((p -> s) -> p -> s) -> (p -> p -> s) -> p -> p -> s
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap(((p -> s) -> p -> s) -> (p -> p -> s) -> p -> p -> s)
-> ((s -> s) -> (p -> s) -> p -> s)
-> (s -> s)
-> (p -> p -> s)
-> p
-> p
-> s
forall b c a. (b -> c) -> (a -> b) -> a -> c
.(s -> s) -> (p -> s) -> p -> s
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap) s -> s
forall a. Floating a => a -> a
sqrt p -> p -> s
forall p v.
(AffineSpace p, v ~ Diff p, InnerSpace v) =>
p -> p -> Scalar v
distanceSq
alerp :: (AffineSpace p, VectorSpace (Diff p)) =>
p -> p -> Scalar (Diff p) -> p
alerp :: p -> p -> Scalar (Diff p) -> p
alerp p
p p
p' Scalar (Diff p)
s = p
p p -> Diff p -> p
forall p. AffineSpace p => p -> Diff p -> p
.+^ (Scalar (Diff p)
s Scalar (Diff p) -> Diff p -> Diff p
forall v. VectorSpace v => Scalar v -> v -> v
*^ (p
p' p -> p -> Diff p
forall p. AffineSpace p => p -> p -> Diff p
.-. p
p))
affineCombo :: (AffineSpace p, v ~ Diff p, VectorSpace v) => p -> [(p,Scalar v)] -> p
affineCombo :: p -> [(p, Scalar v)] -> p
affineCombo p
z [(p, Scalar v)]
l = p
z p -> Diff p -> p
forall p. AffineSpace p => p -> Diff p -> p
.+^ [(v, Scalar v)] -> v
forall v. VectorSpace v => [(v, Scalar v)] -> v
linearCombo (((p, Scalar v) -> (v, Scalar v))
-> [(p, Scalar v)] -> [(v, Scalar v)]
forall a b. (a -> b) -> [a] -> [b]
map ((p -> v) -> (p, Scalar v) -> (v, Scalar v)
forall (a :: * -> * -> *) b c d.
Arrow a =>
a b c -> a (b, d) (c, d)
first (p -> p -> Diff p
forall p. AffineSpace p => p -> p -> Diff p
.-. p
z)) [(p, Scalar v)]
l)
#define ScalarTypeCon(con,t) \
instance con => AffineSpace (t) where \
{ type Diff (t) = t \
; (.-.) = (-) \
; (.+^) = (+) }
#define ScalarType(t) ScalarTypeCon((),t)
ScalarType(Int)
ScalarType(Integer)
ScalarType(Double)
ScalarType(Float)
ScalarType(CSChar)
ScalarType(CInt)
ScalarType(CShort)
ScalarType(CLong)
ScalarType(CLLong)
ScalarType(CIntMax)
ScalarType(CDouble)
ScalarType(CFloat)
ScalarTypeCon(Integral a,Ratio a)
instance (AffineSpace p, AffineSpace q) => AffineSpace (p,q) where
type Diff (p,q) = (Diff p, Diff q)
(p
p,q
q) .-. :: (p, q) -> (p, q) -> Diff (p, q)
.-. (p
p',q
q') = (p
p p -> p -> Diff p
forall p. AffineSpace p => p -> p -> Diff p
.-. p
p', q
q q -> q -> Diff q
forall p. AffineSpace p => p -> p -> Diff p
.-. q
q')
(p
p,q
q) .+^ :: (p, q) -> Diff (p, q) -> (p, q)
.+^ (u,v) = (p
p p -> Diff p -> p
forall p. AffineSpace p => p -> Diff p -> p
.+^ Diff p
u, q
q q -> Diff q -> q
forall p. AffineSpace p => p -> Diff p -> p
.+^ Diff q
v)
instance (AffineSpace p, AffineSpace q, AffineSpace r) => AffineSpace (p,q,r) where
type Diff (p,q,r) = (Diff p, Diff q, Diff r)
(p
p,q
q,r
r) .-. :: (p, q, r) -> (p, q, r) -> Diff (p, q, r)
.-. (p
p',q
q',r
r') = (p
p p -> p -> Diff p
forall p. AffineSpace p => p -> p -> Diff p
.-. p
p', q
q q -> q -> Diff q
forall p. AffineSpace p => p -> p -> Diff p
.-. q
q', r
r r -> r -> Diff r
forall p. AffineSpace p => p -> p -> Diff p
.-. r
r')
(p
p,q
q,r
r) .+^ :: (p, q, r) -> Diff (p, q, r) -> (p, q, r)
.+^ (u,v,w) = (p
p p -> Diff p -> p
forall p. AffineSpace p => p -> Diff p -> p
.+^ Diff p
u, q
q q -> Diff q -> q
forall p. AffineSpace p => p -> Diff p -> p
.+^ Diff q
v, r
r r -> Diff r -> r
forall p. AffineSpace p => p -> Diff p -> p
.+^ Diff r
w)
instance (AffineSpace p) => AffineSpace (a -> p) where
type Diff (a -> p) = a -> Diff p
.-. :: (a -> p) -> (a -> p) -> Diff (a -> p)
(.-.) = (p -> p -> Diff p) -> (a -> p) -> (a -> p) -> a -> Diff p
forall (f :: * -> *) a b c.
Applicative f =>
(a -> b -> c) -> f a -> f b -> f c
liftA2 p -> p -> Diff p
forall p. AffineSpace p => p -> p -> Diff p
(.-.)
.+^ :: (a -> p) -> Diff (a -> p) -> a -> p
(.+^) = (p -> Diff p -> p) -> (a -> p) -> (a -> Diff p) -> a -> p
forall (f :: * -> *) a b c.
Applicative f =>
(a -> b -> c) -> f a -> f b -> f c
liftA2 p -> Diff p -> p
forall p. AffineSpace p => p -> Diff p -> p
(.+^)
newtype GenericDiff p = GenericDiff (Diff (VRep p))
deriving ((forall x. GenericDiff p -> Rep (GenericDiff p) x)
-> (forall x. Rep (GenericDiff p) x -> GenericDiff p)
-> Generic (GenericDiff p)
forall x. Rep (GenericDiff p) x -> GenericDiff p
forall x. GenericDiff p -> Rep (GenericDiff p) x
forall a.
(forall x. a -> Rep a x) -> (forall x. Rep a x -> a) -> Generic a
forall p x. Rep (GenericDiff p) x -> GenericDiff p
forall p x. GenericDiff p -> Rep (GenericDiff p) x
$cto :: forall p x. Rep (GenericDiff p) x -> GenericDiff p
$cfrom :: forall p x. GenericDiff p -> Rep (GenericDiff p) x
Generic)
instance AdditiveGroup (Diff (VRep p)) => AdditiveGroup (GenericDiff p)
instance VectorSpace (Diff (VRep p)) => VectorSpace (GenericDiff p)
instance InnerSpace (Diff (VRep p)) => InnerSpace (GenericDiff p)
instance HasBasis (Diff (VRep p)) => HasBasis (GenericDiff p)
data AffineDiffProductSpace f g p = AffineDiffProductSpace
!(Diff (f p)) !(Diff (g p)) deriving ((forall x.
AffineDiffProductSpace f g p
-> Rep (AffineDiffProductSpace f g p) x)
-> (forall x.
Rep (AffineDiffProductSpace f g p) x
-> AffineDiffProductSpace f g p)
-> Generic (AffineDiffProductSpace f g p)
forall x.
Rep (AffineDiffProductSpace f g p) x
-> AffineDiffProductSpace f g p
forall x.
AffineDiffProductSpace f g p
-> Rep (AffineDiffProductSpace f g p) x
forall a.
(forall x. a -> Rep a x) -> (forall x. Rep a x -> a) -> Generic a
forall (f :: * -> *) (g :: * -> *) p x.
Rep (AffineDiffProductSpace f g p) x
-> AffineDiffProductSpace f g p
forall (f :: * -> *) (g :: * -> *) p x.
AffineDiffProductSpace f g p
-> Rep (AffineDiffProductSpace f g p) x
$cto :: forall (f :: * -> *) (g :: * -> *) p x.
Rep (AffineDiffProductSpace f g p) x
-> AffineDiffProductSpace f g p
$cfrom :: forall (f :: * -> *) (g :: * -> *) p x.
AffineDiffProductSpace f g p
-> Rep (AffineDiffProductSpace f g p) x
Generic)
instance (AffineSpace (f p), AffineSpace (g p))
=> AdditiveGroup (AffineDiffProductSpace f g p)
instance ( AffineSpace (f p), AffineSpace (g p)
, VectorSpace (Diff (f p)), VectorSpace (Diff (g p))
, Scalar (Diff (f p)) ~ Scalar (Diff (g p)) )
=> VectorSpace (AffineDiffProductSpace f g p)
instance ( AffineSpace (f p), AffineSpace (g p)
, InnerSpace (Diff (f p)), InnerSpace (Diff (g p))
, Scalar (Diff (f p)) ~ Scalar (Diff (g p))
, Num (Scalar (Diff (f p))) )
=> InnerSpace (AffineDiffProductSpace f g p)
instance (AffineSpace (f p), AffineSpace (g p))
=> AffineSpace (AffineDiffProductSpace f g p) where
type Diff (AffineDiffProductSpace f g p) = AffineDiffProductSpace f g p
.+^ :: AffineDiffProductSpace f g p
-> Diff (AffineDiffProductSpace f g p)
-> AffineDiffProductSpace f g p
(.+^) = AffineDiffProductSpace f g p
-> Diff (AffineDiffProductSpace f g p)
-> AffineDiffProductSpace f g p
forall v. AdditiveGroup v => v -> v -> v
(^+^)
.-. :: AffineDiffProductSpace f g p
-> AffineDiffProductSpace f g p
-> Diff (AffineDiffProductSpace f g p)
(.-.) = AffineDiffProductSpace f g p
-> AffineDiffProductSpace f g p
-> Diff (AffineDiffProductSpace f g p)
forall v. AdditiveGroup v => v -> v -> v
(^-^)
instance ( AffineSpace (f p), AffineSpace (g p)
, HasBasis (Diff (f p)), HasBasis (Diff (g p))
, Scalar (Diff (f p)) ~ Scalar (Diff (g p)) )
=> HasBasis (AffineDiffProductSpace f g p) where
type Basis (AffineDiffProductSpace f g p) = Either (Basis (Diff (f p)))
(Basis (Diff (g p)))
basisValue :: Basis (AffineDiffProductSpace f g p)
-> AffineDiffProductSpace f g p
basisValue (Left bf) = Diff (f p) -> Diff (g p) -> AffineDiffProductSpace f g p
forall (f :: * -> *) (g :: * -> *) p.
Diff (f p) -> Diff (g p) -> AffineDiffProductSpace f g p
AffineDiffProductSpace (Basis (Diff (f p)) -> Diff (f p)
forall v. HasBasis v => Basis v -> v
basisValue Basis (Diff (f p))
bf) Diff (g p)
forall v. AdditiveGroup v => v
zeroV
basisValue (Right bg) = Diff (f p) -> Diff (g p) -> AffineDiffProductSpace f g p
forall (f :: * -> *) (g :: * -> *) p.
Diff (f p) -> Diff (g p) -> AffineDiffProductSpace f g p
AffineDiffProductSpace Diff (f p)
forall v. AdditiveGroup v => v
zeroV (Basis (Diff (g p)) -> Diff (g p)
forall v. HasBasis v => Basis v -> v
basisValue Basis (Diff (g p))
bg)
decompose :: AffineDiffProductSpace f g p
-> [(Basis (AffineDiffProductSpace f g p),
Scalar (AffineDiffProductSpace f g p))]
decompose (AffineDiffProductSpace Diff (f p)
vf Diff (g p)
vg)
= ((Basis (Diff (f p)), Scalar (Diff (g p)))
-> (Either (Basis (Diff (f p))) (Basis (Diff (g p))),
Scalar (Diff (g p))))
-> [(Basis (Diff (f p)), Scalar (Diff (g p)))]
-> [(Either (Basis (Diff (f p))) (Basis (Diff (g p))),
Scalar (Diff (g p)))]
forall a b. (a -> b) -> [a] -> [b]
map ((Basis (Diff (f p))
-> Either (Basis (Diff (f p))) (Basis (Diff (g p))))
-> (Basis (Diff (f p)), Scalar (Diff (g p)))
-> (Either (Basis (Diff (f p))) (Basis (Diff (g p))),
Scalar (Diff (g p)))
forall (a :: * -> * -> *) b c d.
Arrow a =>
a b c -> a (b, d) (c, d)
first Basis (Diff (f p))
-> Either (Basis (Diff (f p))) (Basis (Diff (g p)))
forall a b. a -> Either a b
Left) (Diff (f p) -> [(Basis (Diff (f p)), Scalar (Diff (f p)))]
forall v. HasBasis v => v -> [(Basis v, Scalar v)]
decompose Diff (f p)
vf) [(Either (Basis (Diff (f p))) (Basis (Diff (g p))),
Scalar (Diff (g p)))]
-> [(Either (Basis (Diff (f p))) (Basis (Diff (g p))),
Scalar (Diff (g p)))]
-> [(Either (Basis (Diff (f p))) (Basis (Diff (g p))),
Scalar (Diff (g p)))]
forall a. [a] -> [a] -> [a]
++ ((Basis (Diff (g p)), Scalar (Diff (g p)))
-> (Either (Basis (Diff (f p))) (Basis (Diff (g p))),
Scalar (Diff (g p))))
-> [(Basis (Diff (g p)), Scalar (Diff (g p)))]
-> [(Either (Basis (Diff (f p))) (Basis (Diff (g p))),
Scalar (Diff (g p)))]
forall a b. (a -> b) -> [a] -> [b]
map ((Basis (Diff (g p))
-> Either (Basis (Diff (f p))) (Basis (Diff (g p))))
-> (Basis (Diff (g p)), Scalar (Diff (g p)))
-> (Either (Basis (Diff (f p))) (Basis (Diff (g p))),
Scalar (Diff (g p)))
forall (a :: * -> * -> *) b c d.
Arrow a =>
a b c -> a (b, d) (c, d)
first Basis (Diff (g p))
-> Either (Basis (Diff (f p))) (Basis (Diff (g p)))
forall a b. b -> Either a b
Right) (Diff (g p) -> [(Basis (Diff (g p)), Scalar (Diff (g p)))]
forall v. HasBasis v => v -> [(Basis v, Scalar v)]
decompose Diff (g p)
vg)
decompose' :: AffineDiffProductSpace f g p
-> Basis (AffineDiffProductSpace f g p)
-> Scalar (AffineDiffProductSpace f g p)
decompose' (AffineDiffProductSpace Diff (f p)
vf Diff (g p)
_) (Left bf) = Diff (f p) -> Basis (Diff (f p)) -> Scalar (Diff (f p))
forall v. HasBasis v => v -> Basis v -> Scalar v
decompose' Diff (f p)
vf Basis (Diff (f p))
bf
decompose' (AffineDiffProductSpace Diff (f p)
_ Diff (g p)
vg) (Right bg) = Diff (g p) -> Basis (Diff (g p)) -> Scalar (Diff (g p))
forall v. HasBasis v => v -> Basis v -> Scalar v
decompose' Diff (g p)
vg Basis (Diff (g p))
bg
instance AffineSpace a => AffineSpace (Gnrx.Rec0 a s) where
type Diff (Gnrx.Rec0 a s) = Diff a
Gnrx.K1 a
v .+^ :: Rec0 a s -> Diff (Rec0 a s) -> Rec0 a s
.+^ Diff (Rec0 a s)
w = a -> Rec0 a s
forall k i c (p :: k). c -> K1 i c p
Gnrx.K1 (a -> Rec0 a s) -> a -> Rec0 a s
forall a b. (a -> b) -> a -> b
$ a
v a -> Diff a -> a
forall p. AffineSpace p => p -> Diff p -> p
.+^ Diff a
Diff (Rec0 a s)
w
Gnrx.K1 a
v .-. :: Rec0 a s -> Rec0 a s -> Diff (Rec0 a s)
.-. Gnrx.K1 a
w = a
v a -> a -> Diff a
forall p. AffineSpace p => p -> p -> Diff p
.-. a
w
instance AffineSpace (f p) => AffineSpace (Gnrx.M1 i c f p) where
type Diff (Gnrx.M1 i c f p) = Diff (f p)
Gnrx.M1 f p
v .+^ :: M1 i c f p -> Diff (M1 i c f p) -> M1 i c f p
.+^ Diff (M1 i c f p)
w = f p -> M1 i c f p
forall k i (c :: Meta) (f :: k -> *) (p :: k). f p -> M1 i c f p
Gnrx.M1 (f p -> M1 i c f p) -> f p -> M1 i c f p
forall a b. (a -> b) -> a -> b
$ f p
v f p -> Diff (f p) -> f p
forall p. AffineSpace p => p -> Diff p -> p
.+^ Diff (f p)
Diff (M1 i c f p)
w
Gnrx.M1 f p
v .-. :: M1 i c f p -> M1 i c f p -> Diff (M1 i c f p)
.-. Gnrx.M1 f p
w = f p
v f p -> f p -> Diff (f p)
forall p. AffineSpace p => p -> p -> Diff p
.-. f p
w
instance (AffineSpace (f p), AffineSpace (g p)) => AffineSpace ((f :*: g) p) where
type Diff ((f:*:g) p) = AffineDiffProductSpace f g p
(f p
x:*:g p
y) .+^ :: (:*:) f g p -> Diff ((:*:) f g p) -> (:*:) f g p
.+^ AffineDiffProductSpace ξ υ = (f p
xf p -> Diff (f p) -> f p
forall p. AffineSpace p => p -> Diff p -> p
.+^Diff (f p)
ξ) f p -> g p -> (:*:) f g p
forall k (f :: k -> *) (g :: k -> *) (p :: k).
f p -> g p -> (:*:) f g p
:*: (g p
yg p -> Diff (g p) -> g p
forall p. AffineSpace p => p -> Diff p -> p
.+^Diff (g p)
υ)
(f p
x:*:g p
y) .-. :: (:*:) f g p -> (:*:) f g p -> Diff ((:*:) f g p)
.-. (f p
ξ:*:g p
υ) = Diff (f p) -> Diff (g p) -> AffineDiffProductSpace f g p
forall (f :: * -> *) (g :: * -> *) p.
Diff (f p) -> Diff (g p) -> AffineDiffProductSpace f g p
AffineDiffProductSpace (f p
xf p -> f p -> Diff (f p)
forall p. AffineSpace p => p -> p -> Diff p
.-.f p
ξ) (g p
yg p -> g p -> Diff (g p)
forall p. AffineSpace p => p -> p -> Diff p
.-.g p
υ)