Я уже писал как можно применить кастомный биндинг, чтоб прицепить jquery datepicker. Сегодня я покажу еще один пример.
Иногда возникает необходимость ограничить пользовательский ввод по какому-то критерию, например разрешить вводить только цифры или только текст без цифр, если это имя или город.
В данной статье я покажу как ограничить ввод только цифрами, но в дальнейшем можно будет развить эту тему и фильтровать на основе какого-нибудь дополнительного критерия.
Решение по фильтрации я не изобретал и взял первое попавшееся на stackoverflow. Его мы и будем использовать.
Начнем с написания кастомного биндинга.
Ну и рабочий пример.
Тут ничего военного не появилось, все тот же init и update. В инит находится код, который я полностью скопировал с ответа на stackoverflow.
Если вы посмотрите на html, то увидите там строчку:
data-bind="value: Age, inputFilter: 'Numbers'
Это значит что мы хотим связать наш инпут со значением поля Age и добавляем наш кастомный биндинг с параметром Numbers.
Когда вызывается метод init мы получаем значение, которое передали в inputFilter, в данном случае это будет 'Numbers'. Это значение получается в данной строке
var filter = ko.utils.unwrapObservable(valueAccessor());
ну а дальше все ясно вроде, если это Number цепляем обработчик.
В данном случае, нам не нужен update, нам достаточно прицепить обработчик keydown один раз в init и все, потом он будет жить своей жизнью.
Помоему получилось красиво!
Что можно сделать лучше:
Иногда возникает необходимость ограничить пользовательский ввод по какому-то критерию, например разрешить вводить только цифры или только текст без цифр, если это имя или город.
В данной статье я покажу как ограничить ввод только цифрами, но в дальнейшем можно будет развить эту тему и фильтровать на основе какого-нибудь дополнительного критерия.
Решение по фильтрации я не изобретал и взял первое попавшееся на stackoverflow. Его мы и будем использовать.
Начнем с написания кастомного биндинга.
ko.bindingHandlers.inputFilter = {
init: function (element, valueAccessor, allBindingsAccessor, viewModel) {
var filter = ko.utils.unwrapObservable(valueAccessor());
if (filter == 'Numbers') {
$(element).keydown(function (event) {
// Allow: backspace, delete, tab, escape, and enter
if (event.keyCode == 46 || event.keyCode == 8 || event.keyCode == 9 || event.keyCode == 27 || event.keyCode == 13 ||
// Allow: Ctrl+A
(event.keyCode == 65 && event.ctrlKey === true) ||
// Allow: home, end, left, right
(event.keyCode >= 35 && event.keyCode <= 39)) {
// let it happen, don't do anything
return;
}
else {
// Ensure that it is a number and stop the keypress
if (event.shiftKey || (event.keyCode < 48 || event.keyCode > 57) && (event.keyCode < 96 || event.keyCode > 105)) {
event.preventDefault();
}
}
});
}
},
update: function (element, valueAccessor, allBindingsAccessor, viewModel) {
// we don't need update.
}
};
function CarViewModel (){
this.Age = ko.observable(10);
};
var car = new CarViewModel();
ko.applyBindings(car);
Ну и рабочий пример.
Тут ничего военного не появилось, все тот же init и update. В инит находится код, который я полностью скопировал с ответа на stackoverflow.
Если вы посмотрите на html, то увидите там строчку:
data-bind="value: Age, inputFilter: 'Numbers'
Это значит что мы хотим связать наш инпут со значением поля Age и добавляем наш кастомный биндинг с параметром Numbers.
Когда вызывается метод init мы получаем значение, которое передали в inputFilter, в данном случае это будет 'Numbers'. Это значение получается в данной строке
var filter = ko.utils.unwrapObservable(valueAccessor());
ну а дальше все ясно вроде, если это Number цепляем обработчик.
В данном случае, нам не нужен update, нам достаточно прицепить обработчик keydown один раз в init и все, потом он будет жить своей жизнью.
Помоему получилось красиво!
Что можно сделать лучше:
- Код, который у меня находится в init можно вынести в плагин jquery или просто найти готовый плагин, ссылки на них есть в тоже же ответе.
- Можно ввести в дело еще один параметр например Characters и фитровать только по буквам.
- Можно пойти дальше и передавать допустимые символы, например !"№;%@ или еще лучше принимать шаблон. Я думаю изобретать парсер шаблона не придется, есть jquery плагины, досточно будет просто принять шаблон через custom binding.

