Skip to content

Tea Template Language

jappy edited this page Oct 15, 2011 · 19 revisions

Introduction to Tea
Embedding Tea
Lexical Structure
Templates
Functions
Grammar Summary

3 - Lexical Structure

The lexical structure of Tea is based on Java's. The chapter titled "Lexical Structure" in The Java Language Specification, defines most of Tea’s lexical structure.

Tea templates are defined in the unicode character set and the language is case-sensitive, just like Java. Tea uses Java's style and early processing of unicode escapes. Comments and white space are the same. Tea identifiers do not allow '$' characters, but are otherwise the same. String and number literals are very similar.

A template is composed of code and text regions. Code regions are delimited by "<%" and "%>" symbols. Any text not in the code region is output by the template as-is with only one minor conversion. All line break separator codes in text regions are converted to linefeeds '\n' (ASCII 10). No string escapes (including unicode escapes) are processed in text regions. Here is a template that mixes code and text regions:

<% template MyTemplate(PageInfo page) %>
<html><head><title><% page.title %></title></head>
<body>
<h1><% page.heading %></h1>
<table>
	<% foreach (item in page.list) { %>
		<tr><td><% item.name %></td><td><% item.desc %></td></tr>
	<% } %>
</table>
</body>
</html>

The code delimiter symbols are not actually Tea lexemes, but are instead processed in an initial preprocessing step that identifies text as being in a code region or not. The Tea scanner converts text regions into string literals. Put another way, text regions are simply multi-line string literals delimited by "%>" and "<%" symbols (see section 3.2.3 String Literals).

3.1 - Keywords

The following are the keywords in the Tea language. The reserved words "null", "true", "false", and "isa", discussed in the next sections, are not considered to be keywords.

Keyword:

and   
as   
break   
call   
continue   
define   
else   
foreach   
if   
import   
in   
not
or
reverse
template

3.2 - Literals

Literal:

_NullLiteral _
BooleanLiteral
StringLiteral
IntegerLiteral
FloatingPointLiteral

3.2.1 - Null Literal

NullLiteral:

null

3.2.2 - Boolean Literals

BooleanLiteral:

true
false

3.2.3 - String Literals

StringLiteral:

" StringCharacters (opt) "
' StringCharacters (opt) '

Tea string literals are for the most part, just like Java’s. The first difference is that they may be delimited by apostrophes instead of quotes. Apostrophes are used by Java to delimit character literals, but Tea does not have character literals.

The other difference is that octal escapes are not supported in strings, although \0 (the null character) is.

3.2.4 Integer Literals

IntegerLiteral:

_DecimalIntegerLiteral_   
_HexIntegerLiteral_   

Tea integer literals are just like Java integer literals except Tea does not support octal integer literals. As a result, decimal integer literals may start with any number of '0' digits.

3.2.5 - Floating Point Literals

The only difference between Tea’s floating-point literals and Java’s is that the decimal point must always have a digit before and after it.

3.3 - Separators

(	)	{	}	[	]
;	,	.	=>

3.4 - Operators

=	#	##	..	...	isa
==	!=	<	>	<=	>=
+	-	*	/	%	&

3.4.1 - Ellipsis Operator

The ellipsis operator "..." is used when declaring that templates require a substitution parameter, and it is used where block substitutions should be placed. In Tea, the ellipsis means, "more statements may follow."

4 - Templates

A Tea compilation unit consists of a single template. If templates are stored on a file system, then each template file must contain exactly one template definition. The name of the file must consist of the template name followed by the extension ".tea".

Every template must begin with a template declaration, which is defined in a code region (see section 3 Lexical Structure). Therefore, every template must begin with a code region. The template declaration defines the name of the template and any parameters that must be passed to the template.

Template:

TemplateDeclaration  StatementList (opt)   

TemplateDeclaration:

template Identifier ( FormalParameterList (opt) ) SubstitutionParameter (opt)

The template's formal parameter list follows the template name in the declaration and must be delimited by parentheses. A comma separates each parameter in the list. Formal parameters require a class name followed by an identifier, which names it. Tea does not allow primitive types to be declared, only object types. Object type names must be fully qualified except for those defined in the java.lang and java.util packages. The type declaration may define an array (or an array of arrays) just like in Java, using '[' and ']' symbols.

FormalParameterList

FormalParameter   
FormalParameterList. FormalParameter   

FormalParameter

Type  Variable   

Type:

Name    
Type  [ ]    

After the template declaration, a template can have text regions or code regions with any statements allowed by Tea. Statements can also appear inside the same code region that the template declaration was in.

