2013-02-11 20 views
6

Sto cercando di creare un binding personalizzato per i popover di Twitter boostrap che fa riferimento a un modello ma ho problemi con la parte di bind del contenuto all'interno del popover una volta che è stato creato.Knockout Twitter Bootstrap Popover Binding

Ho visto questa domanda prima, ma mi sento come se fossero per lo più piuttosto disordinati e sono abbastanza vicino a una soluzione riutilizzabile che utilizza i modelli come voglio.

http://jsfiddle.net/billpull/Edptd/

// Bind Twitter Popover 
ko.bindingHandlers.popover = { 
    init: function(element, valueAccessor, allBindingsAccessor, viewModel, bindingContext) { 
     var tmplId = ko.utils.unwrapObservable(valueAccessor()); 
     var tmplHtml = $('#' + tmplId).html(); 
     var uuid = guid(); 
     var domId = "ko-bs-popover-" + uuid; 
     var tmplDom = $('<div/>', { 
      "class" : "ko-popover", 
      "id" : domId 
     }).html(tmplHtml); 

     options = { 
      content: tmplDom[0].outerHTML 
     }; 

     var popoverOptions = ko.utils.extend(ko.bindingHandlers.popover.options, options); 

     console.log($(element)); 
     console.log(element); 

     $(element).bind('click', function() { 
      $(this).popover(popoverOptions).popover('toggle'); 
      ko.applyBindings(bindingContext, document.getElementById(domId)); 
     }); 
    }, 
    options: { 
     placement: "right", 
     title: "", 
     html: true, 
     content: "", 
     trigger: "manual" 
    } 
}; 

=== EDIT

codice aggiornato in base alla risposta al di sotto che permette di farlo senza i withProperties in più vincolanti

// Bind Twitter Popover 
ko.bindingHandlers.popover = { 
    init: function(element, valueAccessor, allBindingsAccessor, viewModel, bindingContext) { 
     // read popover options 
     var popoverBindingValues = ko.utils.unwrapObservable(valueAccessor()); 

     // set popover template id 
     var tmplId = popoverBindingValues.template; 

     // set popover trigger 
     var trigger = popoverBindingValues.trigger; 

     // get template html 
     var tmplHtml = $('#' + tmplId).html(); 

     // create unique identifier to bind to 
     var uuid = guid(); 
     var domId = "ko-bs-popover-" + uuid; 

     // create correct binding context 
     var childBindingContext = bindingContext.createChildContext(viewModel); 

     // create DOM object to use for popover content 
     var tmplDom = $('<div/>', { 
      "class" : "ko-popover", 
      "id" : domId 
     }).html(tmplHtml); 

     // set content options 
     options = { 
      content: tmplDom[0].outerHTML 
     }; 

     // Need to copy this, otherwise all the popups end up with the value of the last item 
     var popoverOptions = $.extend({}, ko.bindingHandlers.popover.options); 
     popoverOptions.content = options.content; 

     // bind popover to element click 
     $(element).bind(trigger, function() { 
      $(this).popover(popoverOptions).popover('toggle'); 

      // if the popover is visible bind the view model to our dom ID 
      if($('#' + domId).is(':visible')){ 
       ko.applyBindingsToDescendants(childBindingContext, $('#' + domId)[0]); 
      } 
     }); 

     return { controlsDescendantBindings: true }; 
    }, 
    options: { 
     placement: "right", 
     title: "", 
     html: true, 
     content: "", 
     trigger: "manual" 
    } 
}; 
+0

Qual è in realtà il problema qui? Non hai descritto quale sia il problema. –

+0

"Ho problemi con la parte di bind del contenuto all'interno del popover una volta che è stata creata." – BillPull

+0

Sono sicuro che dev'essere possibile, anche se non semplice, ma non ho il tempo di guardare il momento. Darò un'occhiata stasera –

risposta

6

È necessario utilizzare il mio vecchio amico, custom bindings.

ko.bindingHandlers.withProperties = { 
    init: function(element, valueAccessor, allBindingsAccessor, viewModel, bindingContext) { 
     // Make a modified binding context, with a extra properties, and apply it to descendant elements 
     var newProperties = valueAccessor(), 
      innerBindingContext = bindingContext.extend(newProperties); 
     ko.applyBindingsToDescendants(innerBindingContext, element); 

     // Also tell KO *not* to bind the descendants itself, otherwise they will be bound twice 
     return { controlsDescendantBindings: true }; 
    } 
}; 

È quindi necessario aggiungere un attributo data-bind per il codice HTML si sta generando:

Ho messo insieme un jsFiddle che mostra questo. C'erano un paio di trucchi, dovevo copiare le opzioni del popover per ogni popover, altrimenti finivano con l'ultimo set di valori.

var popoverOptions = $.extend({}, ko.bindingHandlers.popover.options); 
    popoverOptions.content = options.content; 

E ho avuto anche applicare vincolante alla comparsa solo se è visibile, altrimenti sembra tentare di legare a tutta pagina.

$(element).bind('click', function() { 
      $(this).popover(popoverOptions).popover('toggle'); 
      // If you apply this when the popup isn't visible, I think that it tries to bind to thewhole pageand throws an error 
      if($('#' + domId).is(':visible')) 
      { 
       ko.applyBindings(viewModel, $('#' + domId)[0]); 
      } 
     }); 

