{-# LANGUAGE CPP #-}
{-# LANGUAGE DeriveFoldable #-}
{-# LANGUAGE DeriveFunctor #-}
{-# LANGUAGE DeriveTraversable #-}
{-# LANGUAGE Safe #-}
#ifndef MIN_VERSION_transformers
#define MIN_VERSION_transformers(x,y,z) 0
#endif
#ifndef MIN_VERSION_transformers_compat
#define MIN_VERSION_transformers_compat(x,y,z) 0
#endif
#if MIN_VERSION_base(4,9,0)
#define LIFTED_FUNCTOR_CLASSES 1
#elif MIN_VERSION_transformers(0,5,0)
#define LIFTED_FUNCTOR_CLASSES 1
#elif MIN_VERSION_transformers_compat(0,5,0) && !MIN_VERSION_transformers(0,4,0)
#define LIFTED_FUNCTOR_CLASSES 1
#endif
module Data.Function.Step (
SF (..),
Bound (..),
constant,
step,
fromList,
normalise,
(!),
values,
showSF,
putSF,
) where
import Control.Applicative (liftA2)
import Control.DeepSeq (NFData (..))
import Control.Monad (ap)
import Data.Functor.Classes
import Data.List (intercalate)
import Data.Map (Map)
import Prelude ()
import Prelude.Compat
#ifdef LIFTED_FUNCTOR_CLASSES
import Text.Show (showListWith)
#endif
import qualified Data.Map as Map
import qualified Test.QuickCheck as QC
data SF k v = SF !(Map (Bound k) v) !v
deriving (Eq, Ord, Functor, Foldable, Traversable)
data Bound k
= Open k
| Closed k
deriving (Eq, Show, Functor, Foldable, Traversable)
instance Ord k => Ord (Bound k) where
compare (Open k) (Open k') = compare k k'
compare (Closed k) (Closed k') = compare k k'
compare (Open k) (Closed k') = case compare k k' of
LT -> LT
EQ -> LT
GT -> GT
compare (Closed k) (Open k') = case compare k k' of
LT -> LT
EQ -> GT
GT -> GT
instance Ord k => Applicative (SF k) where
pure = constant
(<*>) = ap
instance Ord k => Monad (SF k) where
return = pure
SF m def0 >>= f = SF
(Map.fromDistinctAscList $ mkDistinctAscList $ pieces ++ pieces1)
def1
where
pieces =
[ (min k k', v')
| (k, v) <- Map.toList m
, let SF m' def = f v
, (k', v') <- Map.toList m' ++ [(k, def)]
]
(pieces1, def1) = let SF m' def = f def0 in (Map.toList m', def)
instance (Ord k, Semigroup v) => Semigroup (SF k v) where
(<>) = liftA2 (<>)
instance (Ord k, Monoid v) => Monoid (SF k v) where
mempty = pure mempty
mappend = liftA2 mappend
instance (Ord k, QC.Arbitrary k, QC.Arbitrary v) => QC.Arbitrary (SF k v) where
arbitrary = fromList <$> QC.arbitrary <*> QC.arbitrary
shrink (SF m v) = uncurry fromList <$> QC.shrink (Map.toList m, v)
instance QC.Arbitrary k => QC.Arbitrary (Bound k) where
arbitrary = QC.oneof [Open <$> QC.arbitrary, Closed <$> QC.arbitrary]
instance NFData k => NFData (Bound k) where
rnf (Open k) = rnf k
rnf (Closed k) = rnf k
instance (NFData k, NFData v) => NFData (SF k v) where
rnf (SF m v) = rnf (m, v)
#if LIFTED_FUNCTOR_CLASSES
instance Show2 SF where
liftShowsPrec2 spk slk spv slv d (SF m v) = showsBinaryWith
(\_ -> showListWith $ liftShowsPrec2 (liftShowsPrec spk slk) (liftShowList spk slk) spv slv 0)
spv
"fromList" d (Map.toList m) v
instance Show k => Show1 (SF k) where
liftShowsPrec = liftShowsPrec2 showsPrec showList
instance (Show k, Show v) => Show (SF k v) where
showsPrec = showsPrec2
instance Show1 Bound where
liftShowsPrec sp _ d (Open k) = showsUnaryWith sp "Open" d k
liftShowsPrec sp _ d (Closed k) = showsUnaryWith sp "Closed" d k
#else
instance (Show k, Show v) => Show (SF k v) where
showsPrec d (SF m v) = showParen (d > 10)
$ showString "fromList"
. showsPrec 11 (Map.toList m)
. showChar ' '
. showsPrec 11 v
instance Show k => Show1 (SF k) where showsPrec1 = showsPrec
instance Show1 Bound where showsPrec1 = showsPrec
#endif
mkDistinctAscList :: Ord k => [(k, b)] -> [(k, b)]
mkDistinctAscList [] = []
mkDistinctAscList ((k, v) : kv) = (k, v) : mkDistinctAscList' k kv
mkDistinctAscList' :: Ord k => k -> [(k, b)] -> [(k, b)]
mkDistinctAscList' _ [] = []
mkDistinctAscList' k (p@(k', _) : kv)
| k < k' = p : mkDistinctAscList' k' kv
| otherwise = mkDistinctAscList' k kv
infixl 9 !
(!) :: Ord k => SF k v -> k -> v
SF m def ! x = case Map.lookupGE (Closed x) m of
Nothing -> def
Just (_, v) -> v
constant :: a -> SF k a
constant = SF Map.empty
step :: k -> v -> v -> SF k v
step k = SF . Map.singleton (Open k)
fromList :: Ord k => [(Bound k, v)] -> v -> SF k v
fromList = SF . Map.fromList
values :: SF k v -> [v]
values (SF m v) = Map.elems m ++ [v]
normalise :: Eq v => SF k v -> SF k v
normalise (SF m v) = uncurry mk $ foldr go ([], v) (Map.toList m) where
mk m' _ = SF (Map.fromDistinctAscList m') v
go p@(_, v') p'@(m', x)
| v' == x = p'
| otherwise = (p : m', v')
showSF :: (Show a, Show b) => SF a b -> String
showSF (SF m v) | Map.null m = "\\_ -> " ++ show v
showSF (SF m v) = intercalate "\n" $
"\\x -> if" : [ " | " ++ leftPad k ++ " -> " ++ x | (k, x) <- cases ]
where
cases = cases' ++ [ ("otherwise", show v) ]
m' = Map.toList m
cases' = case traverse fromOpen m' of
Nothing -> [ ("x " ++ showBound k, show x) | (k, x) <- m' ]
Just m'' -> [ ("x < " ++ show k, show x) | (k, x) <- m'' ]
fromOpen (Open k, x) = Just (k, x)
fromOpen _ = Nothing
len = maximum (map (length . fst) cases)
leftPad s = s ++ replicate (len - length s) ' '
showBound :: Show k => Bound k -> String
showBound (Open k) = "< " ++ showsPrec 5 k ""
showBound (Closed k) = "<= " ++ showsPrec 5 k ""
putSF :: (Show a, Show b) => SF a b -> IO ()
putSF = putStrLn . showSF