{-# LANGUAGE RankNTypes, ScopedTypeVariables #-} module Alignment ( hasChanges , numberedRows , alignDiff , alignBranch , applyThese , modifyJoin ) where import Data.Bifunctor (bimap, first, second) import Control.Arrow ((***)) import Control.Monad (join) import Control.Monad.Free (wrap) import Data.Align import Data.Bifunctor.Join import Data.Foldable (toList) import Data.Function (on) import Data.Functor.Both import Data.Functor.Foldable import Data.Functor.Identity import Data.List (partition, sortBy) import Data.Maybe (catMaybes, fromJust, fromMaybe, listToMaybe) import Data.Range import Data.Semigroup ((<>)) import Data.Source import Data.Record import Data.These import Diff import Info import Patch import Prelude hiding (fst, snd) import SplitDiff import Term -- | Assign line numbers to the lines on each side of a list of rows. numberedRows :: [Join These a] -> [Join These (Int, a)] numberedRows = countUp (both 1 1) where countUp _ [] = [] countUp from (row : rows) = numberedLine from row : countUp (nextLineNumbers from row) rows numberedLine from row = fromJust ((,) <$> modifyJoin (uncurry These) from `applyThese` row) nextLineNumbers from row = modifyJoin (fromThese id id) (succ <$ row) <*> from -- | Determine whether a line contains any patches. hasChanges :: (Foldable f, Functor f) => SplitDiff f annotation -> Bool hasChanges = or . (True <$) -- | Align a Diff into a list of Join These SplitDiffs representing the (possibly blank) lines on either side. alignDiff :: Traversable f => HasField fields Range => Both Source -> Diff f (Record fields) -> [Join These (SplitDiff [] (Record fields))] alignDiff sources = evalDiff $ \ diff env -> case diff of Copy _ ann r -> alignSyntax (runBothWith ((Join .) . These)) wrap getRange sources (ann :< r) Var v -> fromMaybe [] (envLookup v env) Patch patch -> alignPatch sources patch -- | Align the contents of a patch into a list of lines on the corresponding side(s) of the diff. alignPatch :: forall fields f. (Traversable f, HasField fields Range) => Both Source -> Patch (Term f (Record fields)) -> [Join These (SplitDiff [] (Record fields))] alignPatch sources patch = case patch of Delete term -> fmap (pure . SplitDelete) <$> alignSyntax' this (fst sources) term Insert term -> fmap (pure . SplitInsert) <$> alignSyntax' that (snd sources) term Replace term1 term2 -> fmap (pure . SplitReplace) <$> alignWith (fmap (these id id const . runJoin) . Join) (alignSyntax' this (fst sources) term1) (alignSyntax' that (snd sources) term2) where getRange = byteRange . extract alignSyntax' :: (forall a. Identity a -> Join These a) -> Source -> Term f (Record fields) -> [Join These (Term [] (Record fields))] alignSyntax' side source = hylo (alignSyntax side Term getRange (Identity source)) unTerm . fmap Identity this = Join . This . runIdentity that = Join . That . runIdentity -- | The Applicative instance f is either Identity or Both. Identity is for Terms in Patches, Both is for Diffs in unchanged portions of the diff. alignSyntax :: (Applicative f, HasField fields Range, Foldable g) => (forall a. f a -> Join These a) -> (TermF [] (Record fields) term -> term) -> (term -> Range) -> f Source -> TermF g (f (Record fields)) [Join These term] -> [Join These term] alignSyntax toJoinThese toNode getRange sources (infos :< syntax) = catMaybes $ wrapInBranch <$> alignBranch getRange (join (toList syntax)) bothRanges where bothRanges = modifyJoin (fromThese [] []) lineRanges lineRanges = toJoinThese $ sourceLineRangesWithin . byteRange <$> infos <*> sources wrapInBranch = applyThese $ toJoinThese (makeNode <$> infos) makeNode info (range, children) = toNode (setByteRange info range :< children) -- | Given a function to get the range, a list of already-aligned children, and the lists of ranges spanned by a branch, return the aligned lines. alignBranch :: (term -> Range) -> [Join These term] -> Both [Range] -> [Join These (Range, [term])] -- There are no more ranges, so we’re done. alignBranch _ _ (Join ([], [])) = [] -- There are no more children, so we can just zip the remaining ranges together. alignBranch _ [] ranges = runBothWith (alignWith Join) (fmap (flip (,) []) <$> ranges) -- There are both children and ranges, so we need to proceed line by line alignBranch getRange children ranges = case intersectingChildren of -- No child intersects the current ranges on either side, so advance. [] -> (flip (,) [] <$> headRanges) : alignBranch getRange children (drop 1 <$> ranges) -- At least one child intersects on at least one side. _ -> case intersectionsWithHeadRanges <$> listToMaybe symmetricalChildren of -- At least one child intersects on both sides, so align symmetrically. Just (True, True) -> let (line, remaining) = lineAndRemaining intersectingChildren (Just headRanges) in line $ alignBranch getRange (remaining <> nonIntersectingChildren) (drop 1 <$> ranges) -- A symmetrical child intersects on the right, so align asymmetrically on the left. Just (False, True) -> alignAsymmetrically leftRange first -- A symmetrical child intersects on the left, so align asymmetrically on the right. Just (True, False) -> alignAsymmetrically rightRange second -- No symmetrical child intersects, so align asymmetrically, picking the left side first to match the deletion/insertion order convention in diffs. _ -> if any (isThis . runJoin) asymmetricalChildren then alignAsymmetrically leftRange first else alignAsymmetrically rightRange second where (intersectingChildren, nonIntersectingChildren) = partition (or . intersects getRange headRanges) children (symmetricalChildren, asymmetricalChildren) = partition (isThese . runJoin) intersectingChildren intersectionsWithHeadRanges = fromThese True True . runJoin . intersects getRange headRanges Just headRanges = Join <$> bisequenceL (runJoin (listToMaybe <$> Join (runBothWith These ranges))) (leftRange, rightRange) = splitThese headRanges alignAsymmetrically range advanceBy = let (line, remaining) = lineAndRemaining asymmetricalChildren range in line $ alignBranch getRange (remaining <> symmetricalChildren <> nonIntersectingChildren) (modifyJoin (advanceBy (drop 1)) ranges) lineAndRemaining _ Nothing = (id, []) lineAndRemaining children (Just ranges) = let (intersections, remaining) = alignChildren getRange children ranges in ((:) $ (,) <$> ranges `applyToBoth` (sortBy (compare `on` getRange) <$> intersections), remaining) -- | Given a list of aligned children, produce lists of their intersecting first lines, and a list of the remaining lines/nonintersecting first lines. alignChildren :: (term -> Range) -> [Join These term] -> Join These Range -> (Both [term], [Join These term]) alignChildren _ [] _ = (both [] [], []) alignChildren getRange (first:rest) headRanges | ~(l, r) <- splitThese first = case intersectionsWithHeadRanges first of -- It intersects on both sides, so we can just take the first line whole. (True, True) -> ((<>) <$> toTerms first <*> firstRemaining, restRemaining) -- It only intersects on the left, so split it up. (True, False) -> ((<>) <$> toTerms (fromJust l) <*> firstRemaining, maybe id (:) r restRemaining) -- It only intersects on the right, so split it up. (False, True) -> ((<>) <$> toTerms (fromJust r) <*> firstRemaining, maybe id (:) l restRemaining) -- It doesn’t intersect at all, so skip it and move along. (False, False) -> (firstRemaining, first:restRemaining) | otherwise = alignChildren getRange rest headRanges where (firstRemaining, restRemaining) = alignChildren getRange rest headRanges toTerms line = modifyJoin (fromThese [] []) (pure <$> line) intersectionsWithHeadRanges = fromThese False False . runJoin . intersects getRange headRanges -- | Test ranges and terms for intersection on either or both sides. intersects :: (term -> Range) -> Join These Range -> Join These term -> Join These Bool intersects getRange ranges line = intersectsRange <$> ranges `applyToBoth` modifyJoin (fromThese (Range (-1) (-1)) (Range (-1) (-1))) (getRange <$> line) -- | Split a These value up into independent These values representing the left and right sides, if any. splitThese :: Join These a -> (Maybe (Join These a), Maybe (Join These a)) splitThese these = fromThese Nothing Nothing $ bimap (Just . Join . This) (Just . Join . That) (runJoin these) infixl 4 `applyThese` -- | Like `<*>`, but it returns its result in `Maybe` since the result is the intersection of the shapes of the inputs. applyThese :: Join These (a -> b) -> Join These a -> Maybe (Join These b) applyThese (Join fg) (Join ab) = fmap Join . uncurry maybeThese $ uncurry (***) (bimap (<*>) (<*>) (unpack fg)) (unpack ab) where unpack = fromThese Nothing Nothing . bimap Just Just infixl 4 `applyToBoth` -- | Like `<*>`, but it takes a `Both` on the right to ensure that it can always return a value. applyToBoth :: Join These (a -> b) -> Both a -> Join These b applyToBoth (Join fg) (Join (a, b)) = Join $ these (This . ($ a)) (That . ($ b)) (\ f g -> These (f a) (g b)) fg -- Map over the bifunctor inside a Join, producing another Join. modifyJoin :: (p a a -> q b b) -> Join p a -> Join q b modifyJoin f = Join . f . runJoin -- | Given a pair of Maybes, produce a These containing Just their values, or Nothing if they haven’t any. maybeThese :: Maybe a -> Maybe b -> Maybe (These a b) maybeThese (Just a) (Just b) = Just (These a b) maybeThese (Just a) _ = Just (This a) maybeThese _ (Just b) = Just (That b) maybeThese _ _ = Nothing