{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE CPP #-}
{-# LANGUAGE DeriveDataTypeable #-}
{-# LANGUAGE DeriveFunctor #-}
{-# LANGUAGE ViewPatterns #-}
module Patience
(
diff
, Item(..)
, longestIncreasing
) where
import Data.Data (Data)
import qualified Data.Foldable as F
import qualified Data.IntMap as IM
import Data.List
import qualified Data.Map as M
import qualified Data.Map.Strict as MS
import Data.Ord
import Data.Sequence ( (<|), (|>), (><), ViewL(..), ViewR(..) )
import qualified Data.Sequence as S
import Data.Typeable (Typeable)
adjMove :: (a -> a) -> Int -> Int -> IM.IntMap a -> IM.IntMap a
adjMove f !xi !xf m = case IM.updateLookupWithKey (\_ _ -> Nothing) xi m of
(Just v, mm) -> IM.insert xf (f v) mm
(Nothing, _) -> m
data Card a = Card {-# UNPACK #-} !Int a (Maybe (Card a))
longestIncreasing :: [(Int,a)] -> [(Int,a)]
longestIncreasing = extract . F.foldl' ins IM.empty where
ins m (x,a) =
let (lt, gt) = IM.split x m
prev = (head . fst) `fmap` IM.maxView lt
new = Card x a prev
in case IM.minViewWithKey gt of
Nothing -> IM.insert x [new] m
Just ((k,_),_) -> adjMove (new:) k x m
extract (IM.maxView -> Just (c,_)) = walk $ head c
extract _ = []
walk (Card x a c) = (x,a) : maybe [] walk c
unique :: (Ord k) => S.Seq (a,k) -> M.Map k a
unique = M.mapMaybe id . F.foldr ins M.empty where
ins (a,x) = MS.insertWith (\_ _ -> Nothing) x (Just a)
solveLCS :: (Ord a) => S.Seq (Int,a) -> S.Seq (Int,a) -> [(Int,Int)]
solveLCS ma mb =
let xs = M.elems $ M.intersectionWith (,) (unique ma) (unique mb)
in longestIncreasing $ sortBy (comparing snd) xs
data Piece a
= Match a a
| Diff (S.Seq a) (S.Seq a)
deriving (Show)
chop :: S.Seq a -> S.Seq a -> [(Int,Int)] -> [Piece a]
chop xs ys []
| S.null xs && S.null ys = []
| otherwise = [Diff xs ys]
chop xs ys (!(!nx,!ny):ns) =
let (xsr, S.viewl -> (x :< xse)) = S.splitAt nx xs
(ysr, S.viewl -> (y :< yse)) = S.splitAt ny ys
in Diff xse yse : Match x y : chop xsr ysr ns
zipLS :: [a] -> S.Seq b -> S.Seq (a, b)
zipLS = S.zip . S.fromList
number :: S.Seq a -> S.Seq (Int,a)
number xs = zipLS [0..S.length xs - 1] xs
data Item a
= Old a
| New a
| Both a a
deriving (Eq, Ord, Show, Read, Typeable, Data, Functor)
diff :: (Ord a) => [a] -> [a] -> [Item a]
diff xsl ysl = F.toList $ go (S.fromList xsl) (S.fromList ysl) where
go (S.viewl -> (x :< xs)) (S.viewl -> (y :< ys))
| x == y = Both x y <| go xs ys
go (S.viewr -> (xs :> x)) (S.viewr -> (ys :> y))
| x == y = go xs ys |> Both x y
go xs ys = case chop xs ys $ solveLCS (number xs) (number ys) of
[Diff _ _] -> fmap Old xs >< fmap New ys
ps -> recur ps
recur [] = S.empty
recur (Match x y : ps) = recur ps |> Both x y
recur (Diff xs ys : ps) = recur ps >< go xs ys