module Haskus.Utils.HArray
( HArray
, HArrayIndex
, HArrayIndexT
, HArrayTryIndexT
, emptyHArray
, singleHArray
, getHArrayN
, getHArray0
, setHArrayN
, getHArrayT
, setHArrayT
, tryGetHArrayT
, appendHArray
, prependHArray
, concatHArray
, initHArray
, tailHArray
, HArrayT (..)
, (>~:~>)
)
where
import Data.Vector as V
import Unsafe.Coerce
import Haskus.Utils.Types.List
import Haskus.Utils.Types
import Haskus.Utils.Flow
data HArray (types :: [*]) = forall a. HArray (Vector a)
type role HArray representational
emptyHArray :: HArray '[]
emptyHArray = HArray V.empty
singleHArray :: a -> HArray '[a]
singleHArray = HArray . V.singleton
type HArrayIndex (n :: Nat) t (ts :: [*]) =
( KnownNat n
, t ~ Index n ts
, KnownNat (Length ts)
, CmpNat n (Length ts) ~ 'LT
)
type HArrayIndexT t (ts :: [*]) =
( IsMember t ts ~ 'True
, HArrayIndex (IndexOf t ts) t ts
)
type HArrayTryIndexT t (ts :: [*]) =
( HArrayIndex (MaybeIndexOf t ts) t (t ': ts)
)
getHArrayN :: forall (n :: Nat) (ts :: [*]) t.
( HArrayIndex n t ts) => HArray ts -> t
getHArrayN (HArray as) = unsafeCoerce (as ! natValue @n)
getHArray0 :: (HArrayIndex 0 t ts) => HArray ts -> t
getHArray0 = getHArrayN @0
setHArrayN :: forall (n :: Nat) (ts :: [*]) t.
(HArrayIndex n t ts) => t -> HArray ts -> HArray ts
setHArrayN a (HArray as) = HArray (as V.// [(natValue @n,unsafeCoerce a)])
getHArrayT :: forall t ts.
(HArrayIndexT t ts) => HArray ts -> t
getHArrayT = getHArrayN @(IndexOf t ts)
setHArrayT :: forall t ts.
(HArrayIndexT t ts) => t -> HArray ts -> HArray ts
setHArrayT = setHArrayN @(IndexOf t ts)
tryGetHArrayT :: forall t ts.
(HArrayTryIndexT t ts) => HArray ts -> Maybe t
tryGetHArrayT as = if n == 0
then Nothing
else Just $ getHArrayN @(MaybeIndexOf t ts) as'
where
n = natValue' @(MaybeIndexOf t ts)
as' :: HArray (t ': ts)
as' = prependHArray undefined as
appendHArray :: HArray ts -> t -> HArray (Snoc ts t)
appendHArray (HArray as) a = HArray (V.snoc as (unsafeCoerce a))
prependHArray :: t -> HArray ts -> HArray (t ': ts)
prependHArray a (HArray as) = HArray (V.cons (unsafeCoerce a) as)
concatHArray :: HArray ts1 -> HArray ts2 -> HArray (Concat ts1 ts2)
concatHArray (HArray as1) (HArray as2) = HArray (V.concat [as1,unsafeCoerce as2])
initHArray :: HArray ts -> HArray (Init ts)
initHArray (HArray as) = HArray (V.init as)
tailHArray :: HArray ts -> HArray (Tail ts)
tailHArray (HArray as) = HArray (V.tail as)
newtype HArrayT m xs ys = HArrayT
{ runHArrayT :: HArray xs -> m (HArray ys)
}
(>~:~>) :: (Monad m) => HArrayT m xs ys -> HArrayT m ys zs -> HArrayT m xs zs
(>~:~>) (HArrayT f) (HArrayT g) = HArrayT (f >=> g)