If a resource returns nested objects then those are normally treated as just basic JavaScript objects stored on the resource instance. Using the serializers we can specify that a nested object is in fact a resouce and the object(s) will be constructed as a resource instance during deserialization.
For example, suppose we have an Author that serializes all of the books that author has written. If we specify that the books are a nested resource that allows us to more easily perform updates against those books without having to worry about creating a resource instance for the book.
angular.module('book.services', ['rails']);
angular.module('book.services').factory('Book', ['railsResourceFactory', function (railsResourceFactory) {
return railsResourceFactory({url: '/books', name: 'book'});
}]);
angular.module('book.services').factory('Author', ['railsResourceFactory', 'railsSerializer', function (railsResourceFactory, railsSerializer) {
return railsResourceFactory({
url: '/authors',
name: 'author',
serializer: railsSerializer(function () {
this.resource('books', 'Book');
})
});
}]);
angular.module('book.controllers').controller('AuthorCtrl', ['$scope', 'Author', function ($scope, Author) {
$scope.author = Author.get(123);
// allow the view to trigger an update to a book from $scope.author.books
$scope.updateBook = function (book) {
book.update();
}
}]);
While we don't have logic for full nested attributes support, the new serializer does allow you to specify which fields
should be passed with the _attributes
suffix.
angular.module('book.services').factory('Book', ['railsResourceFactory', 'railsSerializer', function (railsResourceFactory, railsSerializer) {
return railsResourceFactory({
url: '/books',
name: 'book',
serializer: railsSerializer(function () {
this.nestedAttribute('author');
})
});
}]);
Sometimes you don't want to serialize certain fields when updating an object. Take for instance the case of the author on a book. We know that we don't accept nested attributes for the author on the server so we want to exclude it from the JSON to reduce the amount of data being sent to the server.
angular.module('book.services').factory('Book', ['railsResourceFactory', 'railsSerializer', function (railsResourceFactory railsSerializer) {
return railsResourceFactory({
url: '/books',
name: 'book',
serializer: railsSerializer(function () {
this.exclude('author');
})
});
}]);
You can also be very restrictive and only include specific attributes that you want to send to the server. All other attribtues would be excluded by default.
angular.module('book.services').factory('Book', ['railsResourceFactory', 'railsSerializer', function (railsResourceFactory. railsSerializer) {
return railsResourceFactory({
url: '/books',
name: 'book',
serializer: railsSerializer(function () {
this.only('id', 'isbn', 'publicationDate');
})
});
}]);
You can add additional "class" or "instance" methods by modifying the resource returned from the factory call.
For instance, if you wanted to add a method that would search for Books by the title without having to construct the query params
each time you could add a new findByTitle
class function.
angular.module('book.services', ['rails']);
angular.module('book.services').factory('Book', ['railsResourceFactory', function (railsResourceFactory) {
var resource = railsResourceFactory({url: '/books', name: 'book'});
resource.findByTitle = function (title) {
return resource.query({title: title});
};
return resource;
}]);
angular.module('book.controllers').controller('BookShelfCtrl', ['$scope', 'Book', function ($scope, Book) {
$scope.searching = true;
// Find all books matching the title
$scope.books = Book.findByTitle({title: title});
$scope.books.then(function(results) {
$scope.searching = false;
}, function (error) {
$scope.searching = false;
});
}]);
You can also add additional methods on the object prototype chain so all instances of the resource have that function available.
The following example exposes a getAuthor
instance method that would be accessible on all Book instances.
angular.module('book.services', ['rails']);
angular.module('book.services').factory('Author', ['railsResourceFactory', function (railsResourceFactory) {
return railsResourceFactory({url: '/authors', name: 'author'});
}]);
angular.module('book.services').factory('Book', ['railsResourceFactory', 'Author', function (railsResourceFactory, Author) {
var resource = railsResourceFactory({url: '/books', name: 'book'});
resource.prototype.getAuthor = function () {
return Author.get(this.authorId);
};
return resource;
}]);
angular.module('book.controllers').controller('BookShelfCtrl', ['$scope', 'Book', function ($scope, Book) {
$scope.getAuthorDetails = function (book) {
$scope.author = book.getAuthor();
};
}]);
Or say you instead had a nested "references" service call that returned a list of referenced books for a given book instance. In that case you can add your own addition method that calls $http.get and then passes the resulting promise to the processResponse method which will perform the same transformations and handling that the get or query would use.
angular.module('book.services', ['rails']);
angular.module('book.services').factory('Book', ['railsResourceFactory', '$http', function (railsResourceFactory, $http) {
var resource = railsResourceFactory({url: '/books', name: 'book'});
resource.prototype.getReferences = function () {
var self = this;
return resource.$get(self.$url('references')).then(function (references) {
self.references = references;
return self.references;
});
};
}]);
Transformers can be specified by an array of transformers in the configuration options passed to railsResourceFactory.
However, a cleaner way to write it is to use the beforeRequest
which can take a new anonymous function or
a function returned by a factory if you want to share a transformer across multiple resources.
Both of these examples can be accomplished using the serializers now.
angular.module('test').factory('excludePrivateKeysTransformer', function () {
return function (data) {
angular.forEach(data, function (value, key) {
if (key[0] === '_') {
delete data[key];
}
});
});
});
angular.module('test').factory('Book', function (railsResourceFactory, excludePrivateKeysTransformer) {
var Book = railsResourceFactory({url: '/books', name: 'book'});
Book.beforeRequest(excludePrivateKeysTransformer);
Book.beforeRequest(function (data) {
data['release_date'] = data['publicationDate'];
delete data['publicationDate'];
});
return Book;
});