The WEM-Specific script language for WEM Widgets
Besides the WEMScript for Custom HTML, WEM also has WEMScript with specific features to use when creating Custom WEM Widgets.
WEMScript is a specific script created by WEM to make it possible to use specific WEM Elements (properties, fields, events, functions) available to the Script Editor in the Widgets. This is typically the low-code part of WEM.
General usage:
The elseif
and else
parts are optional. The expressions of the if
and elseif
statements should return a boolean (true/false).
General usage:
The expression after while
should return a boolean (true/false) - all statements between while
and end
will be executed as long as the while-expression is true.
There are two ways to declare a variable:
The type of variable @b
is derived from the expression. The following types are supported by wem:
actionflowchart
, boolean
, buttonexit
, concept
, conceptset
, datetime
, duration
, file
, hyperlink
, list
, multiselect
, navigationitem
, number
, richtext
and text
.
Array variables are available in WEMScript. Array variables are declared as follows:
The type array<T>
is generic, but T
can itself not be an array type. The following declaration is not supported:
An array variable is never null
(or unknown
). As soon as the variable is declared, an empty array is instantiated. The following methods can be used to add or remove values from an array:
The new values will be appended to the end of the array. If you only want to add a value if it is not already in the array, then you can use the unique
keyword:
You can use the remove
statement to remove values from an array:
Use the count()
function to get the length of an array. Other aggregation functions are also available, such as concat()
, max()
, min()
, sum()
and average()
.
You can initialize an array by using the list()
function.
Use parentheses to get the value at a specific index. The index is one-based (meaning the first element has index value of 1):
If the index is out of bounds, then the result will be unknown
. You should not get an error in the script when this should happen.
You can use the in
operator to check if the array contains a certain value:
The register
statement can only be used in render scripts and is used to indicate to the runtime that a form <input>
field will be rendered that is bound to a Data Field item from the project Data Model. This enables two-way binding. General usage:
The most basic use is register input <data-field>
. This can followed by some (optional) annotations:
required
– annotates that the user is required to provide a value.
notrim
– by default all the user input is trimmed. Use this annotation to disable trimming. The = <expression>
part is optional.
precision
– this annotation is only valid for number
and duration
data-fields. The precision should be a numeric value between -1
and 12
(including). If the precision is -1
, or if the precision is omitted, then precision is limited by the 64bit float (double) that stores the value.
include time
– this annotation is only valid for datetime
data-fields. It states that it expects that the user specifies both the date and the time.
unit
– this annotation is only valid for duration
data-fields. Possible values are day
, hour
, minute
, and second
.
invariant culture
– this annotation is only valid for number
, duration
and datetime
data-fields. It states that it always expects the user input to be in an international (US English) format. For dates, it means that the input should be a valid ISO8601 format. For numeric values, it means that the .
should be used as the decimal point (and comma as thousand-separator).
base64
- this annotation is used for File Input Properties.
The annotations required
, notrim
, include time
and invariant culture
can be used with or without an expression. If it is used without an expression, then <annotation> = true
. is implied.
Examples:
The order in which the annotations are stated is not relevant.
Files that are part of the datamodel are stored in Base64 and may be prefixed with a filename and semicolon. For example if your Widget uses a canvas element you can create and store an image based on that canvas by using the Javascript toDataUrl function. That function returns a data URL containing Base64 data, the data you can use to store images. Note that WEM is only interested in the data, with an optional filename prefix.
If your data URL is:
"…ADAMBAAIRAxEAPwD/AD/6AP/Z
"
you need to remove "data:image/jpeg;base64,".
A valid value with a filename prefix would be:
"image.jpg;/9j/4AAQSkZJRgABAQ…ADAMBAAIRAxEAPwD/AD/6AP/Z
".
Print statements can only be used in render scripts. The general usage is:
Renders the result of the given expression. If html
is provided, then the result will be html encoded; if attr
is provided, then the result will be html attribute encoded.
General usage:
The result of the expression should be "iterable". That is, it should either be a (filtered) list or a concept set. This means the following examples are valid:
When looping through a concept set, you can use the keyword concept
to reference the current concept within the loop.
The follow
statement can only be used in the event-handler script of a Widget.
The general usage is:
This statement will terminate the script and follow the given (button) exit.
The execute
statement can only be used in the event-handler script of a Widget.
The general usage is:
This statement will execute the given (action) flowchart. After the flowchart is executed, it will continue the execution of the event-handler script.
The navigate to
statement can only be used in the event-handler script of a Widget.
The general usage is:
This statement will terminate the script and redirect the user to the given location. The given expression should return a navigationitem
, hyperlink
or string
. If the return type is a string
, it will be interpreted as a URL (hyperlink).
The scriptmodule
statement can only be used in the render-script of a Widget.
The general usage is:
Use the scriptmodule
statement to add JavaScript code that your widget needs. This JavaScript code will be added to the page only once: the first time that the widget is used on the page. This makes it possible to add a Widget more than once on a Template while the code being used is only added once to the output (optimizing the HTML response).
The scriptreference
statement can only be used in the render-script of a Widget.
The general usage is:
Use the scriptreference
statement to add a JavaScript reference that your widget needs. This JavaScript reference will be added to the page only once: the first time that the widget is used on the page.
The startupscript
can only be used in the render-script of a Widget. The startupscript
block will be invoked after the whole HTML result has been rendered. You can use this code block to initialize instances in JavaScript.
The general usage is:
The submitscript
can only be used in the render script of a Widget. The submitscript
block will be invoked before the new request is sent, for example immediately after a button click. You can use this code block to alter the input data before it will be send with the request.
The general usage is:
The submitscript
can only be used in the render-script of a Widget. The unloadscript
block will be invoked after the request has responded but before the new HTML has been drawn. You can use this code block to cleanup JavaScript instances before drawing the new HTML.
The general usage is:
The once
statement can only be used in the render script of a Widget.
The general usage is:
The once
statement can be used to make sure that an html fragment is only added to the page (html-output) once. You should not use this keyword to add scripts. Use the scriptmodule
, scriptreference
or startupscript
statements to add scripts in the correct way.
This keyword is only available in render scripts and will return the WEM output Id for the current custom control or for the given data-field. Usage:
An example that uses the id
keyword in combination with the register input field to create a custom control for text input. It assumes that the property @DataField
exists and is bound to a text datafield.
When the custom control properties are resolved, it will become something like this:
The range of
keyword returns the range of a concept (or concept set) data-field (single-select or multi-select). Usage:
This is an expression that returns a concept set. You can use this construct as part in a bigger expression:
The range of
keyword is very useful when making a custom control bound to a concept (set) data-field. For the following example, assume that @ConceptSet
is a property that is bound to a concept set data-field:
This keyword is only available in render-scripts. The invalidvalue
keyword returns the invalid value that is stored in the given data-field. It returns null if the value of the given data-field is valid. Usage:
The use case for this keyword is to get the invalid value that the user provided. For the following example, assume that @number
is a property that is bound to a numeric data-field:
The viewstate
is accessible and modifiable in the Runtime. The WEM Runtime.js
has an API that widgets can use to read and modify the viewstate
. This greatly improves the possibilities of a widget.
You can define Viewstate fields in the Viewstate tab. The type of a field is restricted to one of the following: