module Darcs.Patch.Invert
( Invert(..), invertFL, invertRL, dropInverses
)
where
import Darcs.Prelude
import Darcs.Patch.Witnesses.Ordered
( FL(..), RL(..), reverseFL, reverseRL, (:>)(..) )
import Darcs.Patch.Witnesses.Eq ( EqCheck(IsEq), Eq2((=\/=)) )
class Invert p where
invert :: p wX wY -> p wY wX
invertFL :: Invert p => FL p wX wY -> RL p wY wX
invertFL NilFL = NilRL
invertFL (x:>:xs) = invertFL xs :<: invert x
invertRL :: Invert p => RL p wX wY -> FL p wY wX
invertRL NilRL = NilFL
invertRL (xs:<:x) = invert x :>: invertRL xs
instance Invert p => Invert (FL p) where
invert = reverseRL . invertFL
instance Invert p => Invert (RL p) where
invert = reverseFL . invertRL
instance Invert p => Invert (p :> p) where
invert (a :> b) = invert b :> invert a
dropInverses :: (Invert p, Eq2 p) => FL p wX wY -> Maybe (FL p wX wY)
dropInverses (x :>: y :>: z)
| IsEq <- invert x =\/= y = Just z
| otherwise = do
yz <- dropInverses (y :>: z)
dropInverses (x :>: yz)
dropInverses _ = Nothing