Templates can invoke other templates, just like calling a function. Any template can be invoked by another template. Templates may also return a value to the caller. No special syntax is required in the template declaration to support callable templates or return types.

A template may require that it receive one substitution block parameter. This parameter is denoted by the ellipsis symbol '...'. With this parameter, any template invoking this template must pass in a block of code. A block of code begins with a left curly brace and ends with a right curly brace. The template that accepts the substitution parameter uses the substitution statement to insert the substitution.

Examples:

From SmallTemplate.tea:

<% template SmallTemplate(pkg.ContentObject obj) %>
<html><body>
This page was created on <% obj.date %>.
<br>
<%
user = obj.user
if (user != null and user.name != null) {
    'Hello ' user.name '!'
}
%>
</body></html>

From NewsPage.tea:

<% template NewsPage(org.news.NewsStory story)
// Pass a title and a substitution block to SimplePage 
call SimplePage(story.title, "#ffffff") {
    '<h1>' story.headline '</h1>'
    foreach (paragraph in story.body) {
        '<p>' paragraph.body
    }
}
%>

From SimplePage.tea, called from NewsPage:

<% template SimplePage(String title, String color) { ...  }
'<html><head><title>' title '</title></head>'
if (color == null) {
    '<body>'
}
else {
    '<body bgcolor="' & color & '">'
}
...
'</body></html>'
%>

4.1 - The Import Directive

The import reserved word provides an alternative to providing fully qualified Java class names and is analogous to the import statement in Java. It impacts the type names of the formal parameters that are passed into a template via the template keyword. It also can be used to qualify the type names referenced by the define and as reserved words. The import reserved word must appear before the template definition. An example of this follows:

<% 
import com.disney.vidio
template displayTitles()

define Title dvd = getRequest().attributes["dvd"]
...

In this example the Java data container class Title exists in the fully qualified Java class com.disney.video.Title. Note: Since the entire contents of a package is imported, if a class name exists in multiple packages then you must use the fully qualified names of the conflicting classes.

4.2 - Statements

Tea statements are used for variable assignments, controlling the flow of execution, performing looping, calling functions, and calling other templates. Statements are also used to output from a template and perform block substitutions. Statements need not terminate with a semi-colon, as they do in Java. Statements are not separated by new lines, but by a carefully designed language grammar. In very rare circumstances, a semi-colon is required to disambiguate statement separation.

Statement:

	EmptyStatement
	IfStatement
	ForeachStatement
	SubstitutionStatement
	AssignmentStatement
	CallStatement
	ExpressionStatement

The set of operations that can be performed by Tea statements is less than that provided by most application programming languages. This aids in enforcing a policy that keeps templates from performing complex operations that should be left to a hosting system.

4.2.1 - Semi-colons

Although Tea does not require the use of the semi-colon as a statement terminator, they can still be used. Java programmers are likely to use them out of habit or for stylistic reasons. Under certain special conditions, a semi-colon may be required to enforce statement separation. The following two statements

a
-b

would not be interpreted as printing the value of variable “a” followed by the negative value of variable “b”. Instead it would be

a – b

which means print the value of “a” minus “b”. This would not be a problem if the “-” token was not overloaded to mean both subtraction and negation. Here is the separated version:

a; -b

Alternatively, parentheses can be used, but it is a clumsier notation:

(a)
(-b)

In this example, if the parentheses were omitted from the first expression, the compiler would decide that "a" is a function receiving parameter "-b".

4.2.2 - Break Statement

The break statement can be used to exit from a loop. See 4.2.5 Looping.

Example:

foreach (count in 1..10) {
    count
    if (count > 4) {
        break
    }
}

Placing unreachable code after the break statement will result in an error at compile time.

4.2.3 - Continue Statement

The continue statement can be used to unconditionally branch to the top of a loop. See 4.2.5 Looping.

Example:

foreach (count in 1..10) {
    if (count % 2 == 0) {
        continue
    }
    count  // only odd numbers are printed  
}

Placing unreachable code after the continue statement will result in an error at compile time.

4.2.4 - If Statement

Tea's if statement is just like Java’s except in one way: braces are always required for its enclosed statements.

IfStatement:

if ( Expression ) Block ElseStatement(opt)

ElseStatement:

else Block
else ifStatement

Block:

{ StatementList (op)opt }

Examples:

if (fileName == null or fileName.length == 0) {
    // Assign a default filename if empty
    fileName = "default"
}

