Zephyr Expressions And Operators
Anything inside
{curly braces}
is evaluated as an expression. Expressions are constructed using a Javascript-like format. Be sure not to include a space immediately following your opening bracket. If you do, your code won't be recognized as Zephyr.
Strings
Use either single quotes or double quotes to denote strings, as you prefer. The backslash is used to escape quotes.
{"apples"} is the same as {'apples'} {'What\'s up?'}
Note: Double quotes are not supported in Email Composer. Use single quotes in Email Composer so your Zephyr renders properly.
Numbers
Integers and floating-point values are both supported.
{3 + 3.0} = 6
Booleans
true
and
false
are reserved words used for boolean math.
Arrays
You can define arrays inline, or pass them in as variables. They are defined using JSON syntax and indexed starting at 0.
{set('myvar', [1,2,"three"])} {myvar[2]} equals three
Objects
Objects are also defined using JSON syntax. You can access objects using dot notation or array index notation.
{set('myvar',{"attrib1":"value1","attrib2":"value2"})} {myvar.attrib2} equals value2 {myvar['attrib2']} equals value2
Comparisons
Standard comparison operators are supported:
==
,
!=
,
>=
,
>
{if 2 == 2.0}this will display{/if} {if 2 <= 2}so will this{/if} {if 2 < 2}but this won't{/if}
And / Or
&&
functions as a boolean
and
, while
||
functions as a boolean
or
.
{if true && false} this will not display {/if} {if true || false} but this will {/if}
Not
You can use the ! as a not-operator. This is especially handy when checking whether a variable has been set.
{if !true}This won't display{/if} {if !source} Hey {name}, we don't know where you came from! {/if}
Math
Basic math is supported:
+
,
-
,
*
,
/
,
Useful, among other situations, when applying multipliers to Horizon interests.
{if horizon_interest('apples')*2 >= horizon_interest('oranges')}
You're at least half as interested in apples as you are in oranges!
{/if}
Modulo
To find the remainder of division of one number by another, use the
modulo operator %
.
{if (i % 2) != 0}
"i am an odd number!"{/if}
Note: Parentheses necessary around the modulo operator when using a comparison.
This can be useful for styling content. For instance, creating a two-column table so that any even-numbered item is aligned left, and any odd-numbered item is aligned right:
{foreach content as i,c}
{if (i % 2) == 0}
<tr>
<td align="left"><a href="{c.url}">{c.title}</a></td>
{else}
<td align="right"><a href="{c.url}">{c.title}</a></td>
</tr>
{/if}
{/foreach}
Parentheses
You can use parentheses to specify the order-of-operations.
{5*3+1} = 16
{5*(3+1)} = 20
Concatenation
The
+
symbol adds two numbers together, but has other purposes when used for other types:
- If you use
+
on strings string1 + string2
, it concatenates them (appends string2
to the end of string1
) - If you use
+
on lists list1 + list2
, it appends the list2
to the end of list1
- If you use
+
on objects object1 + object2
, it merges them (returns object1
with all of object2
's keys copied onto it)
{'Zehpyr' + ' is fun'} = Zephyr is fun
{[1, 2, 3] + [4, 5, 6]} = [1, 2, 3, 4, 5, 6]
{{'a':1,'b':2} + {'b':3,'c':4}} = {'a':1,'b':3,'c':4}
Lambdas
Zephyr supports lambdas (anonymous functions). These are typically used as parameters to functions such as
map,
filter, or
sort.
The lambda syntax is similar to Python:
lambda x: x*2
lambda x,y: (x != y)
For more specific examples on how to use lambdas, see the documentation on the functions above.
List Comprehensions
Zephyr supports Python-style list comprehensions. This is a concise, elegant way to apply a map or filter to a list.
The format is:
[map for varname in list]
or
[map for varname in list if condition]
The
map
expression is applied to each member of the
list
that matches
condition
. This is probably easier illustrated with some examples:
applying a map function: {[x*2 for x in [1,2,3,4,5]]} = [2,4,6,8,10]
applying a filter: {[x for x in [1, 2, 3, 4, 5] if x > 2]} = [3, 4, 5]
applying both: {[x*2 for x in [1, 2, 3, 4, 5] if x > 2]} = [6, 8, 10]
This is especially useful for filtering content pulled from a data feed. For example, to filter for the stories tagged
football
in a list called
content
pulled from your data feed, you could do:
{content = [c for c in content if contains(c.tags, 'football')]}