Commit 233db94c authored by Pascal Hartig's avatar Pascal Hartig

Knockout.js: cancel on Escape

parent 48a00e7b
This source diff could not be displayed because it is too large. You can view the blob instead.
...@@ -22,7 +22,7 @@ ...@@ -22,7 +22,7 @@
<label data-bind="text: title, event: { dblclick: $root.editItem }"></label> <label data-bind="text: title, event: { dblclick: $root.editItem }"></label>
<button class="destroy" data-bind="click: $root.remove"></button> <button class="destroy" data-bind="click: $root.remove"></button>
</div> </div>
<input class="edit" data-bind="value: title, valueUpdate: 'afterkeydown', enterKey: $root.stopEditing, selectAndFocus: editing, event: { blur: $root.stopEditing }"> <input class="edit" data-bind="value: title, valueUpdate: 'afterkeydown', enterKey: $root.saveEditing, escapeKey: $root.cancelEditing, selectAndFocus: editing, event: { blur: $root.stopEditing }">
</li> </li>
</ul> </ul>
</section> </section>
......
...@@ -3,15 +3,18 @@ ...@@ -3,15 +3,18 @@
'use strict'; 'use strict';
var ENTER_KEY = 13; var ENTER_KEY = 13;
var ESCAPE_KEY = 27;
// a custom binding to handle the enter key (could go in a separate library) // A factory function we can use to create binding handlers for specific
ko.bindingHandlers.enterKey = { // keycodes.
function keyhandlerBindingFactory(keyCode) {
return {
init: function (element, valueAccessor, allBindingsAccessor, data, bindingContext) { init: function (element, valueAccessor, allBindingsAccessor, data, bindingContext) {
var wrappedHandler, newValueAccessor; var wrappedHandler, newValueAccessor;
// wrap the handler with a check for the enter key // wrap the handler with a check for the enter key
wrappedHandler = function (data, event) { wrappedHandler = function (data, event) {
if (event.keyCode === ENTER_KEY) { if (event.keyCode === keyCode) {
valueAccessor().call(this, data, event); valueAccessor().call(this, data, event);
} }
}; };
...@@ -27,6 +30,13 @@ ...@@ -27,6 +30,13 @@
ko.bindingHandlers.event.init(element, newValueAccessor, allBindingsAccessor, data, bindingContext); ko.bindingHandlers.event.init(element, newValueAccessor, allBindingsAccessor, data, bindingContext);
} }
}; };
}
// a custom binding to handle the enter key
ko.bindingHandlers.enterKey = keyhandlerBindingFactory(ENTER_KEY);
// another custom binding, this time to handle the escape key
ko.bindingHandlers.escapeKey = keyhandlerBindingFactory(ESCAPE_KEY);
// wrapper to hasFocus that also selects text and applies focus async // wrapper to hasFocus that also selects text and applies focus async
ko.bindingHandlers.selectAndFocus = { ko.bindingHandlers.selectAndFocus = {
...@@ -103,10 +113,11 @@ ...@@ -103,10 +113,11 @@
// edit an item // edit an item
this.editItem = function (item) { this.editItem = function (item) {
item.editing(true); item.editing(true);
item.previousTitle = item.title();
}.bind(this); }.bind(this);
// stop editing an item. Remove the item, if it is now empty // stop editing an item. Remove the item, if it is now empty
this.stopEditing = function (item) { this.saveEditing = function (item) {
item.editing(false); item.editing(false);
var title = item.title(); var title = item.title();
...@@ -124,6 +135,12 @@ ...@@ -124,6 +135,12 @@
} }
}.bind(this); }.bind(this);
// cancel editing an item and revert to the previous content
this.cancelEditing = function (item) {
item.editing(false);
item.title(item.previousTitle);
}.bind(this);
// count of all completed todos // count of all completed todos
this.completedCount = ko.computed(function () { this.completedCount = ko.computed(function () {
return this.todos().filter(function (todo) { return this.todos().filter(function (todo) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment