使用handlebars.js模板以数组中的最后一项为条件


74

我将handlebars.js用于模板引擎,并希望仅当条件段是模板配置对象中包含的数组的最后一项时才显示条件段。

{
  columns: [{<obj>},{<obj>},{<obj>},{<obj>},{<obj>}]
}

我已经拉了一个助手来做一些相等/更大/小于的比较,并且已经成功地通过这种方式识别了初始项目,但是没有运气来访问目标数组的长度。

Handlebars.registerHelper('compare', function(lvalue, rvalue, options) {...})

"{{#each_with_index columns}}"+
"<div class='{{#equal index 0}} first{{/equal}}{{#equal index ../columns.length()}} last{{/equal}}'>"+
"</div>"+
"{{/each_with_index}}"

有谁知道捷径,不同的方法以及一些车把的优点,这些使我不必再进入handlebars.js引擎来确定最佳路线?


我建议使用underscoreJS的模板库,它比handlebars.js更具意义并且效率更高
奥斯汀

1
嗯,我和下一个家伙一样下划线<3,但是您显然从来没有认真看过把手。Underscore的模板系统无法(甚至永远也不会尝试)完成Handlebars的一半工作。
machineghost

6
我没有问其他模板引擎。我已经做出了选择,很多迭代之前。许多技术人员都可以为该问题提供出色的解决方案,但是a,我不与这些技术人员合作,因此它们对我现在毫无用处。
techie.brandon

如果您只是在说“显示”,可以用CSS处理吗?我意识到这已经是个警察了,但是如果数据已经在客户端上,style="display: none"
那就

实际上,使用css是行不通的,因为我使用它来表示要使用哪个类。如果在行的第一个(发现是特殊的)则得到一个特殊的类,如果在行的最后(并且发现是特殊的),则得到一个特殊的类(一些附加变量告诉我特殊类是什么,所以不是此表的标准)。
techie.brandon 2012年

Answers:


113

从Handlebars v1.1.0开始,您现在可以在每个帮助器中使用@firstand@last布尔值来解决此问题:

{{#each foo}}
    <div class='{{#if @first}}first{{/if}}
                {{#if @last}} last{{/if}}'>
      {{@key}} - {{@index}}
    </div>
{{/each}}

我写来做这个技巧的快速帮助是:

Handlebars.registerHelper("foreach",function(arr,options) {
    if(options.inverse && !arr.length)
        return options.inverse(this);

    return arr.map(function(item,index) {
        item.$index = index;
        item.$first = index === 0;
        item.$last  = index === arr.length-1;
        return options.fn(item);
    }).join('');
});

然后您可以编写:

{{#foreach foo}}
    <div class='{{#if $first}} first{{/if}}{{#if $last}} last{{/if}}'></div>
{{/foreach}}

好工作!尚未验证它是否有效,但我看起来应该可以,如果可以的话,它将在周末重新访问并重新分配。谢谢!
techie.brandon 2012年

我刚刚使用了上面的帮助程序,完全按照广告中的说明工作...谢谢!gist.github.com/zeroasterisk/5360895
zeroasterisk

5
好的答案,但我确实注意到一个挂断。如果数组中的项目是字符串或与此相关的任何其他原语,则将无法为其添加属性。我的解决方法是创建一个具有item值的新字符串,从技术上讲它是一个对象,并将属性附加到该字符串,如以下要点:https
//gist.github.com/jordancooperman/5440241

3
为什么这个答案时接受的答案first,并last为本土更有效?
Dan Dascalescu 2014年

1
惯性。这个答案是两年前写下并被接受的。带有原生结构的Handlebars版本是一年多以前发布的。我猜OP已经进行了。
卡拉布赖特韦尔2014年

159

从Handlebars 1.1.0开始,第一个和最后一个已成为每个帮助器的本机。参见门票#483

用法类似于Eberanov的帮助程序类:

{{#each foo}}
    <div class='{{#if @first}}first{{/if}}{{#if @last}} last{{/if}}'>{{@key}} - {{@index}}</div>
{{/each}}

28
{{#unless @last}},{{/ unless}} FTW!
Crisboot,2015年

26

如果您只是尝试处理数组的第一项,这可能会有所帮助

{{#each data-source}}{{#if @index}},{{/if}}"{{this}}"{{/each}}

@index由每个助手提供,对于第一个项目,@ index等于零,因此可以由if助手处理。


7
如果发现并希望将其与Meteor的Handlebars实现一起使用,则请注意,它将无法正常工作。@打破了一切。
Mike Turley

1

解:

<div class='{{#compare index 1}} first{{/compare}}{{#compare index total}} last{{/compare}}'></div>

利用以下博客和要点中的帮助者...

https://gist.github.com/2889952

http://doginthehat.com.au/2012/02/comparison-block-helper-for-handlebars-templates/

// {{#each_with_index records}}
//  <li class="legend_item{{index}}"><span></span>{{Name}}</li>
// {{/each_with_index}}

Handlebars.registerHelper("each_with_index", function(array, fn) {
  var total = array.length;
  var buffer = "";

  //Better performance: http://jsperf.com/for-vs-foreach/2
  for (var i = 0, j = total; i < j; i++) {
    var item = array[i];

    // stick an index property onto the item, starting with 1, may make configurable later
    item.index = i+1;
    item.total = total;
    // show the inside of the block
    buffer += fn(item);
  }

  // return the finished buffer
  return buffer;

});

Handlebars.registerHelper('compare', function(lvalue, rvalue, options) {

    if (arguments.length < 3)
        throw new Error("Handlerbars Helper 'compare' needs 2 parameters");

    operator = options.hash.operator || "==";

    var operators = {
        '==':       function(l,r) { return l == r; },
        '===':      function(l,r) { return l === r; },
        '!=':       function(l,r) { return l != r; },
        '<':        function(l,r) { return l < r; },
        '>':        function(l,r) { return l > r; },
        '<=':       function(l,r) { return l <= r; },
        '>=':       function(l,r) { return l >= r; },
        'typeof':   function(l,r) { return typeof l == r; }
    }

    if (!operators[operator])
        throw new Error("Handlerbars Helper 'compare' doesn't know the operator "+operator);

    var result = operators[operator](lvalue,rvalue);

    if( result ) {
        return options.fn(this);
    } else {
        return options.inverse(this);
    }

});

请注意起始索引正确1。


这是答案是不完整的定义comparetotal。另外,0是第一个索引,而不是1
凯文边框

0

我对Matt Brennan的辅助程序进行了一些改进,您可以将此辅助程序与Objects或Arrays一起使用,此解决方案需要Underscore库:

Handlebars.registerHelper("foreach", function(context, options) {
  options = _.clone(options);
  options.data = _.extend({}, options.hash, options.data);

  if (options.inverse && !_.size(context)) {
    return options.inverse(this);
  }

  return _.map(context, function(item, index, list) {
    var intIndex = _.indexOf(_.values(list), item);

    options.data.key = index;
    options.data.index = intIndex;
    options.data.isFirst = intIndex === 0;
    options.data.isLast = intIndex === _.size(list) - 1;

    return options.fn(item, options);
  }).join('');
});

用法:

{{#foreach foo}}
    <div class='{{#if @first}}first{{/if}}{{#if @last}} last{{/if}}'>{{@key}} - {{@index}}</div>
{{/foreach}}

0

仅供参考:如果您的手把<1.1.0(像我一样)陷于困境,则可以尝试以下解决方法:

isLast在要迭代的对象上定义一个属性,然后像使用它一样

{{#each objectsInList}}"{{property}}": "{{value}}"{{#unless isLast}},{{/unless}}{{/each}}

建立一个JSON对象。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.