if (name == "Bob") {
    // Bob is welcome...
    "Hello Bob!"
}
else {
    // ...others are not
    "Go away, " & name & "."
}

if (count == -1) { count = 0 } else if (count >= 15) { count = count – 15 }

4.2.5 - Looping

The foreach statement is the only loop flow control statement in Tea. There is no way to modify the loop count variable from within the loop, but the break statement can be used to exit the loop. Foreach statements iterate over the values of an array, a collection or a range of values. When the optional reverse keyword is specified, the values are iterated in reverse order.

Like the if statement, blocks are required to be enclosed in braces.

ForeachStatement:

foreach ( Variable in Expression ) Block
foreach ( Variable as Type in Expression ) Block
foreach ( Variable in Expression reverse ) Block
foreach ( Variable as Type in Expression reverse ) Block
foreach ( Variable in Expression .. Expression ) Block
foreach ( Variable in Expression .. Expression reverse ) Block

The variable used in the foreach statement is re-assigned during the loop’s execution, and its type is redeclared to be the appropriate element type of the given expression (use the 'as' keyword to type collection and array iterators). If the loop variable was already declared outside of the loop, then the new declaration is promoted outside of the foreach scope. See also sections 4.2.7 Variable Assignment.

Example:

// Loop through and print every item in the page's list
foreach (item in page.list) {
    item
}

// Loop through in reverse
foreach (item in page.list reverse) {
    item
}

// Loop from 1 to 10, inclusive
foreach (count in 1..10) {
    "Count is " & count
}

// Print all the characters of a string
// NOTE: if the message length is zero, the range is 0..-1.
// Whenever the second value in the range is less than the
// first value, the foreach will loop zero times.  
message = "hello"
foreach (index in 0..message.length - 1) {
    "Letter at index " & index & " is: " & message[index] & "\n"
}

// Loop from 10 to 0, inclusive
foreach (count in 0..10 reverse) { ...  }

Unless looping through a range, the actual object looped through can be an array or a Collection (java.util.Collection). Objects that implement the Collection interface can be iterated in the reverse direction as well. Collection objects can implement the java.util.List, java.util.Set, and java.util.Map interfaces. When iterating java.util.Map, the loop variable will contain the map’s key, not the value. Use the key to dereference the value using normal associative array notation (i.e. value = myMap[key]). See 4.3.7 Array Creation.

If looping over a value of ranges, they can only be integers. Both the beginning and ending of the range is inclusive. If a number at the beginning or ending of a range is not an integer, it is rounded down to the nearest integer.

Because Java has no parameterized types, when a Collection is used in the foreach loop, Tea can only infer that its elements are Objects, even if they may be subclasses of Object (unless declaratively typed with ‘as’). If the template only needs to print the values of the elements, then this isn’t a problem, but if the template needs a specific property of the object, use the ‘as’ keyword to type the loop variable. Properties can be accessed as expected and the Tea compiler will perform the appropriate compile time type checks.

You can exit a loop by using the break statement (see 4.2.2 Break Statement).

You can unconditionally branch to the top of a loop by using the continue statement (see 4.2.3 Continue Statement).

4.2.6 - Substitution Statement

The substitution statement is only allowed in templates that declare that they receive a block substitution parameter. Templates can declare at most one block substitution parameter. The substitution statement is used to substitute the code for the block passed in at the statement’s location. The substitution statement is simply the ellipsis (...).

SubstitutionStatement:

...

A substitution statement can appear in multiple locations in a template and even in foreach and if statements. The most common use for a substitution is for creating “shell” templates, from the example earlier:

<% template SimplePage(String title, String color) { ...  }
'<html><head><title>' title '</title></head>'
if (color == null) {
    '<body>'
}
else {
    '<body bgcolor="' & color & '">'
}
...
'</body></html>'
%>

A substitution can also be passed to a context-defined function, using the special com.go.tea.runtime.Substitution class. This example defines a simple looping function:

public void loop(int count, Substitution s) throws Exception {
    while (--count >= 0) {
        s.substitute();
    }
}

The template might invoke this function as:

loop (100) {
    "This message is printed 100 times\n"
}

4.2.7 - Variable Assignment/Declaration

With the exception of passed-in parameters, Tea variables are normally not declared. Rather, their type is inferred based on assignment. If necessary, a variables type can be declared using the define or as reserved words. Unless specifically declared using these reserved words, a variable's type is dynamic, and can change after another assignment. Passed-in parameters behave like ordinary local variables in that they can be re-assigned, and they can change type. Variables can also be assigned by a foreach statement (see 4.2.5 Looping).

AssignmentStatement:

Variable = Expression

Assignments can only be made to variables. Array elements cannot be assigned values, like they can in Java. Assignments of this form are not allowed: a[i] = x; This restriction makes it possible to pass collections to templates without fear of the template modifying it.

Examples:

message = "Hello"		// Assign the string literal “Hello” to message

result = 50			// Assign the integer literal 50 to result

amount = result + 20		// Assign the calculated sum of result and 20

message = amount		// Re-assign message with the integer amount

Unlike Java, assignment statements cannot be expressions. This means that assignments cannot be chained together like they can in Java: a = b = c = 2.

Tea variables can also be declaratively typed as follows:

define Integer amount

or

amount = result + 20 as Integer

A variable’s type can also be a Java class. For example, a JavaBean class called Title that exists in the hypothetical com.disney.videos could be decared in this way:

define com.disney.videos.Title video

When used in conjunction with the import reserved word, this can be shortened to:

define Title video

4.2.8 - Expression Statement

All expressions can be statements (see 4.2 Expressions). When an expression takes on the form of a statement, the return value is sent directly to the template’s execution context. The execution context will typically take this value, convert it to a string, and write it to an output stream. This is the main way in which templates output data.

ExpressionStatement:

Expression   

Any expression, which is not part of another statement, is automatically passed to the built-in print function as if it were done directly. The following two statements are equivalent:

"Hello world"
print("Hello world")

Calling the print function directly in templates is not preferred, and tools that display execution context functions should hide the print function.

If the last statement in a template is an expression statement, the expression results (complete with type information) are passed back to the calling template. The caller can then use the call statement as a call expression (see 5.1 Calling Functions and Templates). If the caller does not do anything special, then the call expression is treated as an expression statement, and the caller automatically prints out the results.

4.3 - Expressions

An expression is any program fragment that produces some kind of value. Expressions are evaluated in a specific order, and parentheses can be used to override that ordering. A sub-expression bounded by parentheses is evaluated first within a sub-expression. Otherwise, the order of operations is the same as that for Java, wherever applicable. See section 7 Grammar Summary for the order of operations.

4.3.1 - Accessing Properties

All objects have properties, and these properties are accessible using the JavaBeans Introspector. Essentially, any object passed to or used in a template is a JavaBean. JavaBeans defines two kinds of properties, non-indexed and indexed. Tea can access all non-indexed properties on a JavaBean, but can only access a certain kind of indexed property.

Defining a non-indexed property in a Java class is really simple. The method must return something, must take no parameters, and its name must start with "get". The part of the method name that follows "get" defines the name of the property. Capitalization rules are applied so that the method getSizeOfThing defines the property "sizeOfThing" and the method getURL defines the property "URL".

Arrays, Collections and strings also have a special property named "length" which isn’t defined by a get method. For arrays and Collections, length is the number of elements they contain. For strings, it is the number of characters they contain. JavaBeans defines several different forms for indexed properties, but Tea only understands one kind: unnamed indexed properties. A method that returns something, takes a single parameter and is named "get" follows the design rules for unnamed indexed properties. Java's Map and List interfaces follow these rules, and Tea accesses elements from them via this design pattern.

LookupExpression:

Factor  Lookupopt

Lookup:

Lookupopt  .  Identifier   
Lookupopt  [ Expression ]   

A property access is called a lookup, and Tea has different syntax for its two supported kinds of lookups. A lookup can be applied to any expression, but it is usually applied to variables or chained to other lookups. Non-indexed properties are looked up from an expression using a dot (.) followed by the property name. Indexed properties are looked up using an expression bounded by square brackets.

If the type of an indexed property is Object, (like it is in Map and List) and an element type is defined for that class (by extending Map or List or the use of 'as'), then Tea will cast the object to that element type. An element type for a class that has an indexed property is defined the same as for Collections described by the foreach statement (see 4.1.4 Looping).

The Java signature of the field is

public static final Class ELEMENT_TYPE

Like its use in foreach, if an element is returned that can’t be cast to the specified element type, a ClassCastException is thrown at runtime.

Examples:

// Access the name object from user
name = user.name
// Access the first name
f = name.first
// Access the last name by chaining
last = user.name.last

// Access an element from the users array and get the age
"Age is " & users[10].age

// Lookup a team object by code
team = allTeams[“SEA”]

// Get the last character from a string
str[str.length - 1]

4.3.2 - Arithmetic

Tea can perform arithmetic on both integers and floating-point numbers. Supported operations are addition, subtraction, multiplication, division, division remainder and negation. The operators are +, -, *, /, %, and - respectively. In complex arithmetic expressions, negate is performed first, left-to-right. Multiplication, division, and remainder are performed next, followed by addition and subtraction. This order of operations matches Java's.

AdditiveExpression:

MultiplicativeExpression   
AdditiveExpression + MultiplicativeExpression   
AdditiveExpression – MultiplicativeExpression   

MultiplicativeExpression:

UnaryExpression   
MultiplicativeExpression * UnaryExpression   
MultiplicativeExpression / UnaryExpression   
MultiplicativeExpression % UnaryExpression   

Examples:

// Simple addition
x = a + b

// Increment a count
count = count + 1

// More complex example
result = -((value – 10.0/y) * 50) % 2

In order to perform a division on integer values, but produce a fractional result, adding 0.0 to at least one of the values forces a floating point division. The addition by zero is optimized away and is not actually performed.

x = 3
y = 2
// Integer division prints 1
x / y
"<br>"
// Floating point division prints 1.5
(x + 0.0) / y
"<br>"

4.3.3 - String Concatenation

String concatenation in Java is performed with the plus operator. Tea uses the ampersand (&) operator instead because of the way the template language determines type information. Overloading the plus operator would create typing ambiguities.

ConcatenateExpression:
AdditiveExpression
ConcatenateExpression & AdditiveExpression

Examples:

// Equivalent to text = "Hello World"
text = "Hello" & " " & 'World!'

y = 5
'498234 divided by "y" is ' & 498234 / y

Output:
498234 divided by "y" is 99646

Both the left and right values of a concatenation are converted to strings (if they are not already strings) using the hidden toString function. Any formatting settings are applied when performing the string conversion.

String concatenation can also be used to force an object to be converted to a string. Concatenating the empty string forces a string conversion, but the actual concatenation operation is optimized away.

x = 43554
x = x & ""
// x is now a string, and 5 is printed out
x.length

4.3.4 - Relational Tests

Tea supports expressions for performing relational tests. They are used most often in the context of an if statement condition, although their result can be assigned to a variable. The resulting type of a relational test is always boolean. Six operators are supported, and they fall into two categories: equality tests and comparison tests.

Equality tests are equals (==) and not-equals (!=), and they work for any kind of data type. When an equality test is performed against a string, the other operand is also converted to a string. The following test is legal and performs a string equality test: 5 == "5".

Whenever an equality test is performed on objects, the "equals" method is invoked is possible. If either operand is null, the Tea compiler ensures that no NullPointerException is ever generated, and it may not call the equals method at all if a test is being made against null. The following test will never invoke the equals method: x == null.

Comparison tests are less than (<), greater than (>), less than or equal (<=), and greater than or equal (>=). Comparison tests can be performed on numbers, strings, or any object that implements the Comparable interface. Both values in the comparison must be of the same type. Unlike equality tests against strings, comparisons against strings never force a string conversion.

A comparison between two numbers will perform a numerical comparison, and comparisons between all other objects is performed using the "compareTo" method defined in the Comparable interface.

EqualityExpression:
RelationalExpression
EqualityExpression == RelationalExpression
EqualityExpression != RelationalExpression

RelationalExpression:
ConcatenateExpression
RelationalExpression < ConcatenateExpression
RelationalExpression > ConcatenateExpression
RelationalExpression <= ConcatenateExpression
RelationalExpression >= ConcatenateExpression
RelationalExpression isa Type

Examples:

// Check that the items are not null before looping through them
if (items != null) {
    foreach(item in items) {
        "Item is " & item
    }
}

// Do names A-M followed by names N-Z
"Everyone from A-M<br>"
foreach (name in allNames) {
    if (name < "N") {
        name & "<br>"
    }
}
"<p>Everyone from N-Z<br>"
foreach (name in allNames) {
    if (name >= "N") {
        name & "<br>"
    }
}

4.3.5 - "isa" Operator

The "isa" operator is a special operator that works both like Java's instanceof operator and Java's cast expressions. The left side of isa must be a simple variable expression. When used with an if statement, the tested variable is automatically cast to the given type, within the scope of the if statement’s then or else blocks.

Examples:

if (content isa app.pkg.News) {
    "Story is: " content.story
}

if (item isa app.pkg.Feature and content isa app.pkg.News) {
    "Info is: " item.info
    "Story is: " content.story
}
else if (item isa Object[]) {
    foreach (element in item) {
        element “<br>”
    }
}

4.3.6 - Logical Operators