Questo sembra anche essere a 2 vie, in quanto è possibile modificare i valori nel popup e aggiorna gli elementi non popup, ma non voglio mentire, non mi aspettavo che accadesse !

1

mi sono adattato un'altra risposta qui: https://stackoverflow.com/a/16876013/1061602

Questo funziona molto meglio per me, soprattutto per un semplice popover.

ko.bindingHandlers.popover = { 
    init: function (element, valueAccessor) { 
     var local = ko.utils.unwrapObservable(valueAccessor()), 
      options = {}; 

     ko.utils.extend(options, ko.bindingHandlers.popover.options); 
     ko.utils.extend(options, local); 

     $(element).popover(options); 

     ko.utils.domNodeDisposal.addDisposeCallback(element, function() { 
      $(element).popover("destroy"); 
     }); 
    }, 
    options: { 
     placement: "top" 
    } 
}; 

Poi il legame è:

<span data-bind="popover: { content: mySimpleTextContent }"></span> 

È possibile ignorare le altre opzioni, ovviamente.

+1

Bello. Da quando ho postato questa domanda ho creato una libreria di estensioni bootstrap-knockout. Sto cercando qualsiasi feedback disponibile https://github.com/billpull/knockout-bootstrap – BillPull

2

Ecco un'altra versione del binding del popover Knockout che utilizza un modello html definito all'interno del documento.

Partenza questo violino: https://jsfiddle.net/2cpcgz3o/

(function() { 
 
    var templateEngine = new ko.nativeTemplateEngine(); 
 

 
    ko.bindingHandlers.customPopover = { 
 
     init: function (element, valueAccessor, allBindings, viewModel, bindingContext) { 
 
      var placement = allBindings.get("placement") || "top", 
 
       trigger = allBindings.get("trigger") || "click", 
 
       templateName = allBindings.get("customPopover") || null, 
 
       $element = $(element); 
 

 
      $element.popover({ placement: placement, trigger: trigger, html: true, content: "&nbsp;" }); 
 

 
      $element.on("inserted.bs.popover", function() { 
 
       var container = $element.next().find(".popover-content")[0]; 
 
       if (templateName) { 
 
        ko.renderTemplate(templateName, viewModel, { templateEngine: templateEngine }, container); 
 
       } 
 
       else { 
 
        container.innerHTML = $element.attr("data-content"); 
 
       } 
 
      }); 
 

 
      ko.utils.domNodeDisposal.addDisposeCallback(element, function() { 
 
       $element.popover("destroy"); 
 
      }); 
 
     } 
 
    }; 
 
})(); 
 

 
var model = { 
 
    linkText: "Click me!", 
 
    innerText: "Some fancy text" 
 
}; 
 

 
ko.applyBindings(model);
<link href="https://cdn.jsdelivr.net/bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet"> 
 
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.9.1/jquery.min.js"></script> 
 
<script src="https://cdn.jsdelivr.net/bootstrap/3.3.7/js/bootstrap.min.js"></script> 
 
<script src="https://cdnjs.cloudflare.com/ajax/libs/knockout/3.2.0/knockout-min.js"></script> 
 

 
<a data-bind="text: linkText, customPopover: 'popover-template', trigger: 'focus', placement: 'bottom'" tabindex="0" role="button"></a> 
 

 
<script type="text/html" id="popover-template"> 
 
    <span data-bind="text: innerText"></span> 
 
</script>

0

leggermente modificata dodbrian s' esempio. Il contenuto è legato a un osservabile.

https://jsfiddle.net/SergeyZhilyakov/0zxamcj6/14/

var model = { 
 
    linkText: "Hover me!", 
 
    innerText: ko.observable("Please, wait...") 
 
}; 
 

 
ko.bindingHandlers.popover = { 
 
    init: function(element, valueAccessor, allBindings, viewModel, bindingContext) { 
 
    var $element = $(element); 
 
    var placement = allBindings.get("placement") || "top"; 
 
    var trigger = allBindings.get("trigger") || "hover"; 
 
    var content = allBindings.get("popover"); 
 

 
    $element.popover({ 
 
     placement: placement, 
 
     trigger: trigger, 
 
     content: content() 
 
    }); 
 

 
    var popover = $element.data("bs.popover"); 
 
    content.subscribe(function(newValue) { 
 
     popover.options.content = newValue; 
 
     popover.setContent(); 
 
     popover.$tip.addClass(popover.options.placement); 
 
    }); 
 

 
    ko.utils.domNodeDisposal.addDisposeCallback(element, function() { 
 
     $element.popover("destroy"); 
 
    }); 
 
    } 
 
}; 
 

 
ko.applyBindings(model); 
 

 
setTimeout(function() { 
 
    model.innerText("Done!"); 
 
}, 3000);
body { 
 
    padding: 20px; 
 
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script> 
 
<link href="https://cdn.jsdelivr.net/bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet"/> 
 
<script src="https://cdn.jsdelivr.net/bootstrap/3.3.7/js/bootstrap.min.js"></script> 
 
<script src="https://cdnjs.cloudflare.com/ajax/libs/knockout/3.4.2/knockout-min.js"></script> 
 

 

 
<button type="button" class="btn btn-default" data-bind="text: linkText, popover: innerText, placement: 'bottom'">Comment</button>

Problemi correlati