{-# LANGUAGE TemplateHaskell, RankNTypes, TypeSynonymInstances, FlexibleInstances, OverloadedStrings, GeneralizedNewtypeDeriving, FlexibleContexts #-} module Data.Text.Region.Types ( Point(..), pointLine, pointColumn, pointRegion, Size, (.-.), (.+.), Region(..), regionFrom, regionTo, Map(..), Contents, emptyContents, concatCts, splitCts, splitted, Editable(..), contents, by, measure, Replace(..), replaceRegion, replaceWith, Edit(..), replaces, Regioned(..), module Data.Group ) where import Prelude hiding (id, (.)) import Prelude.Unicode import Control.Category import Control.Lens hiding ((.=)) import Data.Aeson import qualified Data.ByteString.Lazy.Char8 as L import Data.Group import Data.List import Data.Text (Text) import qualified Data.Text as T -- | Point at text: zero-based line and column data Point = Point { _pointLine ∷ Int, _pointColumn ∷ Int } deriving (Eq, Ord, Read, Show) makeLenses ''Point -- | As empty region pointRegion ∷ Iso' Point Region pointRegion = iso (\p → Region p p) _regionFrom instance ToJSON Point where toJSON (Point l c) = object ["line" .= l, "column" .= c] instance FromJSON Point where parseJSON = withObject "point" $ \v → Point <$> v .: "line" <*> v .: "column" instance Monoid Point where mempty = Point 0 0 Point l c `mappend` Point bl bc | l ≡ 0 = Point bl (c + bc) | otherwise = Point (l + bl) c instance Group Point where invert (Point l c) = Point (negate l) (negate c) -- | Distance between 'Point's is measured in lines and columns. -- And it is defined, that distance between point at l:c and point (l + 1):0 is one line no matter c is -- because we need to go to new line to reach destination point -- Columns are taken into account only if points are on the same line type Size = Point -- | @pt .-. base@ is distance from @base@ to @pt@ -- Distance can't be less then zero lines and columns (.-.) ∷ Point → Point → Point Point l c .-. Point bl bc | bl < l = Point (l - bl) c | bl ≡ l = Point 0 (max 0 (c - bc)) | otherwise = Point 0 0 -- | Opposite to '.-.' -- -- prop> (pt .-. base) .+. base = pt (.+.) ∷ Point → Point → Point (Point l c) .+. (Point bl bc) | l ≡ 0 = Point bl (c + bc) | otherwise = Point (l + bl) c -- | Region from 'Point' to another data Region = Region { _regionFrom ∷ Point, _regionTo ∷ Point } deriving (Eq, Ord, Read, Show) makeLenses ''Region instance ToJSON Region where toJSON (Region f t) = object ["from" .= f, "to" .= t] instance FromJSON Region where parseJSON = withObject "region" $ \v -> Region <$> v .: "from" <*> v .: "to" -- | Main idea is that there are only two basic actions, that changes regions: inserting and cutting -- When something is cutted out or inserted in, 'Region' positions must be updated -- All editings can be represented as many cuts and inserts, so we can combine them to get function -- which maps source regions to regions on updated data -- Because insert is dual to cut (and therefore composes something like iso), we can also get function to map regions back -- Combining this functions while edit, we get function, that maps regions from source data to edited one -- To get back function, we must also combine opposite actions, or we can represent actions as 'Iso' -- Same idea goes for modifying contents, represent each action as isomorphism and combine them together -- This works if we don't use overlapped regions newtype Map = Map { mapIso :: Iso' Region Region } instance Monoid Map where mempty = Map $ iso id id Map l `mappend` Map r = Map (r . l) instance Group Map where invert (Map f) = Map (from f) -- | Contents is list of lines, list must have at least one (maybe empty) line type Contents a = [a] -- | Empty contents are contents with one empty line emptyContents ∷ Monoid a ⇒ Contents a emptyContents = [mempty] checkCts ∷ Contents a → Contents a checkCts [] = error "Contents can't be empty" checkCts cs = cs concatCts ∷ Monoid a ⇒ Contents a → Contents a → Contents a concatCts ls rs = init (checkCts ls) ++ [last (checkCts ls) `mappend` head (checkCts rs)] ++ tail (checkCts rs) -- | Split 'Contents' at some 'Point' splitCts ∷ Editable a ⇒ Point → Contents a → (Contents a, Contents a) splitCts (Point l c) cts = (take l cts ++ [p], s : drop (succ l) cts) where (p, s) = splitContents c (cts !! l) -- | Get splitted 'Contents' at some 'Point' splitted ∷ Editable a ⇒ Point → Iso' (Contents a) (Contents a, Contents a) splitted p = iso (splitCts p) (uncurry concatCts) -- | Something editable, string types implements this class Monoid a ⇒ Editable a where -- | Split editable at some position splitContents ∷ Int → a → (a, a) contentsLength ∷ a → Int splitLines ∷ a → [a] joinLines ∷ [a] → a -- | Get 'Contents' for some 'Editable', splitting lines contents ∷ (Editable a, Editable b) ⇒ Iso a b (Contents a) (Contents b) contents = iso splitLines joinLines by ∷ Editable a ⇒ a → Contents a by = splitLines instance Editable String where splitContents = splitAt contentsLength = length splitLines s = case break (≡ '\n') s of (pre', "") → [pre'] (pre', _:post') → pre' : splitLines post' joinLines = intercalate "\n" instance Editable Text where splitContents = T.splitAt contentsLength = T.length splitLines = T.split (≡ '\n') joinLines = T.intercalate "\n" -- | Contents 'Size' measure ∷ Editable s ⇒ Contents s → Size measure [] = error "Invalid input" measure cts = Point (pred $ length cts) (contentsLength $ last cts) -- | Serializable edit action data Replace s = Replace { -- | 'Region' to replace _replaceRegion ∷ Region, -- | 'Contents' to replace with _replaceWith ∷ Contents s } deriving (Eq) makeLenses ''Replace instance (Editable s, ToJSON s) ⇒ ToJSON (Replace s) where toJSON (Replace e c) = object ["region" .= e, "contents" .= view (from contents) c] instance (Editable s, FromJSON s) ⇒ FromJSON (Replace s) where parseJSON = withObject "edit" $ \v → Replace <$> v .: "region" <*> (view contents <$> v .: "contents") instance (Editable s, ToJSON s) ⇒ Show (Replace s) where show = L.unpack ∘ encode -- | Edit is several replace actions, applied simultaneously, must not overlap newtype Edit s = Edit { _replaces ∷ [Replace s] } deriving (Eq, Show, Monoid) makeLenses ''Edit instance (Editable s, ToJSON s) ⇒ ToJSON (Edit s) where toJSON = toJSON ∘ _replaces instance (Editable s, FromJSON s) ⇒ FromJSON (Edit s) where parseJSON = fmap Edit ∘ parseJSON class Regioned a where regions ∷ Traversal' a Region instance Regioned Point where regions = pointRegion instance Regioned Region where regions = id instance Regioned (Replace s) where regions = replaceRegion instance Regioned (Edit s) where regions = replaces . each . replaceRegion