Logical operations are often used in conjunction with relational tests because they only operate on booleans. Like Java, logical operations will short circuit. Tea's logical operators are "or", "and" and "not". This differs from Java in that the symbols ||, &&, and ! are used to denote these operations.

Examples:

notitle = (title == null or title.sub == null)
	
if (section == "Sports" and sport == "MLB") {
    "Major League Baseball"
}

value = not (x and y)

4.3.7 - Array Creation

Tea templates can define read-only, pre-initialized arrays. Arrays created by this expression can be indexed (zero-based) or associative (a Map). The array values can have any data type, and they can be composed of either constants or any expression. Associative arrays can have any type used for their keys, but usually strings or numbers will be used.

Based on the type of elements defined in the array, a common type is inferred. For any object retrieved from the array, the only properties that are accessible are those defined for the common type.

Arrays defined in templates can be used in the same way as an array passed to a template. This means that elements are accessed using array property syntax (i.e. x[i]). In the case of indexed arrays, they have a length property and can be used in a foreach statement.

The array creation expression is prefixed with a single hash (#) to denote an indexed array, and a double hash (##) to denote an associative array. The list of elements is comma-delimited (the Perl like "=>" can be used as an alternative to comma for associative arrays) and is contained inside parentheses. Associative arrays must have an even number of elements in the list, and list pattern is key, value, key, value, etc. Associative arrays are backed by an instance of java.util.LinkedHashMap. This preserves the item ordering when iterating associative arrays (with foreach) defined in this way.

NewArrayExpression:
	# ( Listopt )
        ## ( Listopt )

Examples:

// An indexed array containing some letters
vowels = #("a", "e", "i", "o", "u")
// Access the 3rd vowel
letter = vowels[2]

// Loop through some words
foreach (word in #("The", "quick", "brown", "fox", "jumped")) {
    "Word: " & word & "<br>"
}

// Iterate over some special values
foreach (value in #(2,3,5,7,11,13,17,19,23,29)) {
    "The value is " & cardinal(value) & "<br>"
}

// Map abbreviations to descriptions
map = ##("MLB" => "Major League Baseball", 
         "NBA" => "National Basketball Association",
         "NHL" => "National Hockey League",
         "NFL" => "National Football League")
"<title>" & map[sport.abbrev] & "</title>"

New indexed arrays are always implemented as ordinary Java arrays, and can be passed to functions and other templates in the usual fashion. Associative arrays will implement the Map interface, and in the current implementation, LinkedHashMap is used. When associative arrays are passed to and from other templates, the element type information is preserved.

5 FUNCTIONS

5.1 - Calling Functions and Templates

The call statement is used to invoke other templates or to call built-in Java functions. Both templates and Java functions may return a value. Ones that don't, (they return void) are invoked with a call statement. Otherwise, it is invoked with a call expression. The syntax of the two types of calls are identical, and are only distinguished by the presence or absence of a return value. A template returns a value using its last expression statement (see 4.1.7 Expression Statement).

CallStatement / CallExpression:
Name ( Listopt ) Blockopt
call  Name ( Listopt ) Blockopt

The optional block part of the call statement is used to pass a block to another template or to a built in Java function. A template can be called by its short name or its full name. A template's short name is the same as defined in its declaration. Its full name is based on the root directory from which it is stored. Essentially, templates can be packaged, but templates don’t have package declarations. For example, the full name of a template declared as "header" in the "common" directory is named "common.header". When a template invokes another template within the same package, the full name is not required.

Functions are defined in a template's runtime context class. The template has no control over what context it receives. It is the responsibility of the hosting system to provide one. Any function that contains '$' characters in the name can be invoked in Tea by substituting '.' characters.

A set of standard functions are defined in the Java interface com.go.tea.runtime.Context and its sub-interface, com.go.tea.runtime.UtilityContext. Functions that apply to locale and formatting are "sticky" and remain active until the top-most template returns or the setting is changed again. In addition, a called template can change these settings, possibly affecting the caller. The string manipulation functions do not alter the first string argument, but instead return (possibly) a new string with changes.

Examples:

// Convert to lowercase 
message = toLowerCase(message)

// Invoke a template named "header" in the same package as this one
call header("Header text")

// Invoke a template using its full name
call common.header("Header text")

// Invoke a template and pass a substitution block
name = "John"
call bigtext() {
    if (name == "Bob") {
	// Bob is welcome...
	"Hello Bob!"
    } else {
	// ...others are not
	"Go away, " & name & "."
    }
}

The header and bigtext templates are defined as follows:

