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