module Darcs.Patch.TouchesFiles
( lookTouch
, chooseTouching
, deselectNotTouching
, selectNotTouching
) where
import Darcs.Prelude
import Data.List ( nub )
import Darcs.Patch.Apply
(Apply, ApplyState, applyToPaths)
import Darcs.Patch.Choices
(PatchChoices, Label, LabelledPatch, patchChoices, label,
getChoices, forceFirsts, forceLasts, unLabel)
import Darcs.Patch.Commute (Commute)
import Darcs.Patch.Inspect (PatchInspect)
import Darcs.Patch.Witnesses.Ordered
(FL(..), (:>)(..), mapFL_FL, (+>+))
import Darcs.Patch.Witnesses.Sealed (Sealed, seal)
import Darcs.Util.Path (AnchoredPath, isPrefix)
import Darcs.Util.Tree (Tree)
labelTouching
:: (Apply p, PatchInspect p, ApplyState p ~ Tree)
=> Bool -> [AnchoredPath] -> FL (LabelledPatch p) wX wY -> [Label]
labelTouching _ _ NilFL = []
labelTouching wantTouching fs (lp :>: lps) =
case lookTouchOnlyEffect fs (unLabel lp) of
(doesTouch, fs') ->
let rest = labelTouching wantTouching fs' lps
in (if doesTouch == wantTouching
then (label lp :)
else id)
rest
labelNotTouchingFM
:: (Apply p, Commute p, PatchInspect p, ApplyState p ~ Tree)
=> [AnchoredPath] -> PatchChoices p wX wY -> [Label]
labelNotTouchingFM paths pc =
case getChoices pc of
fc :> mc :> _ -> labelTouching False paths (fc +>+ mc)
selectTouching
:: (Apply p, Commute p, PatchInspect p, ApplyState p ~ Tree)
=> Maybe [AnchoredPath] -> PatchChoices p wX wY -> PatchChoices p wX wY
selectTouching Nothing pc = pc
selectTouching (Just paths) pc = forceFirsts xs pc
where
xs =
case getChoices pc of
_ :> mc :> lc -> labelTouching True paths (mc +>+ lc)
deselectNotTouching
:: (Apply p, Commute p, PatchInspect p, ApplyState p ~ Tree)
=> Maybe [AnchoredPath] -> PatchChoices p wX wY -> PatchChoices p wX wY
deselectNotTouching Nothing pc = pc
deselectNotTouching (Just paths) pc =
forceLasts (labelNotTouchingFM paths pc) pc
selectNotTouching
:: (Apply p, Commute p, PatchInspect p, ApplyState p ~ Tree)
=> Maybe [AnchoredPath] -> PatchChoices p wX wY -> PatchChoices p wX wY
selectNotTouching Nothing pc = pc
selectNotTouching (Just paths) pc = forceFirsts (labelNotTouchingFM paths pc) pc
chooseTouching
:: (Apply p, Commute p, PatchInspect p, ApplyState p ~ Tree)
=> Maybe [AnchoredPath] -> FL p wX wY -> Sealed (FL p wX)
chooseTouching Nothing p = seal p
chooseTouching paths p =
case getChoices $ selectTouching paths $ patchChoices p of
fc :> _ :> _ -> seal $ mapFL_FL unLabel fc
lookTouchOnlyEffect
:: (Apply p, ApplyState p ~ Tree)
=> [AnchoredPath] -> p wX wY -> (Bool, [AnchoredPath])
lookTouchOnlyEffect fs p = (wasTouched, fs')
where
(wasTouched, _, fs', _) = lookTouch Nothing fs p
lookTouch
:: (Apply p, ApplyState p ~ Tree)
=> Maybe [(AnchoredPath, AnchoredPath)]
-> [AnchoredPath]
-> p wX wY
-> (Bool, [AnchoredPath], [AnchoredPath], [(AnchoredPath, AnchoredPath)])
lookTouch renames fs p = (anyTouched, touchedFs, fs', renames')
where
touchedFs = nub . concatMap fsAffectedBy $ affected
fsAffectedBy af = filter (affectedBy af) fs
anyTouched = length touchedFs > 0
affectedBy :: AnchoredPath -> AnchoredPath -> Bool
touched `affectedBy` f =
touched `isPrefix` f || f `isPrefix` touched
(affected, fs', renames') = applyToPaths p renames fs