<% template header(String title) %>
<html>
<head><title><% title %></title></head>
<body>

<% template bigtext() {...} %>
<font size="50"><% ...  %></font>

5.2 - Function List

setLocale
setLocale(Locale locale)
setLocale(String language, String country)
setLocale(String language, String country, String variant)

Setting the locale resets date and number formats to the default for that locale. Setting a locale of null resets date and number formats to the system defaults. Since a template cannot create a specific Locale object itself, the first form is most useful if the hosting system provides a Locale object via a function or bean property.

getLocale
java.util.Locale getLocale()

Returns the current locale setting. Null if none set.

getAvailableLocales
Locale[] getAvailableLocales()

This function returns, in Locale objects, a list of all the available locales in the system. This is useful for writing a template that displays the capabilities of its hosting system.

nullFormat
nullFormat(String format)

By default, null references are printed with the string "null". This function allows that setting to be overridden. Passing null sets the format back to "null".

getNullFormat
String getNullFormat()

Returns the current null format specification. Null if none set.

dateFormat
dateFormat(String format)
dateFormat(String format, String timeZoneID)

Defines a format to use when printing dates from templates. Passing null sets the format back to the default. The format string is of the form "MM d yyyy". Date formatting is provided by the Java class java.text.SimpleDateFormat. Refer to its documentation for more information.

getDateFormat
String getDateFormat()

Returns the current date format specification.

getAvailableTimeZones
TimeZone[] getAvailableTimeZones()

Returns, in TimeZone objects, a list of all the available time zones in the system. This is useful for writing a template that displays the capabilities of its hosting system.

getDateFormatTimeZone
String getDateFormatTimeZone()

Returns the current date format time zone.

numberFormat
numberFormat(String)

Defines a format to use when printing numbers from templates. Passing null sets the format back to the default. The format string is of the form "#.#". Number formatting is provided by the Java class java.text.DecimalFormat. Refer to its documentation for more information.

getNumberFormat
String getNumberFormat()

Returns the current number format specification. Null if none set.

getNumberFormatInfinity
String getNumberFormatInfinity()

Returns the current number format for infinity. Null if none set.

getNumberFormatNaN
String getNumberFormatNaN()

Returns the current number format for NaN (Not-A-Number). Null if none set.

currentDate
Date currentDate()

Returns a Date object with the current date and time of when this function is called. Subsequent calls within the same template may return a different date.

startsWith
boolean startsWith(String str, String prefix)

Tests if the given string starts with the given prefix.

endsWith
boolean endsWith(String str, String suffix)

Tests if the given string ends with the given suffix.

find
int[] find(String str, String search)
int[] find(String str, String search, int fromIndex)

Finds the indices (in an array) for each occurrence of the given search string in the source string, optionally starting from the given index.

findFirst
int findFirst(String str, String search)
int findFirst(String str, String search, int fromIndex)

Finds the index of the first occurrence of the given search string in the source string, optionally starting from the given index, or -1 if not found.

findLast
int findLast(String str, String search)
int findLast(String str, String search, int fromIndex)

Finds the index of the last occurrence of the given search string in the source string, optionally starting from the given index, or -1 if not found.

substring
String substring(String str, int start)
String substring(String str, int start, int end)

Returns a substring using the substring method defined for the Java String class. Start and end position is zero-based, and the end index is exclusive.

toLowerCase
String toLowerCase(String str)

Returns a new string of all lowercase letters from the contents of the string passed in.

toUpperCase
String toUpperCase(String str)

Returns a new string of all uppercase letters from the contents of the string passed in.

trim
String trim(String str)

Trims all leading and trailing whitespace characters from the given string.

trimLeading
String trimLeading(String str)

Trims all leading whitespace characters from the given string.

trimTrailing
String trimTrailing(String str)

Trims all trailing whitespace characters from the given string.

replace
String replace(String source, String pattern, String replacement)
String replace(String source, String pattern, String replacement, 
               int fromIndex)
String replace(String source, Map patternReplacements)

Replaces all exact matches of the given pattern in the source string with the provided replacement, optionally starting from the given index. Another form applies string replacements using the pattern-replacement pairs provided by the given map (associative array). The longest matching pattern is used for selecting an appropriate replacement.

replaceFirst
String replaceFirst(String source, String pattern, String replacement)
String replaceFirst(String source, String pattern, String replacement,
                    int fromIndex)

Replaces the first exact match of the given pattern in the source string with the provided replacement, optionally starting from the given index.

replaceLast
String replaceLast(String source, String pattern, String replacement)
String replaceLast(String source, String pattern, String replacement,
                   int fromIndex)

