WebSharper provides several ways to customize the way functions and values are compiled to JavaScript:
- Directly providing JavaScript code;
- Customizing the compiled name of the value;
- Classes that directly represent JavaScipt build-in types;
- Access JavaScipt properties dynamically;
- Transforming the code during compilation, a concept known as metaprogramming.
There are two ways of directly inserting JavaScipt code into a WebSharper project.
The Direct
attribute takes a JavaScript expression as a string parameter.
This will get parsed by the compiler and any syntax errors will be reported in a
compile error message.
This parsed expression will be used as a function body, return
is automatically
added for the last value.
If you don't want the use function from .NET, the WebSharper.JavaScript.Interop.X<T>()
method
throws an exception of type WebSharper.JavaScript.ClientSideOnly
with the message
"This function is intended for client-side use only.".
You can use placeholders for the function or method arguments.
For named parameters, the name with a $
prepended is recognised.
For example:
[Direct("$x + $y")]
public static int Add(int x, int y) => Interop.X<int>();
Also you can access parameters by index. In let-bound functions in modules and static methods of classes, the parameters are indexed from 0.
[Direct("$0 + $1")]
public static int Add(int x, int y) => Interop.X<int>();
In instance methods, $0
translates to the self indentifier, and method parameters
are indexed from 1.
(You can also use $this
for the self identifier, but this recommended against, as
a parameter named this
can override it, and it does not work for extension members
which are actually static methods in translated form.)
[Direct("Math.sqrt($0.x * $0.x + $0.y * $0.y)")]
public float this.GetLength() => Interop.X<float>();
The Inline
attribute takes a JavaScript expression as a string parameter.
(It can also be used together with the JavaScript
attribute to inline a function
translated from F#.)
This will be parsed, and inlined to all call sites of this function.
Only a subset of JavaScript operators and keywords can be used which can be translated
to the "core" AST used internally by WebSharper to optimize output.
Parameter placeholders work exactly as with Direct
.
[Inline("$x + $y")]
public static int Add(int x, int y) => Interop.X<int>();
The WebSharper.JavaScript.Pervasives.JS.Inline
function parses its first parameter at compile-time as JS code and includes
that in the result. It can contain holes, named $0
, $1
, ... and variable arguments will
be passed to the inline. Examples:
using static JSI = WebSharper.JavaScipt.Pervasives.JS
//...
var zeroDate = JSI.Inline("new Date()");
var date = JSI.Inline("new Date($0)", 1472226125177);
The Constant
attribute takes a literal value as parameter.
It can annotate a property or a union case which will be translated to the literal provided.
The Name
attribute takes a string parameter and allows specifying
the name of a function or class in the translation.
For example:
[Name("add")]
public static int OriginalNameForAdd(int x, int y) => x + y;
If you set a fixed translated name with the Name
attribute on an abstract member of a class
or interface, all inheriting and overriding members will have that exact translated name.
If a class is overriding or implementing two abstract members that has the same fixed name,
it will result in a compile-time error, and you have to change one of the fixed names to resolve it.
Automatically, interface members have a unique long translated name generated that contains the full type name.
This guarantees no conflicts without using the Name
attribute.
If you want to shorten it for readability of the JS output and making it smaller,
you can use the Name
attribute on the interface type to specify a short name for the interface.
It is recommended that it is unique across your solution.
You can also use [Name("")]
on the interface type to make all interface methods have the same translated name
as their original .NET name (if not specified otherwise by a Name
on the member).
There are classes in the WebSharper.JavaScript
namespace that are direct representations of ECMA standard
library JavaScipt types.
The WebSharper.JavaScript
namespace declares a .ToJS()
extension method on all .NET types to safely
convert them to their JavaScipt representation.
For JavaScipt functions, the Function
class is an untyped representation, but if you know the signature of
the JavaScipt function, there are more strongly typed alternatives:
-
For any function, that do not care about the
this
argument, you can use delegates. -
For functions that work with the
this
argument, useThisAction
andThisFunc
classes. They have a constructor that takes a delegate, for which the first argument will have thethis
value.var logger = new ThisAction<object>(x => WebSharper.JavaScript.Console.Log(x));
-
For functions taking variadic arguments, use
ParamsAction
andParamsFunc
classes. -
Finally for functions using the
this
value and have variadic arguments, useThisParamsAction
andThisParamsFunc
The WebSharper.JavaScript
namespace declares a .GetJS
extension method, that can be used to get JavaScipt properties dynamically.
Example: x.GetJS<int>("size", "width")
is translated to x.size.width
and usable as an int
value.
You can use x.GetJS<T>()
to just use the value of x
exposed as another .NET type T
.
You can use the dynamic
type to access JavaScipt properties and functions without any extra helpers:
dynamic d = names;
var name = d.getItems()[3].name; // translates directly to: d.getItems()[3].name
Also, operators on dynamic values are translated directly if there is a JavaScipt equivalent.
Macro and generator documentation will be available for stable release of WebSharper 4.