{-# LANGUAGE OverloadedStrings #-} module Main where import Data.Semigroup import Data.Text (Text) import qualified Data.Text as Text import Test.QuickCheck.Function as QC import Test.Tasty import Test.Tasty.QuickCheck import qualified Data.Rope.UTF16 as Rope import qualified Data.Rope.UTF16.Internal.Text as Rope main :: IO () main = defaultMain $ testGroup "Tests" [ testProperty "Eq matches Text" $ \s1 s2 -> do let t1 = Text.pack s1 t2 = Text.pack s2 (Rope.fromText t1 == Rope.fromText t2) == (t1 == t2) , testProperty "Ord matches Text" $ \s1 s2 -> do let t1 = Text.pack s1 t2 = Text.pack s2 (Rope.fromText t1 `compare` Rope.fromText t2) == (t1 `compare` t2) , testProperty "toText . fromText = id" $ \s -> do let t = Text.pack s Rope.toText (Rope.fromText t) == t , testProperty "append matches Text" $ \s1 s2 -> do let t1 = Text.pack s1 t2 = Text.pack s2 Rope.toText (Rope.fromText t1 <> Rope.fromText t2) == t1 <> t2 , testProperty "length is UTF-16 code units" $ \s -> do let t = Text.pack s Rope.length (Rope.fromText t) == Rope.lengthWord16 t , testProperty "splitAt matches Text" $ \s i -> do let t = Text.pack s (r1, r2) = Rope.splitAt i $ Rope.fromText t (Rope.toText r1, Rope.toText r2) == Rope.split16At i t , testProperty "take matches Text" $ \s i -> do let t = Text.pack s Rope.toText (Rope.take i $ Rope.fromText t) == Rope.take16 i t , testProperty "drop matches Text" $ \s i -> do let t = Text.pack s Rope.toText (Rope.drop i $ Rope.fromText t) == Rope.drop16 i t , testProperty "rowColumnCodeUnits first line" $ \s i8 -> do let t = Text.pack $ takeWhile (/= '\n') s i16 = Rope.index8To16 i8 t i16 == Rope.rowColumnCodeUnits (Rope.RowColumn 0 i16) (Rope.fromText t) , testProperty "rowColumnCodeUnits subsequent lines" $ \s (NonNegative newlines) (NonNegative i8) -> do let t = Text.pack $ takeWhile (/= '\n') s newlinest = Text.replicate newlines "\n" <> t i16 = Rope.index8To16 i8 t newlines + i16 == Rope.rowColumnCodeUnits (Rope.RowColumn newlines i16) (Rope.fromText newlinest) , testProperty "codeUnitsRowColumn" $ \s i -> do let t = Text.pack s rowColumn = Rope.codeUnitsRowColumn i (Rope.fromText t) codeUnitsRowColumnSpec i t == rowColumn , testProperty "splitAtLine" $ \s i -> do let t = Text.pack s (rows1', rows2') = Rope.splitAtLine i (Rope.fromText t) splitAtLineSpec i t == (Rope.toText rows1', Rope.toText rows2') , testProperty "span matches Text" $ \s p -> do let t = Text.pack s f = QC.applyFun p (r1, r2) = Rope.span f $ Rope.fromText t (Rope.toText r1, Rope.toText r2) == Text.span f t , testProperty "break matches Text" $ \s p -> do let t = Text.pack s f = QC.applyFun p (r1, r2) = Rope.break f $ Rope.fromText t (Rope.toText r1, Rope.toText r2) == Text.break f t , testProperty "takeWhile matches Text" $ \s p -> do let t = Text.pack s f = QC.applyFun p Rope.toText (Rope.takeWhile f $ Rope.fromText t) == Text.takeWhile f t , testProperty "dropWhile matches Text" $ \s p -> do let t = Text.pack s f = QC.applyFun p Rope.toText (Rope.dropWhile f $ Rope.fromText t) == Text.dropWhile f t , testProperty "null matches Text" $ \s -> do let t = Text.pack s Rope.null (Rope.fromText t) == Text.null t , testProperty "rows is number of newlines" $ \s -> do let t = Text.pack s newlines = length $ filter (== '\n') s Rope.rows (Rope.fromText t) == newlines , testProperty "columns is number of code units of last line" $ \s -> do let t = Text.pack s lastLine = Text.takeWhileEnd (/= '\n') t cols = Rope.lengthWord16 lastLine Rope.columns (Rope.fromText t) == cols , testProperty "foldl matches Text" $ \s p a -> do let t = Text.pack s f = QC.applyFun2 p Rope.foldl f (a :: Int) (Rope.fromText t) == Text.foldl f a t , testProperty "foldl' matches Text" $ \s p a -> do let t = Text.pack s f = QC.applyFun2 p Rope.foldl' f (a :: Int) (Rope.fromText t) == Text.foldl' f a t , testProperty "foldr matches Text" $ \s p a -> do let t = Text.pack s f = QC.applyFun2 p Rope.foldr f (a :: Int) (Rope.fromText t) == Text.foldr f a t , testProperty "any matches Text" $ \s p -> do let t = Text.pack s f = QC.applyFun p Rope.any f (Rope.fromText t) == Text.any f t , testProperty "all matches Text" $ \s p -> do let t = Text.pack s f = QC.applyFun p Rope.all f (Rope.fromText t) == Text.all f t , testProperty "map matches Text" $ \s p -> do let t = Text.pack s f = QC.applyFun p Rope.toText (Rope.map f (Rope.fromText t)) == Text.map f t , testProperty "intercalate matches Text" $ \s ss -> do let t = Text.pack s ts = Text.pack <$> ss Rope.toText (Rope.intercalate (Rope.fromText t) (Rope.fromText <$> ts)) == Text.intercalate t ts ] splitAtLineSpec :: Int -> Text -> (Text, Text) splitAtLineSpec i t = case Text.splitOn "\n" t of [] -> ("", "") ts -> do let (pres, posts) = splitAt i (fmap (<> "\n") (init ts) <> [last ts]) (Text.concat pres, Text.concat posts) codeUnitsRowColumnSpec :: Int -> Text -> Rope.RowColumn codeUnitsRowColumnSpec i t = r where ts = Text.splitOn "\n" (Rope.take16 i t) r = case ts of [] -> Rope.RowColumn 0 0 [""] -> Rope.RowColumn 0 0 _ -> do Rope.RowColumn (length (init ts)) (Rope.lengthWord16 (last ts))