Replaces the last exact match of the given pattern in the source string with the provided replacement, optionally starting from the given index.

shortOrdinal
String shortOrdinal(Number n)

Returns a number’s abbreviated ordinal text i.e. 1st, 2nd, 3rd etc. Tea currently does not support localized ordinal numbers.

ordinal
String ordinal(Number n)

Returns a number’s ordinal text i.e. first, second, third etc. Tea currently does not support localized ordinal numbers.

cardinal
String cardinal(Number n)

Returns a number’s cardinal text i.e. one, two, three etc. Tea currently does not support localized cardinal numbers.

6 - TYPE CONVERSION

Although variables in Tea are implicitly typed, Tea is still strongly typed and type conversion operations are performed automatically.

The exception to this are variables that are declaratively typed using define or as. In this case, when an assignment occurs along with a type cast, the compiler will verify that this cast is valid and report an error if not.

Conversions that are automatically applied are usually guaranteed to always succeed. For example, an automatic conversion from Integer to String is preferred over String to Integer because Strings can represent the value of any Integer, but the opposite is not always true. Other kinds of conversions are applied to convert primitive data types to objects and vice versa. When performing addition on an int and an Integer, the Integer is converted to an int in order for the addition to be applied. This conversion could fail, however, if the Integer type being converted references null. In this case, a NullPointerException is thrown.

If the compiler cannot perform a conversion for any reason, the type checker generates an error and the template doesn’t compile. This is generally caused by a true error on the part of the template writer, for example trying to use a boolean where an int should go.

6.1 - Method Binding

Since Java methods can be overloaded to accept parameters of different types, the Tea compiler needs to decide on which method to bind to based on which types fit best. In Java, a cast operation can be applied which forces the binding if there is an ambiguity. Since Tea has no cast operation, the compiler binds to a method by calculating a conversion cost to all the parameters.

A conversion from an int to an Integer is deemed to cost higher than converting an int to a long. If a method is overloaded to accept an Integer or a long as a parameter and the template has an int, the method that takes the long is preferred and is called instead.

6.2 - Primitive Types

Tea treats primitive types as though they were objects, except it will use primitive values wherever possible for performance. If a template passes an int to a method that accepts an Integer, a Number or just any Object, the int is converted to an Integer. If the template passes a Number to a method that accepts an int, the intValue method is called to get at the int value.

Conversion can also be applied between primitives of differing precision. If given a choice, the compiler attempts to apply the conversion with the least loss of precision. A conversion of an int to a double is preferred over the reverse, for example. Also, if given a choice, the compiler will choose to apply a conversion that does not create a new object. Tea only performs conversions between primitive data types that represent numbers. For example, a number cannot be used where a boolean is accepted. Even though Java treats char as an unsigned 16-bit number, Tea treats it as a character, and it often gets converted to a String.

A numerical conversion from an integer to a double or float can always be forced by adding 0.0 or 0.0f. This is useful for displaying the result of a division without losing the fractional part.

6.3 - Conversion to String

Any object and primitive type in Java can be converted to a string. If given the choice, however, the Tea compiler avoids doing this as it is considered a last resort. Binding an int to an Object parameter is preferred over a String parameter because it will convert it to an Integer, which is a more precise representation.

String conversion may cause a formatting operation to be applied. If the object being converted is null, the current null format is applied (see 5.2 Function List). If a date or a number, the current date or number format is applied.

The string concatenation operation, '&', always forces its operands to be converted to strings, and the resulting expression is (of course) a string. Concatenating with "" is a convenient way of forcing an expression to be converted to a string.

6.4 - Variable Promotion

A variable assignment in the scope of a block can be promoted to appear after that scope. This applies to if statements, foreach statements and calls that accept blocks. Consider the following code:

w = "Monday" // w is a String
if (value == "something") {
    x = "Hello" // x is a String
    y = 76.6    // y is a double
    z = "ZZZ"   // z is a String
}
else {
    w = currentDate() // w is a Date
    x = 56            // x is an int
    y = 89            // y is an int
}

After the if statement, the following variables will be available for use: w, x and y with the respective types: Object, Object and double. The z variable is unavailable since it was only assigned in the then part of the if statement, and therefore there is no guarantee that will contain a legal value.

When promoting variables from an inner scope, the Tea compiler chooses a compatible type that represents the merged type as specifically as possible. For objects, it does this by analyzing the inheritance graph. The common type produced for java.util.Vector and java.util.Set is java.util.Collection.