{-# LANGUAGE TemplateHaskell #-}
module Data.String.Interpolate (
  i
) where
import           Language.Haskell.TH.Quote (QuasiQuoter(..))
import           Language.Haskell.Meta.Parse (parseExp)
import           Data.String.Interpolate.Internal.Util
import           Data.String.Interpolate.Parse
import           Language.Haskell.TH
i :: QuasiQuoter
i = QuasiQuoter {
    quoteExp = toExp . parseNodes . decodeNewlines
  , quotePat = err "pattern"
  , quoteType = err "type"
  , quoteDec = err "declaration"
  }
  where
    err name = error ("Data.String.Interpolate.i: This QuasiQuoter can not be used as a " ++ name ++ "!")
    toExp :: [Node ()] -> Q Exp
    toExp input = do
      nodes <- mapM generateName input
      e <- go nodes
      return $ foldr lambda e [name | Abstraction name <- nodes]
      where
        lambda :: Name -> Exp -> Exp
        lambda name e = LamE [VarP name] e
        generateName :: Node () -> Q (Node Name)
        generateName (Abstraction ()) = Abstraction <$> newName "x"
        generateName (Literal s) = return (Literal s)
        generateName (Expression e) = return (Expression e)
        go :: [Node Name] -> Q Exp
        go nodes = case nodes of
          [] -> [|""|]
          (x:xs) -> eval x `appE` go xs
          where
            eval (Literal s) = [|showString s|]
            eval (Expression e) = interpolate (reifyExpression e)
            eval (Abstraction name) = interpolate $ (return $ VarE name)
            interpolate :: Q Exp -> Q Exp
            interpolate e = [|(showString . toString) $(e)|]
            reifyExpression :: String -> Q Exp
            reifyExpression s = case parseExp s of
              Left _ -> do
                fail "Parse error in expression!" :: Q Exp
              Right e -> return e
decodeNewlines :: String -> String
decodeNewlines = go
  where
    go xs = case xs of
      '\r' : '\n' : ys -> '\n' : go ys
      y : ys -> y : go ys
      [] -> []