module Generate.JavaScript.Ports (incoming, outgoing) where
import Generate.JavaScript.Helpers
import qualified Generate.JavaScript.Variable as V
import AST.Type as T
import qualified AST.Variable as Var
import Language.ECMAScript3.Syntax
data JSType = JSNumber | JSBoolean | JSString | JSArray | JSObject [String]
deriving Show
check :: Expression () -> JSType -> Expression () -> Expression ()
check x jsType continue =
CondExpr () (jsFold OpLOr checks x) continue throw
where
jsFold op checks value = foldl1 (InfixExpr () op) (map ($value) checks)
throw = obj ["_E","raise"] <| InfixExpr () OpAdd msg x
msg = string ("invalid input, expecting " ++ show jsType ++ " but got ")
checks =
case jsType of
JSNumber -> [typeof "number"]
JSBoolean -> [typeof "boolean"]
JSString -> [typeof "string", instanceof "String"]
JSArray -> [(obj ["_U","isJSArray"] <|)]
JSObject fields -> [jsFold OpLAnd (typeof "object" : map member fields)]
incoming :: CanonicalType -> Expression ()
incoming tipe =
case tipe of
Aliased _ t -> incoming t
App (Type v) [t]
| Var.isSignal v -> V.value "Native.Ports" "incomingSignal" <| incoming t
_ -> ["v"] ==> inc tipe (ref "v")
inc :: CanonicalType -> Expression () -> Expression ()
inc tipe x =
case tipe of
Lambda _ _ -> error "functions should not be allowed through input ports"
Var _ -> error "type variables should not be allowed through input ports"
Aliased _ t ->
inc t x
Type (Var.Canonical Var.BuiltIn name)
| name == "Int" -> from JSNumber
| name == "Float" -> from JSNumber
| name == "Bool" -> from JSBoolean
| name == "String" -> from JSString
where
from checks = check x checks x
Type name
| Var.isJson name -> V.value "Native.Json" "fromJS" <| x
| Var.isTuple name -> incomingTuple [] x
| otherwise -> error "bad type got to incoming port generation code"
App f args ->
case f : args of
Type name : [t]
| Var.isMaybe name ->
CondExpr ()
(equal x (NullLit ()))
(V.value "Maybe" "Nothing")
(V.value "Maybe" "Just" <| inc t x)
| Var.isList name ->
check x JSArray (obj ["_L","fromArray"] <| array)
| Var.isArray name ->
check x JSArray (obj ["_A","fromJSArray"] <| array)
where
array = DotRef () x (var "map") <| incoming t
Type name : ts
| Var.isTuple name -> incomingTuple ts x
_ -> error "bad ADT got to incoming port generation code"
Record _ (Just _) ->
error "bad record got to incoming port generation code"
Record fields Nothing ->
check x (JSObject (map fst fields)) object
where
object = ObjectLit () $ (prop "_", ObjectLit () []) : keys
keys = map convert fields
convert (f,t) = (prop f, inc t (DotRef () x (var f)))
incomingTuple :: [CanonicalType] -> Expression () -> Expression ()
incomingTuple types x =
check x JSArray (ObjectLit () fields)
where
fields = (prop "ctor", ctor) : zipWith convert [0..] types
ctor = string ("_Tuple" ++ show (length types))
convert n t =
( prop ('_':show n)
, inc t (BracketRef () x (IntLit () n))
)
outgoing :: CanonicalType -> Expression ()
outgoing tipe =
case tipe of
Aliased _ t -> outgoing t
App (Type v) [t]
| Var.isSignal v -> V.value "Native.Ports" "outgoingSignal" <| outgoing t
_ -> ["v"] ==> out tipe (ref "v")
out :: CanonicalType -> Expression () -> Expression ()
out tipe x =
case tipe of
Aliased _ t -> out t x
Lambda _ _
| numArgs > 1 && numArgs < 10 ->
func (ref ('A':show numArgs) `call` (x:values))
| otherwise -> func (foldl (<|) x values)
where
ts = T.collectLambdas tipe
numArgs = length ts 1
args = map (\n -> '_' : show n) [0..]
values = zipWith inc (init ts) (map ref args)
func body =
function (take numArgs args)
[ VarDeclStmt () [VarDecl () (var "_r") (Just body)]
, ret (out (last ts) (ref "_r"))
]
Var _ -> error "type variables should not be allowed through input ports"
Type (Var.Canonical Var.BuiltIn name)
| name `elem` ["Int","Float","Bool","String"] -> x
Type name
| Var.isJson name -> V.value "Native.Json" "toJS" <| x
| Var.isTuple name -> ArrayLit () []
| otherwise -> error "bad type got to outgoing port generation code"
App f args ->
case f : args of
Type name : [t]
| Var.isMaybe name ->
CondExpr ()
(equal (DotRef () x (var "ctor")) (string "Nothing"))
(NullLit ())
(out t (DotRef () x (var "_0")))
| Var.isArray name ->
DotRef () (obj ["_A","toJSArray"] <| x) (var "map") <| outgoing t
| Var.isList name ->
DotRef () (obj ["_L","toArray"] <| x) (var "map") <| outgoing t
Type name : ts
| Var.isTuple name ->
let convert n t = out t $ DotRef () x $ var ('_':show n)
in ArrayLit () $ zipWith convert [0..] ts
_ -> error "bad ADT got to outgoing port generation code"
Record _ (Just _) ->
error "bad record got to outgoing port generation code"
Record fields Nothing ->
ObjectLit () keys
where
keys = map convert fields
convert (f,t) = (PropId () (var f), out t (DotRef () x (var f)))