Operators
Operator Precedence
Below is the operator precedence table, highest to lowest:
.
[]
! ~ + -
is defined
** * / %
+ -
... ..
<= >= < >
in
== is != is not isnt
is a
&& and || or
?:
= := ?= += -= *= /= %=
not
if unless
Unary Operators
The following unary operators are available, !
, not
, -
, +
, and ~
.
!0
// => true
!!0
// => false
!1
// => false
!!5px
// => true
-5px
// => -5px
--5px
// => 5px
not true
// => false
not not true
// => true
The logical not
operator has low precedence, therefore the following example could be replaced with
a = 0
b = 1
!a and !b
// => false
// parsed as: (!a) and (!b)
With:
not a or b
// => false
// parsed as: not (a or b)
Binary Operators
Subscript []
The subscript operator allows us to grab a value inside an expression via index (zero-based). Negative index values starts with the last element in the expression.
list = 1 2 3
list[0]
// => 1
list[-1]
// => 3
Parenthesized expressions may act as tuples (e.g. (15px 5px)
, (1 2 3)
).
Below is an example that uses tuples for error handling (and showcasing the versatility of this construct):
add(a, b)
if a is a 'unit' and b is a 'unit'
a + b
else
(error 'a and b must be units!')
body
padding add(1,'5')
// => padding: error "a and b must be units";
padding add(1,'5')[0]
// => padding: error;
padding add(1,'5')[0] == error
// => padding: true;
padding add(1,'5')[1]
// => padding: "a and b must be units";
Here’s a more complex example. Now we’re invoking the built-in error()
function with the return error message, whenever the ident (the first value) equals error
.
if (val = add(1,'5'))[0] == error
error(val[1])
Range .. …
Both the inclusive (..
) and exclusive (...
) range operators are provided, expanding to expressions:
1..5
// => 1 2 3 4 5
1...5
// => 1 2 3 4
5..1
// => 5 4 3 2 1
Additive: + -
Multiplicative and additive binary operators work as expected. Type conversion is applied within unit type classes, or default to the literal value. For example 5s - 2px
results in 3s
.
15px - 5px
// => 10px
5 - 2
// => 3
5in - 50mm
// => 3.031in
5s - 1000ms
// => 4s
20mm + 4in
// => 121.6mm
"foo " + "bar"
// => "foo bar"
"num " + 15
// => "num 15"
Multiplicative: / * %
2000ms + (1s * 2)
// => 4000ms
5s / 2
// => 2.5s
4 % 2
// => 0
When using /
within a property value, you must wrap with parens. Otherwise the /
is taken literally (to support CSS line-height
):
font: 14px/1.5;
But the following is evaluated as 14px
÷ 1.5
:
font: (14px/1.5);
This is only required for the /
operator.
Exponent: **
The Exponent operator:
2 ** 8
// => 256
Equality & Relational: == != >= <= > <
Equality operators can be used to equate units, colors, strings, and even identifiers. This is a powerful concept, as even arbitrary identifiers (such as as wahoo
) can be utilized as atoms. A function could return yes
or no
instead of true
or false
(although not advised).
5 == 5
// => true
10 > 5
// => true
#fff == #fff
// => true
true == false
// => false
wahoo == yay
// => false
wahoo == wahoo
// => true
"test" == "test"
// => true
true is true
// => true
'hey' is not 'bye'
// => true
'hey' isnt 'bye'
// => true
(foo bar) == (foo bar)
// => true
(1 2 3) == (1 2 3)
// => true
(1 2 3) == (1 1 3)
// => false
Only exact values match. For example, 0 == false
and null == false
are both false
.
Aliases:
== is
!= is not
!= isnt
Truthfulness
Nearly everything within Stylus resolves to true
, including units with a suffix. Even 0%
, 0px
, etc. will resolve to true
(because it’s common in Stylus for mixins or functions to accept units as valid).
However, 0
itself is false
in terms of arithmetic.
Expressions (or “lists”) with a length greater than 1 are considered truthy.
true
examples:
0%
0px
1px
-1
-1px
hey
'hey'
(0 0 0)
('' '')
false
examples:
0
null
false
''
Logical Operators: && || and or
Logical operators &&
and ||
are aliased and
/ or
which apply the same precedence.
5 && 3
// => 3
0 || 5
// => 5
0 && 5
// => 0
#fff is a 'rgba' and 15 is a 'unit'
// => true
Existence Operator: in
Checks for the existence of the left-hand operand within the right-hand expression.
Simple examples:
nums = 1 2 3
1 in nums
// => true
5 in nums
// => false
Some undefined identifiers:
words = foo bar baz
bar in words
// => true
HEY in words
// => false
Works with tuples too:
vals = (error 'one') (error 'two')
error in vals
// => false
(error 'one') in vals
// => true
(error 'two') in vals
// => true
(error 'something') in vals
// => false
Example usage in mixin:
pad(types = padding, n = 5px)
if padding in types
padding n
if margin in types
margin n
body
pad()
body
pad(margin)
body
pad(padding margin, 10px)
Yielding:
body {
padding: 5px;
}
body {
margin: 5px;
}
body {
padding: 10px;
margin: 10px;
}
Conditional Assignment: ?= :=
The conditional assignment operator ?=
(aliased as :=
) lets us define variables without clobbering old values (if present). This operator expands to an is defined
binary operation within a ternary.
For example, the following are equivalent:
color := white
color ?= white
color = color is defined ? color : white
When using plain =
, we simply reassign:
color = white
color = black
color
// => black
But when using ?=
, our second attempt fails (since the variable is already defined):
color = white
color ?= black
color
// => white
Instance Check: is a
Stylus provides a binary operator named is a
used to type check.
15 is a 'unit'
// => true
#fff is a 'rgba'
// => true
15 is a 'rgba'
// => false
Alternatively, we could use the type()
BIF:
type(#fff) == 'rgba'
// => true
Note: color
is the only special-case, evaluating to true
when the
left-hand operand is an RGBA
or HSLA
node.
Variable Definition: is defined
This pseudo binary operator does not accept a right-hand operator, and does not evaluate the left. This allows us to check if a variable has a value assigned to it.
foo is defined
// => false
foo = 15px
foo is defined
// => true
#fff is defined
// => 'invalid "is defined" check on non-variable #fff'
Alternatively, one can use the lookup(name)
built-in function to do this—or to perform dynamic lookups:
name = 'blue'
lookup('light-' + name)
// => null
light-blue = #80e2e9
lookup('light-' + name)
// => #80e2e9
This operator is essential, as an undefined identifier is still a truthy value. For example:
body
if ohnoes
padding 5px
Will yield the following CSS when undefined:
body {
padding: 5px;
}
However this will be safe:
body
if ohnoes is defined
padding 5px
Ternary
The ternary operator works as we would expect in most languages. It’s the only operator with three operands (the condition expression, the truth expression, and the false expression).
num = 15
num ? unit(num, 'px') : 20px
// => 15px
Casting
As a terse alternative to the unit()
built-in function, the syntax (expr) unit
may be used to force the suffix.
body
n = 5
foo: (n)em
foo: (n)%
foo: (n + 5)%
foo: (n * 5)px
foo: unit(n + 5, '%')
foo: unit(5 + 180 / 2, deg)
Color Operations
Operations on colors provide a terse, expressive way to alter components. For example, we can operate on each RGB:
#0e0 + #0e0
// => #0f0
Another example is adjust the lightness value by adding or subtracting a percentage. To lighten a color, add; to darken, subtract.
#888 + 50%
// => #c3c3c3
#888 - 50%
// => #444
Adjust the hue is also possible by adding or subtracting with degrees. For example, adding 50deg
to this red value results in a yellow:
#f00 + 50deg
// => #ffd500
Values clamp appropriately. For example, we can “spin” the hue 180 degrees, and if the current value is 320deg
, it will resolve to 140deg
.
We may also tweak several values at once (including the alpha) by using rgb()
, rgba()
, hsl()
, or hsla()
:
#f00 - rgba(100,0,0,0.5)
// => rgba(155,0,0,0.5)
Sprintf
The string sprintf-like operator %
can be used to generate a literal value, internally passing arguments through the s()
built-in:
'X::Microsoft::Crap(%s)' % #fc0
// => X::Microsoft::Crap(#fc0)
Multiple values should be parenthesized:
'-webkit-gradient(%s, %s, %s)' % (linear (0 0) (0 100%))
// => -webkit-gradient(linear, 0 0, 0 100%)