1
0
mirror of https://github.com/danog/liquid.git synced 2024-11-30 07:28:56 +01:00

Implement tablerow

This commit is contained in:
Oliver Steele 2017-07-15 10:38:12 -04:00
parent a2a4a1a5ec
commit cd234476f7
7 changed files with 134 additions and 48 deletions

View File

@ -30,7 +30,6 @@ The [feature parity board](https://github.com/osteele/liquid/projects/1) lists d
In brief, these aren't implemented: In brief, these aren't implemented:
- The `tablerow` tag
- Error modes - Error modes
- Whitespace control - Whitespace control

View File

@ -1,7 +1,8 @@
%{ %{
package expressions package expressions
import ( import (
"fmt" "fmt"
"math"
"github.com/osteele/liquid/evaluator" "github.com/osteele/liquid/evaluator"
) )
@ -101,7 +102,7 @@ int_or_var:
| IDENTIFIER { name := $1; $$ = func(ctx Context) interface{} { return ctx.Get(name) } } | IDENTIFIER { name := $1; $$ = func(ctx Context) interface{} { return ctx.Get(name) } }
; ;
loop_modifiers: /* empty */ { $$ = loopModifiers{} } loop_modifiers: /* empty */ { $$ = loopModifiers{Cols: math.MaxUint32} }
| loop_modifiers IDENTIFIER { | loop_modifiers IDENTIFIER {
switch $2 { switch $2 {
case "reversed": case "reversed":
@ -113,6 +114,12 @@ loop_modifiers: /* empty */ { $$ = loopModifiers{} }
} }
| loop_modifiers KEYWORD LITERAL { // TODO can this be a variable? | loop_modifiers KEYWORD LITERAL { // TODO can this be a variable?
switch $2 { switch $2 {
case "cols":
cols, ok := $3.(int)
if !ok {
panic(ParseError(fmt.Sprintf("loop cols must an integer")))
}
$1.Cols = cols
case "limit": case "limit":
limit, ok := $3.(int) limit, ok := $3.(int)
if !ok { if !ok {

View File

@ -37,6 +37,7 @@ type loopModifiers struct {
Limit *int Limit *int
Offset int Offset int
Reversed bool Reversed bool
Cols int
} }
// A When is a parse of a {% when %} clause // A When is a parse of a {% when %} clause

View File

@ -7,15 +7,16 @@ import __yyfmt__ "fmt"
import ( import (
"fmt" "fmt"
"github.com/osteele/liquid/evaluator" "github.com/osteele/liquid/evaluator"
"math"
) )
func init() { func init() {
// This allows adding and removing references to fmt in the rules below, // This allows adding and removing references to fmt in the rules below,
// without having to edit the import statement to avoid erorrs each time. // without having to comment and un-comment the import statement above.
_ = fmt.Sprint("") _ = fmt.Sprint("")
} }
//line expressions.y:15 //line expressions.y:16
type yySymType struct { type yySymType struct {
yys int yys int
name string name string
@ -539,87 +540,87 @@ yydefault:
case 1: case 1:
yyDollar = yyS[yypt-2 : yypt+1] yyDollar = yyS[yypt-2 : yypt+1]
//line expressions.y:45 //line expressions.y:46
{ {
yylex.(*lexer).val = yyDollar[1].f yylex.(*lexer).val = yyDollar[1].f
} }
case 2: case 2:
yyDollar = yyS[yypt-5 : yypt+1] yyDollar = yyS[yypt-5 : yypt+1]
//line expressions.y:46 //line expressions.y:47
{ {
yylex.(*lexer).Assignment = Assignment{yyDollar[2].name, &expression{yyDollar[4].f}} yylex.(*lexer).Assignment = Assignment{yyDollar[2].name, &expression{yyDollar[4].f}}
} }
case 3: case 3:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:49 //line expressions.y:50
{ {
yylex.(*lexer).Cycle = yyDollar[2].cycle yylex.(*lexer).Cycle = yyDollar[2].cycle
} }
case 4: case 4:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:50 //line expressions.y:51
{ {
yylex.(*lexer).Loop = yyDollar[2].loop yylex.(*lexer).Loop = yyDollar[2].loop
} }
case 5: case 5:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:51 //line expressions.y:52
{ {
yylex.(*lexer).When = When{yyDollar[2].exprs} yylex.(*lexer).When = When{yyDollar[2].exprs}
} }
case 6: case 6:
yyDollar = yyS[yypt-2 : yypt+1] yyDollar = yyS[yypt-2 : yypt+1]
//line expressions.y:54 //line expressions.y:55
{ {
yyVAL.cycle = yyDollar[2].cyclefn(yyDollar[1].s) yyVAL.cycle = yyDollar[2].cyclefn(yyDollar[1].s)
} }
case 7: case 7:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:57 //line expressions.y:58
{ {
h, t := yyDollar[2].s, yyDollar[3].ss h, t := yyDollar[2].s, yyDollar[3].ss
yyVAL.cyclefn = func(g string) Cycle { return Cycle{g, append([]string{h}, t...)} } yyVAL.cyclefn = func(g string) Cycle { return Cycle{g, append([]string{h}, t...)} }
} }
case 8: case 8:
yyDollar = yyS[yypt-1 : yypt+1] yyDollar = yyS[yypt-1 : yypt+1]
//line expressions.y:61 //line expressions.y:62
{ {
vals := yyDollar[1].ss vals := yyDollar[1].ss
yyVAL.cyclefn = func(h string) Cycle { return Cycle{Values: append([]string{h}, vals...)} } yyVAL.cyclefn = func(h string) Cycle { return Cycle{Values: append([]string{h}, vals...)} }
} }
case 9: case 9:
yyDollar = yyS[yypt-0 : yypt+1] yyDollar = yyS[yypt-0 : yypt+1]
//line expressions.y:68 //line expressions.y:69
{ {
yyVAL.ss = []string{} yyVAL.ss = []string{}
} }
case 10: case 10:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:69 //line expressions.y:70
{ {
yyVAL.ss = append([]string{yyDollar[2].s}, yyDollar[3].ss...) yyVAL.ss = append([]string{yyDollar[2].s}, yyDollar[3].ss...)
} }
case 11: case 11:
yyDollar = yyS[yypt-2 : yypt+1] yyDollar = yyS[yypt-2 : yypt+1]
//line expressions.y:72 //line expressions.y:73
{ {
yyVAL.exprs = append([]Expression{&expression{yyDollar[1].f}}, yyDollar[2].exprs...) yyVAL.exprs = append([]Expression{&expression{yyDollar[1].f}}, yyDollar[2].exprs...)
} }
case 12: case 12:
yyDollar = yyS[yypt-0 : yypt+1] yyDollar = yyS[yypt-0 : yypt+1]
//line expressions.y:74 //line expressions.y:75
{ {
yyVAL.exprs = []Expression{} yyVAL.exprs = []Expression{}
} }
case 13: case 13:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:75 //line expressions.y:76
{ {
yyVAL.exprs = append([]Expression{&expression{yyDollar[2].f}}, yyDollar[3].exprs...) yyVAL.exprs = append([]Expression{&expression{yyDollar[2].f}}, yyDollar[3].exprs...)
} }
case 14: case 14:
yyDollar = yyS[yypt-1 : yypt+1] yyDollar = yyS[yypt-1 : yypt+1]
//line expressions.y:78 //line expressions.y:79
{ {
s, ok := yyDollar[1].val.(string) s, ok := yyDollar[1].val.(string)
if !ok { if !ok {
@ -629,40 +630,40 @@ yydefault:
} }
case 15: case 15:
yyDollar = yyS[yypt-4 : yypt+1] yyDollar = yyS[yypt-4 : yypt+1]
//line expressions.y:86 //line expressions.y:87
{ {
name, expr, mods := yyDollar[1].name, yyDollar[3].f, yyDollar[4].loopmods name, expr, mods := yyDollar[1].name, yyDollar[3].f, yyDollar[4].loopmods
yyVAL.loop = Loop{name, &expression{expr}, mods} yyVAL.loop = Loop{name, &expression{expr}, mods}
} }
case 16: case 16:
yyDollar = yyS[yypt-6 : yypt+1] yyDollar = yyS[yypt-6 : yypt+1]
//line expressions.y:92 //line expressions.y:93
{ {
yyVAL.f = makeRangeExpr(yyDollar[2].f, yyDollar[5].f) yyVAL.f = makeRangeExpr(yyDollar[2].f, yyDollar[5].f)
} }
case 18: case 18:
yyDollar = yyS[yypt-1 : yypt+1] yyDollar = yyS[yypt-1 : yypt+1]
//line expressions.y:100 //line expressions.y:101
{ {
val := yyDollar[1].val val := yyDollar[1].val
yyVAL.f = func(Context) interface{} { return val } yyVAL.f = func(Context) interface{} { return val }
} }
case 19: case 19:
yyDollar = yyS[yypt-1 : yypt+1] yyDollar = yyS[yypt-1 : yypt+1]
//line expressions.y:101 //line expressions.y:102
{ {
name := yyDollar[1].name name := yyDollar[1].name
yyVAL.f = func(ctx Context) interface{} { return ctx.Get(name) } yyVAL.f = func(ctx Context) interface{} { return ctx.Get(name) }
} }
case 20: case 20:
yyDollar = yyS[yypt-0 : yypt+1] yyDollar = yyS[yypt-0 : yypt+1]
//line expressions.y:104 //line expressions.y:105
{ {
yyVAL.loopmods = loopModifiers{} yyVAL.loopmods = loopModifiers{Cols: math.MaxUint32}
} }
case 21: case 21:
yyDollar = yyS[yypt-2 : yypt+1] yyDollar = yyS[yypt-2 : yypt+1]
//line expressions.y:105 //line expressions.y:106
{ {
switch yyDollar[2].name { switch yyDollar[2].name {
case "reversed": case "reversed":
@ -674,9 +675,15 @@ yydefault:
} }
case 22: case 22:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:114 //line expressions.y:115
{ // TODO can this be a variable? { // TODO can this be a variable?
switch yyDollar[2].name { switch yyDollar[2].name {
case "cols":
cols, ok := yyDollar[3].val.(int)
if !ok {
panic(ParseError(fmt.Sprintf("loop cols must an integer")))
}
yyDollar[1].loopmods.Cols = cols
case "limit": case "limit":
limit, ok := yyDollar[3].val.(int) limit, ok := yyDollar[3].val.(int)
if !ok { if !ok {
@ -696,63 +703,63 @@ yydefault:
} }
case 23: case 23:
yyDollar = yyS[yypt-1 : yypt+1] yyDollar = yyS[yypt-1 : yypt+1]
//line expressions.y:136 //line expressions.y:143
{ {
val := yyDollar[1].val val := yyDollar[1].val
yyVAL.f = func(Context) interface{} { return val } yyVAL.f = func(Context) interface{} { return val }
} }
case 24: case 24:
yyDollar = yyS[yypt-1 : yypt+1] yyDollar = yyS[yypt-1 : yypt+1]
//line expressions.y:137 //line expressions.y:144
{ {
name := yyDollar[1].name name := yyDollar[1].name
yyVAL.f = func(ctx Context) interface{} { return ctx.Get(name) } yyVAL.f = func(ctx Context) interface{} { return ctx.Get(name) }
} }
case 25: case 25:
yyDollar = yyS[yypt-2 : yypt+1] yyDollar = yyS[yypt-2 : yypt+1]
//line expressions.y:138 //line expressions.y:145
{ {
yyVAL.f = makeObjectPropertyExpr(yyDollar[1].f, yyDollar[2].name) yyVAL.f = makeObjectPropertyExpr(yyDollar[1].f, yyDollar[2].name)
} }
case 26: case 26:
yyDollar = yyS[yypt-4 : yypt+1] yyDollar = yyS[yypt-4 : yypt+1]
//line expressions.y:139 //line expressions.y:146
{ {
yyVAL.f = makeIndexExpr(yyDollar[1].f, yyDollar[3].f) yyVAL.f = makeIndexExpr(yyDollar[1].f, yyDollar[3].f)
} }
case 27: case 27:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:140 //line expressions.y:147
{ {
yyVAL.f = yyDollar[2].f yyVAL.f = yyDollar[2].f
} }
case 29: case 29:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:145 //line expressions.y:152
{ {
yyVAL.f = makeFilter(yyDollar[1].f, yyDollar[3].name, nil) yyVAL.f = makeFilter(yyDollar[1].f, yyDollar[3].name, nil)
} }
case 30: case 30:
yyDollar = yyS[yypt-4 : yypt+1] yyDollar = yyS[yypt-4 : yypt+1]
//line expressions.y:146 //line expressions.y:153
{ {
yyVAL.f = makeFilter(yyDollar[1].f, yyDollar[3].name, yyDollar[4].filter_params) yyVAL.f = makeFilter(yyDollar[1].f, yyDollar[3].name, yyDollar[4].filter_params)
} }
case 31: case 31:
yyDollar = yyS[yypt-1 : yypt+1] yyDollar = yyS[yypt-1 : yypt+1]
//line expressions.y:150 //line expressions.y:157
{ {
yyVAL.filter_params = []valueFn{yyDollar[1].f} yyVAL.filter_params = []valueFn{yyDollar[1].f}
} }
case 32: case 32:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:152 //line expressions.y:159
{ {
yyVAL.filter_params = append(yyDollar[1].filter_params, yyDollar[3].f) yyVAL.filter_params = append(yyDollar[1].filter_params, yyDollar[3].f)
} }
case 34: case 34:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:156 //line expressions.y:163
{ {
fa, fb := yyDollar[1].f, yyDollar[3].f fa, fb := yyDollar[1].f, yyDollar[3].f
yyVAL.f = func(ctx Context) interface{} { yyVAL.f = func(ctx Context) interface{} {
@ -762,7 +769,7 @@ yydefault:
} }
case 35: case 35:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:163 //line expressions.y:170
{ {
fa, fb := yyDollar[1].f, yyDollar[3].f fa, fb := yyDollar[1].f, yyDollar[3].f
yyVAL.f = func(ctx Context) interface{} { yyVAL.f = func(ctx Context) interface{} {
@ -772,7 +779,7 @@ yydefault:
} }
case 36: case 36:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:170 //line expressions.y:177
{ {
fa, fb := yyDollar[1].f, yyDollar[3].f fa, fb := yyDollar[1].f, yyDollar[3].f
yyVAL.f = func(ctx Context) interface{} { yyVAL.f = func(ctx Context) interface{} {
@ -782,7 +789,7 @@ yydefault:
} }
case 37: case 37:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:177 //line expressions.y:184
{ {
fa, fb := yyDollar[1].f, yyDollar[3].f fa, fb := yyDollar[1].f, yyDollar[3].f
yyVAL.f = func(ctx Context) interface{} { yyVAL.f = func(ctx Context) interface{} {
@ -792,7 +799,7 @@ yydefault:
} }
case 38: case 38:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:184 //line expressions.y:191
{ {
fa, fb := yyDollar[1].f, yyDollar[3].f fa, fb := yyDollar[1].f, yyDollar[3].f
yyVAL.f = func(ctx Context) interface{} { yyVAL.f = func(ctx Context) interface{} {
@ -802,7 +809,7 @@ yydefault:
} }
case 39: case 39:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:191 //line expressions.y:198
{ {
fa, fb := yyDollar[1].f, yyDollar[3].f fa, fb := yyDollar[1].f, yyDollar[3].f
yyVAL.f = func(ctx Context) interface{} { yyVAL.f = func(ctx Context) interface{} {
@ -812,13 +819,13 @@ yydefault:
} }
case 40: case 40:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:198 //line expressions.y:205
{ {
yyVAL.f = makeContainsExpr(yyDollar[1].f, yyDollar[3].f) yyVAL.f = makeContainsExpr(yyDollar[1].f, yyDollar[3].f)
} }
case 42: case 42:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:203 //line expressions.y:210
{ {
fa, fb := yyDollar[1].f, yyDollar[3].f fa, fb := yyDollar[1].f, yyDollar[3].f
yyVAL.f = func(ctx Context) interface{} { yyVAL.f = func(ctx Context) interface{} {
@ -827,7 +834,7 @@ yydefault:
} }
case 43: case 43:
yyDollar = yyS[yypt-3 : yypt+1] yyDollar = yyS[yypt-3 : yypt+1]
//line expressions.y:209 //line expressions.y:216
{ {
fa, fb := yyDollar[1].f, yyDollar[3].f fa, fb := yyDollar[1].f, yyDollar[3].f
yyVAL.f = func(ctx Context) interface{} { yyVAL.f = func(ctx Context) interface{} {

View File

@ -69,6 +69,7 @@ func loopTagParser(node render.BlockNode) (func(io.Writer, render.Context) error
return nil, err return nil, err
} }
loop := stmt.Loop loop := stmt.Loop
dec := makeLoopDecorator(node.Name, loop)
return func(w io.Writer, ctx render.Context) error { return func(w io.Writer, ctx render.Context) error {
val, err := ctx.Evaluate(loop.Expr) val, err := ctx.Evaluate(loop.Expr)
if err != nil { if err != nil {
@ -98,7 +99,9 @@ func loopTagParser(node render.BlockNode) (func(io.Writer, render.Context) error
"length": len, "length": len,
".cycles": cycleMap, ".cycles": cycleMap,
}) })
dec.before(w, i)
err := ctx.RenderChildren(w) err := ctx.RenderChildren(w)
dec.after(w, i, len)
switch { switch {
case err == nil: case err == nil:
// fall through // fall through
@ -114,6 +117,50 @@ func loopTagParser(node render.BlockNode) (func(io.Writer, render.Context) error
}, nil }, nil
} }
func makeLoopDecorator(tagName string, loop expressions.Loop) loopDecorator {
if tagName == "tablerow" {
return tableRowDecorator(loop.Cols)
}
return nullLoopDecorator{}
}
type loopDecorator interface {
before(io.Writer, int)
after(io.Writer, int, int)
}
type nullLoopDecorator struct{}
func (d nullLoopDecorator) before(io.Writer, int) {}
func (d nullLoopDecorator) after(io.Writer, int, int) {}
type tableRowDecorator int
func (c tableRowDecorator) before(w io.Writer, i int) {
cols := int(c)
row, col := i/cols, i%cols
if col == 0 {
if _, err := fmt.Fprintf(w, `<tr class="row%d">`, row+1); err != nil {
panic(err)
}
}
if _, err := fmt.Fprintf(w, `<td class="col%d">`, col+1); err != nil {
panic(err)
}
}
func (c tableRowDecorator) after(w io.Writer, i, len int) {
cols := int(c)
if _, err := io.WriteString(w, `</td>`); err != nil {
panic(err)
}
if (i+1)%cols == 0 || i+1 == len {
if _, err := io.WriteString(w, `</tr>`); err != nil {
panic(err)
}
}
}
func applyLoopModifiers(loop expressions.Loop, iter iterable) iterable { func applyLoopModifiers(loop expressions.Loop, iter iterable) iterable {
if loop.Reversed { if loop.Reversed {
iter = reverseWrapper{iter} iter = reverseWrapper{iter}

View File

@ -4,6 +4,8 @@ import (
"bytes" "bytes"
"fmt" "fmt"
"io/ioutil" "io/ioutil"
"regexp"
"strings"
"testing" "testing"
"github.com/osteele/liquid/parser" "github.com/osteele/liquid/parser"
@ -68,6 +70,20 @@ var iterationTests = []struct{ in, expected string }{
// range // range
{`{% for i in (3 .. 5) %}{{i}}.{% endfor %}`, "3.4.5."}, {`{% for i in (3 .. 5) %}{{i}}.{% endfor %}`, "3.4.5."},
{`{% for i in (3..5) %}{{i}}.{% endfor %}`, "3.4.5."}, {`{% for i in (3..5) %}{{i}}.{% endfor %}`, "3.4.5."},
// tablerow
{`{% tablerow product in products %}{{ product }}{% endtablerow %}`,
`<tr class="row1"><td class="col1">Cool Shirt</td>
<td class="col2">Alien Poster</td>
<td class="col3">Batman Poster</td>
<td class="col4">Bullseye Shirt</td>
<td class="col5">Another Classic Vinyl</td>
<td class="col6">Awesome Jeans</td></tr>`},
{`{% tablerow product in products cols:2 %}{{ product }}{% endtablerow %}`,
`<tr class="row1"><td class="col1">Cool Shirt</td><td class="col2">Alien Poster</td></tr>
<tr class="row2"><td class="col1">Batman Poster</td><td class="col2">Bullseye Shirt</td></tr>
<tr class="row3"><td class="col1">Another Classic Vinyl</td><td class="col2">Awesome Jeans</td></tr>`},
} }
var iterationSyntaxErrorTests = []struct{ in, expected string }{ var iterationSyntaxErrorTests = []struct{ in, expected string }{
@ -85,9 +101,12 @@ var iterationErrorTests = []struct{ in, expected string }{
var iterationTestBindings = map[string]interface{}{ var iterationTestBindings = map[string]interface{}{
"array": []string{"first", "second", "third"}, "array": []string{"first", "second", "third"},
"hash": map[string]interface{}{"a": 1}, "hash": map[string]interface{}{"a": 1},
"products": []string{
"Cool Shirt", "Alien Poster", "Batman Poster", "Bullseye Shirt", "Another Classic Vinyl", "Awesome Jeans",
},
} }
func TestLoopTag(t *testing.T) { func TestIterationTags(t *testing.T) {
config := render.NewConfig() config := render.NewConfig()
AddStandardTags(config) AddStandardTags(config)
for i, test := range iterationTests { for i, test := range iterationTests {
@ -97,7 +116,13 @@ func TestLoopTag(t *testing.T) {
buf := new(bytes.Buffer) buf := new(bytes.Buffer)
err = render.Render(root, buf, iterationTestBindings, config) err = render.Render(root, buf, iterationTestBindings, config)
require.NoErrorf(t, err, test.in) require.NoErrorf(t, err, test.in)
require.Equalf(t, test.expected, buf.String(), test.in) actual := buf.String()
if strings.Contains(test.in, "{% tablerow") {
replaceWS := regexp.MustCompile(`\n\s*`).ReplaceAllString
actual = replaceWS(actual, "")
test.expected = replaceWS(test.expected, "")
}
require.Equalf(t, test.expected, actual, test.in)
}) })
} }
} }

View File

@ -25,7 +25,7 @@ func AddStandardTags(c render.Config) {
c.AddBlock("for").Compiler(loopTagParser) c.AddBlock("for").Compiler(loopTagParser)
c.AddBlock("if").Clause("else").Clause("elsif").Compiler(ifTagParser(true)) c.AddBlock("if").Clause("else").Clause("elsif").Compiler(ifTagParser(true))
c.AddBlock("raw") c.AddBlock("raw")
c.AddBlock("tablerow") c.AddBlock("tablerow").Compiler(loopTagParser)
c.AddBlock("unless").Compiler(ifTagParser(false)) c.AddBlock("unless").Compiler(ifTagParser(false))
} }