mirror of
https://github.com/typst/typst
synced 2025-05-14 04:56:26 +08:00
466 lines
11 KiB
Typst
466 lines
11 KiB
Typst
// Test binary expressions.
|
|
|
|
--- ops-add-content ---
|
|
// Test adding content.
|
|
#([*Hello* ] + [world!])
|
|
|
|
--- ops-unary-basic ---
|
|
// Test math operators.
|
|
|
|
// Test plus and minus.
|
|
#for v in (1, 3.14, 12pt, 45deg, 90%, 13% + 10pt, 6.3fr) {
|
|
// Test plus.
|
|
test(+v, v)
|
|
|
|
// Test minus.
|
|
test(-v, -1 * v)
|
|
test(--v, v)
|
|
|
|
// Test combination.
|
|
test(-++ --v, -v)
|
|
}
|
|
|
|
#test(-(4 + 2), 6-12)
|
|
|
|
// Addition.
|
|
#test(2 + 4, 6)
|
|
#test("a" + "b", "ab")
|
|
#test("a" + if false { "b" }, "a")
|
|
#test("a" + if true { "b" }, "ab")
|
|
#test(13 * "a" + "bbbbbb", "aaaaaaaaaaaaabbbbbb")
|
|
#test((1, 2) + (3, 4), (1, 2, 3, 4))
|
|
#test((a: 1) + (b: 2, c: 3), (a: 1, b: 2, c: 3))
|
|
|
|
--- ops-add-too-large ---
|
|
// Error: 3-26 value is too large
|
|
#(9223372036854775807 + 1)
|
|
|
|
--- ops-binary-basic ---
|
|
// Subtraction.
|
|
#test(1-4, 3*-1)
|
|
#test(4cm - 2cm, 2cm)
|
|
#test(1e+2-1e-2, 99.99)
|
|
|
|
// Multiplication.
|
|
#test(2 * 4, 8)
|
|
|
|
// Division.
|
|
#test(12pt/.4, 30pt)
|
|
#test(7 / 2, 3.5)
|
|
|
|
// Combination.
|
|
#test(3-4 * 5 < -10, true)
|
|
#test({ let x; x = 1 + 4*5 >= 21 and { x = "a"; x + "b" == "ab" }; x }, true)
|
|
|
|
// With block.
|
|
#test(if true {
|
|
1
|
|
} + 2, 3)
|
|
|
|
// Mathematical identities.
|
|
#let nums = (
|
|
1, 3.14,
|
|
12pt, 3em, 12pt + 3em,
|
|
45deg,
|
|
90%,
|
|
13% + 10pt, 5% + 1em + 3pt,
|
|
2.3fr,
|
|
)
|
|
|
|
#for v in nums {
|
|
// Test plus and minus.
|
|
test(v + v - v, v)
|
|
test(v - v - v, -v)
|
|
|
|
// Test plus/minus and multiplication.
|
|
test(v - v, 0 * v)
|
|
test(v + v, 2 * v)
|
|
|
|
// Integer addition does not give a float.
|
|
if type(v) != int {
|
|
test(v + v, 2.0 * v)
|
|
}
|
|
|
|
if type(v) != relative and ("pt" not in repr(v) or "em" not in repr(v)) {
|
|
test(v / v, 1.0)
|
|
}
|
|
}
|
|
|
|
// Make sure length, ratio and relative length
|
|
// - can all be added to / subtracted from each other,
|
|
// - multiplied with integers and floats,
|
|
// - divided by integers and floats.
|
|
#let dims = (10pt, 1em, 10pt + 1em, 30%, 50% + 3cm, 40% + 2em + 1cm)
|
|
#for a in dims {
|
|
for b in dims {
|
|
test(type(a + b), type(a - b))
|
|
}
|
|
|
|
for b in (7, 3.14) {
|
|
test(type(a * b), type(a))
|
|
test(type(b * a), type(a))
|
|
test(type(a / b), type(a))
|
|
}
|
|
}
|
|
|
|
// Test division of different numeric types with zero components.
|
|
#for a in (0pt, 0em, 0%) {
|
|
for b in (10pt, 10em, 10%) {
|
|
test((2 * b) / b, 2)
|
|
test((a + b * 2) / b, 2)
|
|
test(b / (b * 2 + a), 0.5)
|
|
}
|
|
}
|
|
|
|
--- ops-multiply-inf-with-length ---
|
|
// Test that multiplying infinite numbers by certain units does not crash.
|
|
#(float("inf") * 1pt)
|
|
#(float("inf") * 1em)
|
|
#(float("inf") * (1pt + 1em))
|
|
|
|
--- ops-attempt-nan-length ---
|
|
// Test that trying to produce a NaN scalar (such as in lengths) does not crash.
|
|
#let infpt = float("inf") * 1pt
|
|
#test(infpt - infpt, 0pt)
|
|
#test(infpt + (-infpt), 0pt)
|
|
// TODO: this result is surprising
|
|
#test(infpt / float("inf"), 0pt)
|
|
|
|
--- ops-unary-bool ---
|
|
// Test boolean operators.
|
|
|
|
// Test not.
|
|
#test(not true, false)
|
|
#test(not false, true)
|
|
|
|
// And.
|
|
#test(false and false, false)
|
|
#test(false and true, false)
|
|
#test(true and false, false)
|
|
#test(true and true, true)
|
|
|
|
// Or.
|
|
#test(false or false, false)
|
|
#test(false or true, true)
|
|
#test(true or false, true)
|
|
#test(true or true, true)
|
|
|
|
// Short-circuiting.
|
|
#test(false and dont-care, false)
|
|
#test(true or dont-care, true)
|
|
|
|
--- ops-equality ---
|
|
// Test equality operators.
|
|
|
|
// Most things compare by value.
|
|
#test(1 == "hi", false)
|
|
#test(1 == 1.0, true)
|
|
#test(30% == 30% + 0cm, true)
|
|
#test(1in == 0% + 72pt, true)
|
|
#test(30% == 30% + 1cm, false)
|
|
#test("ab" == "a" + "b", true)
|
|
#test(() == (1,), false)
|
|
#test((1, 2, 3) == (1, 2.0) + (3,), true)
|
|
#test((:) == (a: 1), false)
|
|
#test((a: 2 - 1.0, b: 2) == (b: 2, a: 1), true)
|
|
#test("a" != "a", false)
|
|
|
|
// Functions compare by identity.
|
|
#test(test == test, true)
|
|
#test((() => {}) == (() => {}), false)
|
|
|
|
// Content compares field by field.
|
|
#let t = [a]
|
|
#test(t == t, true)
|
|
#test([] == [], true)
|
|
#test([a] == [a], true)
|
|
#test(grid[a] == grid[a], true)
|
|
#test(grid[a] == grid[b], false)
|
|
|
|
--- ops-compare ---
|
|
// Test comparison operators.
|
|
|
|
#test(13 * 3 < 14 * 4, true)
|
|
#test(5 < 10, true)
|
|
#test(5 > 5, false)
|
|
#test(5 <= 5, true)
|
|
#test(5 <= 4, false)
|
|
#test(45deg < 1rad, true)
|
|
#test(10% < 20%, true)
|
|
#test(50% < 40% + 0pt, false)
|
|
#test(40% + 0pt < 50% + 0pt, true)
|
|
#test(1em < 2em, true)
|
|
#test((0, 1, 2, 4) < (0, 1, 2, 5), true)
|
|
#test((0, 1, 2, 4) < (0, 1, 2, 3), false)
|
|
#test((0, 1, 2, 3.3) > (0, 1, 2, 4), false)
|
|
#test((0, 1, 2) < (0, 1, 2, 3), true)
|
|
#test((0, 1, "b") > (0, 1, "a", 3), true)
|
|
#test((0, 1.1, 3) >= (0, 1.1, 3), true)
|
|
#test((0, 1, datetime(day: 1, month: 12, year: 2023)) <= (0, 1, datetime(day: 1, month: 12, year: 2023), 3), true)
|
|
#test(("a", 23, 40, "b") > ("a", 23, 40), true)
|
|
#test(() <= (), true)
|
|
#test(() >= (), true)
|
|
#test(() <= (1,), true)
|
|
#test((1,) <= (), false)
|
|
|
|
--- ops-in ---
|
|
// Test `in` operator.
|
|
#test("hi" in "worship", true)
|
|
#test("hi" in ("we", "hi", "bye"), true)
|
|
#test("Hey" in "abHeyCd", true)
|
|
#test("Hey" in "abheyCd", false)
|
|
#test(5 in range(10), true)
|
|
#test(12 in range(10), false)
|
|
#test("" in (), false)
|
|
#test("key" in (key: "value"), true)
|
|
#test("value" in (key: "value"), false)
|
|
#test("Hey" not in "abheyCd", true)
|
|
#test("a" not
|
|
/* fun comment? */ in "abc", false)
|
|
|
|
--- ops-not-trailing ---
|
|
// Error: 10 expected keyword `in`
|
|
#("a" not)
|
|
|
|
--- func-with ---
|
|
// Test `with` method.
|
|
|
|
// Apply positional arguments.
|
|
#let add(x, y) = x + y
|
|
#test(add.with(2)(3), 5)
|
|
#test(add.with(2, 3)(), 5)
|
|
#test(add.with(2).with(3)(), 5)
|
|
#test((add.with(2))(4), 6)
|
|
#test((add.with(2).with(3))(), 5)
|
|
|
|
// Make sure that named arguments are overridable.
|
|
#let inc(x, y: 1) = x + y
|
|
#test(inc(1), 2)
|
|
|
|
#let inc2 = inc.with(y: 2)
|
|
#test(inc2(2), 4)
|
|
#test(inc2(2, y: 4), 6)
|
|
|
|
// Apply arguments to an argument sink.
|
|
#let times(..sink) = {
|
|
let res = sink.pos().product()
|
|
if sink.named().at("negate", default: false) { res *= -1 }
|
|
res
|
|
}
|
|
#test((times.with(2, negate: true).with(5))(), -10)
|
|
#test((times.with(2).with(5).with(negate: true))(), -10)
|
|
#test((times.with(2).with(5, negate: true))(), -10)
|
|
#test((times.with(2).with(negate: true))(5), -10)
|
|
|
|
--- ops-precedence-basic ---
|
|
// Multiplication binds stronger than addition.
|
|
#test(1+2*-3, -5)
|
|
|
|
// Subtraction binds stronger than comparison.
|
|
#test(3 == 5 - 2, true)
|
|
|
|
// Boolean operations bind stronger than '=='.
|
|
#test("a" == "a" and 2 < 3, true)
|
|
#test(not "b" == "b", false)
|
|
|
|
--- ops-precedence-boolean-ops ---
|
|
// Assignment binds stronger than boolean operations.
|
|
// Error: 2:3-2:8 cannot mutate a temporary value
|
|
#let x = false
|
|
#(not x = "a")
|
|
|
|
--- ops-precedence-unary ---
|
|
// Precedence doesn't matter for chained unary operators.
|
|
// Error: 3-12 cannot apply '-' to boolean
|
|
#(-not true)
|
|
|
|
--- ops-precedence-not-in ---
|
|
// Not in handles precedence.
|
|
#test(-1 not in (1, 2, 3), true)
|
|
|
|
--- ops-precedence-parentheses ---
|
|
// Parentheses override precedence.
|
|
#test((1), 1)
|
|
#test((1+2)*-3, -9)
|
|
|
|
// Error: 8-9 unclosed delimiter
|
|
#test({(1 + 1}, 2)
|
|
|
|
--- ops-associativity-left ---
|
|
// Math operators are left-associative.
|
|
#test(10 / 2 / 2 == (10 / 2) / 2, true)
|
|
#test(10 / 2 / 2 == 10 / (2 / 2), false)
|
|
#test(1 / 2 * 3, 1.5)
|
|
|
|
--- ops-associativity-right ---
|
|
// Assignment is right-associative.
|
|
#{
|
|
let x = 1
|
|
let y = 2
|
|
x = y = "ok"
|
|
test(x, none)
|
|
test(y, "ok")
|
|
}
|
|
|
|
--- ops-unary-minus-missing-expr ---
|
|
// Error: 4 expected expression
|
|
#(-)
|
|
|
|
--- ops-add-missing-rhs ---
|
|
// Error: 10 expected expression
|
|
#test({1+}, 1)
|
|
|
|
--- ops-mul-missing-rhs ---
|
|
// Error: 10 expected expression
|
|
#test({2*}, 2)
|
|
|
|
--- ops-unary-plus-on-content ---
|
|
// Error: 3-13 cannot apply unary '+' to content
|
|
#(+([] + []))
|
|
|
|
--- ops-unary-plus-on-string ---
|
|
// Error: 3-6 cannot apply '-' to string
|
|
#(-"")
|
|
|
|
--- ops-not-on-array ---
|
|
// Error: 3-9 cannot apply 'not' to array
|
|
#(not ())
|
|
|
|
--- ops-compare-relative-length-and-ratio ---
|
|
// Error: 3-19 cannot compare relative length and ratio
|
|
#(30% + 1pt <= 40%)
|
|
|
|
--- ops-compare-em-with-abs ---
|
|
// Error: 3-14 cannot compare 1em with 10pt
|
|
#(1em <= 10pt)
|
|
|
|
--- ops-compare-normal-float-with-nan ---
|
|
// Error: 3-22 cannot compare 2.2 with float.nan
|
|
#(2.2 <= float("nan"))
|
|
|
|
--- ops-compare-int-and-str ---
|
|
// Error: 3-26 cannot compare integer and string
|
|
#((0, 1, 3) > (0, 1, "a"))
|
|
|
|
--- ops-compare-array-nested-failure ---
|
|
// Error: 3-42 cannot compare 3.5 with float.nan
|
|
#((0, "a", 3.5) <= (0, "a", float("nan")))
|
|
|
|
--- ops-divide-by-zero-float ---
|
|
// Error: 3-12 cannot divide by zero
|
|
#(1.2 / 0.0)
|
|
|
|
--- ops-divide-by-zero-int ---
|
|
// Error: 3-8 cannot divide by zero
|
|
#(1 / 0)
|
|
|
|
--- ops-divide-by-zero-angle ---
|
|
// Error: 3-15 cannot divide by zero
|
|
#(15deg / 0deg)
|
|
|
|
--- ops-binary-arithmetic-error-message ---
|
|
// Special messages for +, -, * and /.
|
|
// Error: 3-10 cannot add integer and string
|
|
#(1 + "2", 40% - 1)
|
|
|
|
--- add-assign-int-and-str ---
|
|
// Error: 15-23 cannot add integer and string
|
|
#{ let x = 1; x += "2" }
|
|
|
|
--- ops-divide-ratio-by-length ---
|
|
// Error: 4-13 cannot divide ratio by length
|
|
#( 10% / 5pt )
|
|
|
|
--- ops-divide-em-by-abs ---
|
|
// Error: 3-12 cannot divide these two lengths
|
|
#(1em / 5pt)
|
|
|
|
--- ops-divide-relative-length-by-ratio ---
|
|
// Error: 3-19 cannot divide relative length by ratio
|
|
#((10% + 1pt) / 5%)
|
|
|
|
--- ops-divide-relative-lengths ---
|
|
// Error: 3-28 cannot divide these two relative lengths
|
|
#((10% + 1pt) / (20% + 1pt))
|
|
|
|
--- ops-subtract-int-from-ratio ---
|
|
// Error: 13-20 cannot subtract integer from ratio
|
|
#((1234567, 40% - 1))
|
|
|
|
--- ops-multiply-int-with-bool ---
|
|
// Error: 3-11 cannot multiply integer with boolean
|
|
#(2 * true)
|
|
|
|
--- ops-divide-int-by-length ---
|
|
// Error: 3-11 cannot divide integer by length
|
|
#(3 / 12pt)
|
|
|
|
--- multiply-negative-int-with-str ---
|
|
// Error: 3-10 number must be at least zero
|
|
#(-1 * "")
|
|
|
|
--- ops-assign ---
|
|
// Test assignment operators.
|
|
|
|
#let x = 0
|
|
#(x = 10) #test(x, 10)
|
|
#(x -= 5) #test(x, 5)
|
|
#(x += 1) #test(x, 6)
|
|
#(x *= x) #test(x, 36)
|
|
#(x /= 2.0) #test(x, 18.0)
|
|
#(x = "some") #test(x, "some")
|
|
#(x += "thing") #test(x, "something")
|
|
|
|
--- ops-assign-unknown-var-lhs ---
|
|
#{
|
|
// Error: 3-6 unknown variable: a-1
|
|
// Hint: 3-6 if you meant to use subtraction, try adding spaces around the minus sign
|
|
a-1 = 2
|
|
}
|
|
|
|
--- ops-assign-unknown-var-rhs ---
|
|
#{
|
|
let a = 2
|
|
a = 1-a
|
|
a = a -1
|
|
|
|
// Error: 7-10 unknown variable: a-1
|
|
// Hint: 7-10 if you meant to use subtraction, try adding spaces around the minus sign
|
|
a = a-1
|
|
}
|
|
|
|
--- ops-assign-unknown-parenthesized-variable ---
|
|
// Error: 4-5 unknown variable: x
|
|
#((x) = "")
|
|
|
|
--- ops-assign-destructuring-unknown-variable ---
|
|
// Error: 4-5 unknown variable: x
|
|
#((x,) = (1,))
|
|
|
|
--- ops-assign-to-temporary ---
|
|
// Error: 3-8 cannot mutate a temporary value
|
|
#(1 + 2 += 3)
|
|
|
|
--- ops-assign-to-invalid-unary-op ---
|
|
// Error: 2:3-2:8 cannot apply 'not' to string
|
|
#let x = "Hey"
|
|
#(not x = "a")
|
|
|
|
--- ops-assign-to-invalid-binary-op ---
|
|
// Error: 7-8 unknown variable: x
|
|
#(1 + x += 3)
|
|
|
|
--- ops-assign-unknown-variable ---
|
|
// Error: 3-4 unknown variable: z
|
|
#(z = 1)
|
|
|
|
--- ops-assign-to-std-constant ---
|
|
// Error: 3-7 cannot mutate a constant: rect
|
|
#(rect = "hi")
|
|
|
|
--- ops-assign-to-shadowed-std-constant ---
|
|
// Works if we define rect beforehand
|
|
// (since then it doesn't resolve to the standard library version anymore).
|
|
#let rect = ""
|
|
#(rect = "hi")
|