83 votes

conditionnel sur le dernier élément du tableau à l'aide du modèle handlebars.js

Je suis en tirant parti de handlebars.js pour mon moteur de template et je suis à la recherche de faire un sursis d'affichage de segment que si c'est le dernier élément dans le tableau contenu dans les modèles de configuration de l'objet.

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

J'ai déjà tiré à l'aide pour faire quelques égalité/plus de/moins de comparaisons et ont réussi à identifier l'élément initial de cette façon, mais n'ont eu aucune chance d'accéder à ma cible longueur du tableau.

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}}"

Quelqu'un sait-il un raccourci, une approche différente, et certains guidon de bonté qui va m'empêcher d'avoir à déchirer dans l'handlebars.js moteur pour déterminer le meilleur cours?

165voto

Jacob van Lingen Points 933

Depuis Handlebars 1.1.0, first et last sont devenus natifs pour chaque aide. Voir billet n ° 483 .

L'utilisation est comme la classe d'assistance d'Eberanov :

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

120voto

Matt Brennan Points 1016

Un assistant rapide que j'ai écrit pour faire le tour est:

 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('');
});
 

Ensuite, vous pouvez écrire:

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

26voto

Yong Qu Points 191

Si vous essayez simplement de gérer le premier élément du tableau, cela peut aider

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

@index est fourni par chaque assistant et pour le premier élément, il est égal à zéro et peut donc être traité par l'assistant if.

1voto

techie.brandon Points 311

Solution:

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

En tirant parti des helpers par le blog suivant, et gist...

https://gist.github.com/2889952

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

Depuis Kevin pense que je devrais nourrir à la main lui au lieu de suivre les liens vers les auteurs:

// {{#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);
    }

});

Avis de l'indice de départ est correctement 1.

0voto

ebaranov Points 323

J'ai apporté quelques améliorations à l'aide de Matt Brennan , vous pouvez utiliser cette aide avec Objects ou Arrays, cette solution nécessite une bibliothèque 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('');
});
 

Usage:

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

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X