I use KnockoutJS in my projects, but want to learn AngularJS as it has a lot of tasty features that Knockout doesn't have. So I'm interested in rewriting some of my code using Angular. But I don't understand how to do some simple things that I use in Knockout. For example, Knockout has a feature of puted observables. It's cool! I've already found that I can use a simple function instead. But Knockout provides "write" function to a puted observables, like:
var first_name = ko.observable('John'),
last_name = ko.observable('Smith'),
full_name = koputed({
read: function(){
return first_name() + ' ' + last_name();
},
write: function(new_value){
var matches = new_value.match(/^(\w+)\s+(\w+)/);
first_name(matches[1]);
last_name(matches[2]);
}
});
This code on JSFiddle: /
This code allows me to update the "first_name" and "last_name" observables when I change the "full_name" one. How it can be done using AngularJS? A function with an argument being checked for existence? Something like this?
first_name = 'John';
last_name = 'Smith';
full_name = function(value){
if (typeof value != 'undefined')
{
// do the same as in the Knockout's write function
}
else
{
// do the same as in the Knockout's read function
}
}
What is the best practice?
I use KnockoutJS in my projects, but want to learn AngularJS as it has a lot of tasty features that Knockout doesn't have. So I'm interested in rewriting some of my code using Angular. But I don't understand how to do some simple things that I use in Knockout. For example, Knockout has a feature of puted observables. It's cool! I've already found that I can use a simple function instead. But Knockout provides "write" function to a puted observables, like:
var first_name = ko.observable('John'),
last_name = ko.observable('Smith'),
full_name = ko.puted({
read: function(){
return first_name() + ' ' + last_name();
},
write: function(new_value){
var matches = new_value.match(/^(\w+)\s+(\w+)/);
first_name(matches[1]);
last_name(matches[2]);
}
});
This code on JSFiddle: http://jsfiddle/Girafa/QNebV/1/
This code allows me to update the "first_name" and "last_name" observables when I change the "full_name" one. How it can be done using AngularJS? A function with an argument being checked for existence? Something like this?
first_name = 'John';
last_name = 'Smith';
full_name = function(value){
if (typeof value != 'undefined')
{
// do the same as in the Knockout's write function
}
else
{
// do the same as in the Knockout's read function
}
}
What is the best practice?
Share Improve this question edited Oct 23, 2013 at 13:50 Anton Telesh asked Oct 22, 2013 at 19:45 Anton TeleshAnton Telesh 3,8523 gold badges24 silver badges33 bronze badges 2- possible duplicate of KO.Computed equivalent in Angular / Breeze Initializer – PW Kad Commented Oct 22, 2013 at 19:48
- 1 I don't think this is a duplicate. He's asking about how to implement a "setter", which the other question doesn't cover. – Ken Smith Commented Oct 22, 2013 at 20:23
2 Answers
Reset to default 12I've found such a solution: http://jsfiddle/Girafa/V8BNc/
Instead of using angular $watch method, we set native javascript getter and setter of the fullName property:
Object.defineProperty($scope, 'fullName', {
get: function(){
#...
},
set: function(newValue){
#...
}
})
Think this is more convenient as I don't need to make any special watchers in the code. But I don't know about browser support of this solution.
Sorry about that. It's true, this is simpler in knockout because a function is called whereas a property is used in angular. This is the way I could resolve it, but I would like to know if there is a better way.
I fixed this time Plunker
app.controller('Ctrl', function($scope) {
$scope.firstName = 'John';
$scope.lastName = 'Smith';
$scope.getFullName = function() {
$scope.fullName = $scope.firstName + ' ' + $scope.lastName;
return $scope.fullName;
}
$scope.$watch('fullName', function(newValue, oldValue) {
var names = newValue.split(' ');
$scope.firstName = names[0];
$scope.lastName = names[1];
});
});