module CompileToJS (showErr, jsModule) where import Control.Arrow (first,second) import Control.Monad (liftM,(<=<),join,ap) import Data.Char (isAlpha,isDigit) import Data.List (intercalate,sortBy,inits,foldl') import qualified Data.Map as Map import Data.Either (partitionEithers) import qualified Text.Pandoc as Pan import Ast import Context import Rename (derename) import Cases import Guid import LetBoundVars import Parse.Library (isOp) import Rename (deprime) import Types.Types ( Type(RecordT) ) showErr :: String -> String showErr err = "Elm.Graphics.text(Elm.Text.monospace(" ++ msg ++ "))" where msg = show . concatMap (++"
") . lines $ err indent = concatMap f where f '\n' = "\n " f c = [c] internalImports = [ ("N" , "Elm.Native"), ("_N", "N.Utils(elm)"), ("_L", "N.List(elm)"), ("_E", "N.Error(elm)"), ("_str", "N.JavaScript(elm).toString") ] parens s = "(" ++ s ++ ")" brackets s = "{" ++ s ++ "}" jsList ss = "["++ intercalate "," ss ++"]" jsFunc args body = "function(" ++ args ++ "){" ++ indent body ++ "}" assign x e = "\nvar " ++ x ++ " = " ++ e ++ ";" ret e = "\nreturn "++ e ++";" iff a b c = a ++ "?" ++ b ++ ":" ++ c quoted s = "'" ++ concatMap f s ++ "'" where f '\n' = "\\n" f '\'' = "\\'" f '\t' = "\\t" f '\"' = "\\\"" f '\\' = "\\\\" f c = [c] globalAssign n e = "\n" ++ assign' n e ++ ";" assign' n e = n ++ " = " ++ e jsModule (Module names exports imports stmts) = setup ("Elm":modNames) ++ globalAssign ("Elm." ++ modName) (jsFunc "elm" program) where modNames = if null names then ["Main"] else names modName = intercalate "." modNames includes = concatMap jsImport imports body = stmtsToJS stmts defs = assign "$op" "{}" program = "\nvar " ++ usefulFuncs ++ ";" ++ defs ++ includes ++ body ++ setup ("elm":"Native":modNames) ++ assign "_" ("elm.Native." ++ modName ++ " || {}") ++ getExports exports stmts ++ setup ("elm":modNames) ++ ret (assign' ("elm." ++ modName) "_") ++ "\n" setup modNames = concatMap (\n -> globalAssign n $ n ++ " || {}") . map (intercalate ".") . tail . inits $ init modNames usefulFuncs = intercalate ", " (map (uncurry assign') internalImports) getExports names stmts = "\n"++ intercalate ";\n" (op : map fnPair fns) where exNames n = either derename id n `elem` names exports | null names = concatMap get stmts | otherwise = filter exNames (concatMap get stmts) (fns,ops) = partitionEithers exports opPair op = "'" ++ op ++ "' : $op['" ++ op ++ "']" fnPair fn = let fn' = derename fn in "_." ++ fn' ++ " = " ++ fn op = ("_.$op = "++) . brackets . intercalate ", " $ map opPair ops get' (FnDef x _ _) = Left x get' (OpDef op _ _ _) = Right op get s = case s of Definition d -> [ get' d ] Datatype _ _ tcs -> map (Left . fst) tcs ImportEvent _ _ x _ -> [ Left x ] ExportEvent _ _ _ -> [] TypeAlias _ _ _ -> [] TypeAnnotation _ _ -> [] jsImport (modul, how) = case how of As name -> assign name ("Elm." ++ modul ++ parens "elm") Importing vs -> "\nvar $ = Elm." ++ modul ++ parens "elm" ++ ";" ++ if null vs then "\nfor (var k in $) {eval('var '+k+'=$[\"'+k+'\"]')}" else "\nvar " ++ intercalate ", " (map def vs) ++ ";" where imprt v = assign' v ("$." ++ v) def (o:p) = imprt (if isOp o then "$op['" ++ o:p ++ "']" else deprime (o:p)) stmtsToJS :: [Statement] -> String stmtsToJS stmts = run $ do program <- mapM toJS (sortBy cmpStmt stmts) return (vars ++ concat program) where vars = "\nvar " ++ intercalate ", " ("e":"case0":letBoundVars stmts) ++ ";" cmpStmt s1 s2 = compare (valueOf s1) (valueOf s2) valueOf s = case s of Datatype _ _ _ -> 1 ImportEvent _ _ _ _ -> 2 Definition (FnDef f [] _) -> if derename f == "main" then 5 else 4 Definition _ -> 3 ExportEvent _ _ _ -> 6 TypeAlias _ _ _ -> 0 TypeAnnotation _ _ -> 0 class ToJS a where toJS :: a -> GuidCounter String instance ToJS Def where toJS (FnDef x [] e) = assign' x `liftM` toJS' e toJS (FnDef f as e) = (assign' f . wrapper . func) `liftM` toJS' e where func body = jsFunc (intercalate ", " as) (ret body) wrapper e | length as == 1 = e | otherwise = 'F' : show (length as) ++ parens e toJS (OpDef op a1 a2 e) = do body <- toJS' e let func = "F2" ++ parens (jsFunc (a1 ++ ", " ++ a2) (ret body)) return (globalAssign ("$op['" ++ op ++ "']") func) instance ToJS Statement where toJS stmt = case stmt of Definition d -> (\asgn -> "\n" ++ asgn ++ ";") `liftM` toJS d Datatype _ _ tcs -> concat `liftM` mapM (toJS . toDef) tcs where toDef (name,args) = let vars = map (('a':) . show) [1..length args] in Definition . FnDef name vars . noContext $ Data (derename name) (map (noContext . Var) vars) ImportEvent js base elm _ -> do v <- toJS' base return $ concat [ "\nvar " ++ elm ++ "=Elm.Signal.constant(" ++ v ++ ");" , "\ndocument.addEventListener('", js , "_' + elm.id, function(e) { elm.notify(", elm , ".id, e.value); });" ] ExportEvent js elm _ -> return $ concat [ "\nlift(function(v) { " , "var e = document.createEvent('Event');" , "e.initEvent('", js, "_' + elm.id, true, true);" , "e.value = v;" , "document.dispatchEvent(e); return v; })(", elm, ");" ] TypeAnnotation _ _ -> return "" TypeAlias n _ t -> return "" toJS' :: CExpr -> GuidCounter String toJS' (C txt span expr) = case expr of MultiIf ps -> multiIfToJS span ps Case e cases -> caseToJS span e cases _ -> toJS expr remove x e = "_N.remove('" ++ x ++ "', " ++ e ++ ")" addField x v e = "_N.insert('" ++ x ++ "', " ++ v ++ ", " ++ e ++ ")" setField fs e = "_N.replace(" ++ jsList (map f fs) ++ ", " ++ e ++ ")" where f (x,v) = "['" ++ x ++ "'," ++ v ++ "]" access x e = parens e ++ "." ++ x makeRecord kvs = record `liftM` collect kvs where combine r (k,v) = Map.insertWith (++) k v r collect = liftM (foldl' combine Map.empty) . mapM prep prep (k, as, e@(C t s _)) = do v <- toJS' (foldr (\x e -> C t s $ Lambda x e) e as) return (k,[v]) fields fs = brackets ("\n "++intercalate ",\n " (map (\(k,v) -> k++":"++v) fs)) hidden = fields . map (second jsList) . filter (not . null . snd) . Map.toList . Map.map tail record kvs = fields . (("_", hidden kvs) :) . Map.toList . Map.map head $ kvs instance ToJS Expr where toJS expr = case expr of IntNum n -> return $ show n FloatNum n -> return $ show n Var x -> return $ x Chr c -> return $ quoted [c] Str s -> return $ "_str" ++ parens (quoted s) Boolean b -> return $ if b then "true" else "false" Range lo hi -> jsRange `liftM` toJS' lo `ap` toJS' hi Access e x -> access x `liftM` toJS' e Remove e x -> remove x `liftM` toJS' e Insert e x v -> addField x `liftM` toJS' v `ap` toJS' e Modify e fs -> do fs' <- (mapM (\(x,v) -> (,) x `liftM` toJS' v) fs) setField fs' `liftM` toJS' e Record fs -> makeRecord fs Binop op e1 e2 -> binop op `liftM` toJS' e1 `ap` toJS' e2 If eb et ef -> parens `liftM` (iff `liftM` toJS' eb `ap` toJS' et `ap` toJS' ef) Lambda v e -> liftM (jsFunc v . ret) (toJS' e) App e1 e2 -> jsApp e1 e2 Let defs e -> jsLet defs e Data name es -> do fs <- mapM toJS' es let fields = zipWith (\n e -> "_" ++ show n ++ ":" ++ e) [0..] fs return (brackets ("ctor:" ++ show name ++ concatMap (", "++) fields)) Markdown doc -> return $ "text('" ++ pad ++ md ++ pad ++ "')" where pad = "
 
" md = formatMarkdown $ Pan.writeHtmlString Pan.def doc jsApp e1 e2 = do f <- toJS' func as <- mapM toJS' args return $ case as of [a] -> f ++ parens a _ -> "A" ++ show (length as) ++ parens (intercalate ", " (f:as)) where (func, args) = go [e2] e1 go args e = case e of (C _ _ (App e1 e2)) -> go (e2 : args) e1 _ -> (e, args) formatMarkdown = concatMap f where f '\'' = "\\'" f '\n' = "\\n" f '"' = "\"" f c = [c] multiIfToJS span ps = case last ps of (C _ _ (Var "otherwise"), e) -> toJS' e >>= \b -> format b (init ps) _ -> format ("_E.If" ++ parens (quoted (show span))) ps where format base ps = foldr (\c e -> parens $ c ++ " : " ++ e) base `liftM` mapM f ps f (b,e) = do b' <- toJS' b e' <- toJS' e return (b' ++ " ? " ++ e') jsLet defs e' = do ds <- jsDefs defs e <- toJS' e' return $ parens (intercalate ", " ds ++ ", " ++ e) where jsDefs defs = mapM toJS (sortBy f defs) f a b = compare (valueOf a) (valueOf b) valueOf (FnDef _ args _) = min 1 (length args) valueOf (OpDef _ _ _ _) = 1 caseToJS span e ps = do match <- caseToMatch ps e' <- toJS' e (match',stmt) <- case (match,e) of (Match name _ _, C _ _ (Var x)) -> return (matchSubst [(name,x)] match, Nothing) (Match name _ _, _) -> return (match, Just (name ++ " = " ++ e')) _ -> return (match, Nothing) matches <- matchToJS span match' return $ case stmt of Nothing -> matches Just exp -> parens (exp ++ ", " ++ matches) matchToJS span match = let sqnc e1 e2 = parens ("e=" ++ e1 ++ ":null,e!==null?e:" ++ e2) in case match of Match name clauses def -> do cases <- intercalate ":" `liftM` mapM (clauseToJS span name) clauses finally <- matchToJS span def return (sqnc cases finally) Fail -> return ("_E.Case" ++ parens (quoted (show span))) Break -> return "null" Other e -> toJS' e Seq ms -> foldr1 sqnc `liftM` mapM (matchToJS span) (dropEnd [] ms) where dropEnd acc [] = acc dropEnd acc (m:ms) = case m of Other _ -> acc ++ [m] _ -> dropEnd (acc ++ [m]) ms clauseToJS span var (Clause name vars e) = do let vars' = map (\n -> var ++ "._" ++ show n) [0..] s <- matchToJS span $ matchSubst (zip vars vars') e return $ concat [ var, ".ctor===", quoted name, "?", s ] jsNil = "_L.Nil" jsCons e1 e2 = "_L.Cons(" ++ e1 ++ "," ++ e2 ++ ")" jsRange e1 e2 = "_L.range" ++ parens (e1 ++ "," ++ e2) jsCompare e1 e2 op = parens ("_N.compare(" ++ e1 ++ "," ++ e2 ++ ").ctor" ++ op) binop (o:p) e1 e2 | isAlpha o || '_' == o = (o:p) ++ parens e1 ++ parens e2 | otherwise = let ops = ["+","-","*","/","&&","||"] in case o:p of "::" -> jsCons e1 e2 "++" -> "_L.append" ++ parens (e1 ++ "," ++ e2) "$" -> e1 ++ parens e2 "." -> jsFunc "x" . ret $ e1 ++ parens (e2 ++ parens "x") "^" -> "Math.pow(" ++ e1 ++ "," ++ e2 ++ ")" "==" -> "_N.eq(" ++ e1 ++ "," ++ e2 ++ ")" "/=" -> "!_N.eq(" ++ e1 ++ "," ++ e2 ++ ")" "<" -> jsCompare e1 e2 "==='LT'" ">" -> jsCompare e1 e2 "==='GT'" "<=" -> jsCompare e1 e2 "!=='GT'" ">=" -> jsCompare e1 e2 "!=='LT'" "<~" -> "A2(lift," ++ e1 ++ "," ++ e2 ++ ")" "~" -> "A3(lift2,F2(function(f,x){return f(x)}),"++e1++","++e2++")" _ | elem (o:p) ops -> parens (e1 ++ (o:p) ++ e2) | otherwise -> concat [ "$op['", o:p, "']" , parens e1, parens e2 ]