mirror of
https://github.com/danog/liquid.git
synced 2024-11-30 07:28:56 +01:00
Implement tablerow
This commit is contained in:
parent
a2a4a1a5ec
commit
cd234476f7
@ -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
|
||||||
|
|
||||||
|
@ -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 {
|
||||||
|
@ -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
|
||||||
|
@ -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{} {
|
||||||
|
@ -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}
|
||||||
|
@ -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)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user