I have a custom binding handler that I am binding to a complex object in my view model.
The binding handler works correctly and the update
function is called when any of the observable's properties update. However, the update
function is called for every updated property, leading to odd behaviour since I am relying on the entire object to be available and up to date.
I understand why this is happening, as each property is causing an update to be called, and I think I know how to prevent this - by using the deferred updates functionality of Knockout.
However, I am unable to find how to enable deferred updates just for the observable in my custom binding. I do not want to enable it application wide as I am writing the binding as a library function.
I have tried many different methods including:
- trying to extend the binding handler itself;
- extending the
init
function; - extending the
valueAccessor
; - replacing the
valueAccessor
with a new observable withdeferred
applied; - creating a computed observable and rebinding the element;
All of which have not worked.
I have not found any other custom binding handler that comes remotely close to this sort of function and have been trying to piece it together from other functions.
My binding code itself is relatively simple, I am taking the bound object and simply splitting out the parameters and passing them to a Code Mirror instance.
ko.bindingHandlers.editor = {
init: function(element, valueAccessor, allBindingsAccessor) {
var observableValue = ko.utils.unwrap(valueAccessor());
initEditor(element, observableValue, allBindingsAccessor);
},
update: function(element, valueAccessor, allBindingsAccessor) {
var observableValue = ko.unwrap(valueAccessor());
createEditor(codeEditorDiv, observableValue);
resize();
updateEditor(element, observableValue, allBindingsAccessor);
}
};
And my HTML code is:
<div id="editor" data-bind="editor: EditorVM"></div>
I am using Dotnetify for the ViewModel so it is a reasonable complex C# class, but suffice it to say that the binding is working and updating, but I need it to only call 'update' once all properties have been updated.
It's unfortunate you haven't shown what
initEditor
,createEditor
andupdateEditor
do with theobservableValue
, because that's probably where you should be extending your observables.The
init
andupdate
methods of a binding create computed dependencies, meaning that any observable that is unwrapped in the call stack starting frominit
will cause theupdate
method to be called.In an abstract example:
Note that
update
is called:There are several ways of solving the issue, of which the simplest is to create your own
computed
inside theinit
method of your binding and extend it to bedeferred
.