module System.Console.Haskeline.Command.Undo where
import System.Console.Haskeline.Command
import System.Console.Haskeline.LineState
import System.Console.Haskeline.Monads
import Control.Monad
data Undo = Undo {pastUndo, futureRedo :: [InsertMode]}
type UndoT = StateT Undo
runUndoT :: Monad m => UndoT m a -> m a
runUndoT = evalStateT' initialUndo
initialUndo :: Undo
initialUndo = Undo {pastUndo = [emptyIM], futureRedo = []}
saveToUndo :: Save s => s -> Undo -> Undo
saveToUndo s undo
| not isSame = Undo {pastUndo = toSave:pastUndo undo,futureRedo=[]}
| otherwise = undo
where
toSave = save s
isSame = case pastUndo undo of
u:_ | u == toSave -> True
_ -> False
undoPast, redoFuture :: Save s => s -> Undo -> (s,Undo)
undoPast ls u@Undo {pastUndo = []} = (ls,u)
undoPast ls u@Undo {pastUndo = (pastLS:lss)}
= (restore pastLS, u {pastUndo = lss, futureRedo = save ls : futureRedo u})
redoFuture ls u@Undo {futureRedo = []} = (ls,u)
redoFuture ls u@Undo {futureRedo = (futureLS:lss)}
= (restore futureLS, u {futureRedo = lss, pastUndo = save ls : pastUndo u})
saveForUndo :: (Save s, MonadState Undo m)
=> Command m s s
saveForUndo s = do
modify (saveToUndo s)
return s
commandUndo, commandRedo :: (MonadState Undo m, Save s) => Command m s s
commandUndo = simpleCommand $ liftM Right . update . undoPast
commandRedo = simpleCommand $ liftM Right . update . redoFuture