module Data.String.Interpolation(str,endline,tab) where
import Language.Haskell.TH as TH
import Language.Haskell.TH.Quote
import Language.Haskell.Meta
import Data.Data
import Data.Maybe
import Data.Char
import Data.List(intercalate)
quoteExprExp :: String -> TH.ExpQ
str :: QuasiQuoter
str = QuasiQuoter quoteExprExp undefined
debugStr :: QuasiQuoter
debugStr = QuasiQuoter (stringE) undefined
endline :: String
endline = "\n"
tab :: String
tab = "\t"
quoteExprExp s = psToStringE (parParse $ norming s)
psToStringE :: PieceString -> TH.Q TH.Exp
psToStringE [] = TH.stringE ""
psToStringE (x:xs) = TH.infixE (Just $ sbitToExp x)
([| (++) |])
(Just $ psToStringE xs)
runEither :: (Monad m) => [Char] -> Either [Char] t -> m t
runEither _ (Right x) = return x
runEither s (Left e) = error $ s ++" : "++ e
appM :: (Monad m) => String -> m Exp
appM expr = runEither ("Parse error in antiquote <"++expr++">") (parseExp expr)
sbitToExp :: StringBits -> ExpQ
sbitToExp (Str x) = TH.stringE x
sbitToExp (Var x) = appM x
sbitToExp (SVar x) = TH.appE (TH.varE (TH.mkName "show"))
(sbitToExp (Var x))
sbitToExp (RepVar varName lstName rep sep)
= [|intercalate $(sepr) (map $(lam) $(lstN))|]
where
sepr = psToStringE (fromMaybe [Str ""] sep)
lam = TH.lamE [varN] (psToStringE rep)
varN = runEither ("Parse error in repetition pattern <"++varName++">")
(parsePat varName)
lstN = runEither ("Parse error in repetition binding <"++lstName++">")
(parseExp lstName)
type PieceString = [StringBits]
data StringBits = Str String | Var String | SVar String
| RepVar String String
PieceString (Maybe PieceString)
deriving (Eq,Ord,Show,Typeable,Data)
parParse :: [Char] -> [StringBits]
parParse [] = []
parParse ('$':'$':s) = Str "$":parParse s
parParse ('#':'#':s) = Str "#":parParse s
parParse ('$':':':s) = SVar (takeWhile (/='$') s)
:parParse (drop 1 (dropWhile (/='$') s))
parParse ('$':s) = Var (takeWhile (/='$') s)
:parParse (drop 1 (dropWhile (/='$') s))
parParse ('#':s) = let (bind,exprS) = escapingBreak (':')
("Repetition <"++s++"> missing body")
$ s
(hasSep,expr,sepS) = takeRep
("Repetition <"++s++"> missing #")$ exprS
(sep,restS) = escapingBreak ('#')
("Repetition <"++s++"> missing #")
$ sepS
rest = restS
(varNameS,listNameS) = break (=="in") . words
$ bind
listName = unwords . drop 1 $ listNameS
varName = unwords varNameS
in if hasSep
then RepVar varName listName
(parParse expr) (Just $ parParse sep)
: parParse rest
else RepVar varName listName
(parParse expr) (Nothing)
: parParse sepS
parParse s = Str (takeWhile (notIn "$#") s)
: parParse (dropWhile (notIn "$#") s)
notIn :: (Eq a) => [a] -> a -> Bool
notIn s x =not $ elem x s
takeRep :: String -> String -> (Bool,String,String)
takeRep e x = takeRep' x []
where
takeRep' :: String -> String -> (Bool,String,String)
takeRep' [] _ = error $ e
takeRep' ('|':'|':s) acc = takeRep' s ('|':acc)
takeRep' ('|':s) acc = (True,reverse acc,s)
takeRep' ('#':s) acc = (False,reverse acc,s)
takeRep' (r:s) acc = takeRep' s (r:acc)
escapingBreak :: (Eq t) => t -> [Char] -> [t] -> ([t], [t])
escapingBreak s e st = eBreak [] st
where
eBreak acc (a:b:c) | a==s&&b==s = eBreak (s:acc) c
| a==s = (reverse acc,b:c)
| otherwise = eBreak (a:acc) (b:c)
eBreak acc (a:c) | a==s = (reverse acc,c)
| otherwise = eBreak (a:acc) c
eBreak _ [] = error e
norming :: String -> String
norming = intercalate "\n" . norming' . lines
norming':: [String] -> [String]
norming' [] = []
norming' (l:lst) = l:map (drop (n)) lst
where
n = minimumD 0 $ map (length . takeWhile (==' ') )
$ (filter (not.isEmpty) lst)
isEmpty = all (isSpace)
minimumD :: (Ord a) => a -> [a] -> a
minimumD d [] = d
minimumD _ x = minimum x