The service integrates with a variety of front-end frameworks, including Angular.
This post will describe how to use Angular's tools to interact with an Rdbhost.com hosted database account. It will take a multi-stage approach to the explanation, showing how to adapt jQuery based code to use Angular's alternative $http, and then to use the higher level $resource feature.
All the examples below assume that the libraries are included in your app, like below. If you use the Rdbhost generated boilerplate, all this will be included.
and that the jQuery plugin is initialized like:<script src="http://www.rdbhost.com/js/jquery.rdbhost.js"></script> <script src="http://www.rdbhost.com/js/jquery.rdbhost.utils.js"></script> <script src="js/rdbhostangular.js"></script>
// tell rdbhost module what role and account we are using // $.rdbHostConfig({ accountNumber : 929, domain : "www.rdbhost.com", userName : 'preauth' }); var mod = angular.module('myApp.controllers', []);
jQuery Plugin
Our first example is to use the Rdbhost jQuery plugin to list records from a table, and to add a record to it.mod.controller('DvdPreauthController', ['$scope', function($scope) { $scope.status = []; $scope.dvds = []; function add_status_line(ln) { $scope.status.push(ln); $scope.$apply(); } var addDVDSQL = 'INSERT INTO sandbox.dvds_jsintro ' + ' (name, rating) VALUES (%s, %s) '; var getDVDListSQL = 'SELECT name, rating FROM sandbox.dvds_jsintro'; // submit handler for form // $scope.addDVD = function () { var args = [ $scope.newTitle, $scope.newRating ]; R.preauthPostData( { 'q' : addDVDSQL, 'args' : args, 'callback' : function (resp) { add_status_line('one DVD added'); return resp; }, 'errback' : function(err) { add_status_line(err[0] + ' ' + err[1]); return err; } }); return false; }; // updater for list // $scope.listDVDs = function () { R.preauthPostData( { 'q' : getDVDListSQL, 'callback' : function (resp) { $scope.dvds.push.apply($scope.dvds, resp.records.rows); add_status_line('DVDs listed'); $scope.$apply(); return resp; }, 'errback' : function(err) { add_status_line(err[0] + ' ' + err[1]); return err; } }); return false; }; }]);
The method R.postPreauthData() is from the rdbhost.jquery.js plugin. It submits the query and passes results to the callback. The $scope.$apply() function is called to allow Angular to work. Because Rdbhost is calling the callback instead of Angular, we use $apply to keep Angular 'in the loop'.
$http
Here is the same code, written to use Angular's lower level $http feature. Well, almost. Actually we use the Rdbhost variation of the $http feature, $rdbhttp. It has the same methods as $http, but ensures that errors returned in the body of the response, as well as errors in the header, are handled by the error handler function.
Notice the absence of $apply() calls. Because Angular itself is calling the success and error callbacks, it invokes its own $apply function as necessary. R.getPost generates the correct url and post data to retrieve the query results from the server.// Produce a new rdbHttp object for interacting with Rdbhost mod.factory('$rdbHttp', ['$http', '$q', RdbhostAngular.makeRdbHttp]); // Change $httpProvider -remove unwanted headers add response transform mod.config(['$httpProvider', RdbhostAngular.providerInit]); mod.controller('DvdPreauthController', ['$scope', '$rdbHttp', function($scope, $http) { var R = window.Rdbhost; $scope.status = []; $scope.dvds = []; function add_status_line(ln) { $scope.status.push(ln); } var addDVDSQL = 'INSERT INTO sandbox.dvds_jsintro ' + ' (name, rating) VALUES (%s, %s) '; var getDVDListSQL = 'SELECT name, rating FROM sandbox.dvds_jsintro'; // submit handler for form $scope.addDVD = function () { var args = [ $scope.newTitle, $scope.newRating ]; var opts = { q: addDVDSQL, userName: 'preauth', args: args, authcode: '' }, gP = R.getPOST(opts), p = $http.post( gP.url, gP.data ); p.success(function(data, status, headers, config) { add_status_line('DVD was added.'); }); p.error(function(data, status, headers, config) { add_status_line(data.error[0] + ' ' + data.error[1]); }); return false; }; // updater for list // $scope.listDVDs = function () { var opts = { q: getDVDListSQL, userName: 'preauth', authcode: '' }, gP = R.getPOST(opts), p = $http.post( gP.url, gP.data ); p.success(function(resp, status, headers, config) { $scope.dvds.length = 0; $scope.dvds.push.apply($scope.dvds, resp.records.rows); add_status_line('DVDs listed'); }); p.error(function(data, status, headers, config) { var err = data.error; add_status_line(err[0] + ' ' + err[1]); }); return false; }; }]);
$resource
The $resource feature wraps a server resource in a JavaScript class, and allows operations on the server through methods on the class. The class can retrieve and send data to the server, and records received have methods to update or delete their own persistent values on the server.With Rdbhost, you can use the $resource from the stock ngResource module, but you need transform functions passed into each of the actions. You must use Angular 1.2 or later, as that version introduces the transformation functions. See example here:
The rdbhostTransformRequest and rdbhostTransformResponseFactory functions handle converting the http requests as Angular generates them into what Rdbhost.com expects, and converts the Rdbhost.com results into the form Angular expects. Other than providing those functions to your $resource constructor, and providing the SQL and parameters in the constructors action list, $resource acts like $resource. If you provide a save or delete action, that action will be available on the retrieved objects, as $save or $delete. The rdbhostTransformResponseFactory takes one parameter isArray.// Produce a new rdbHttp object for interacting with Rdbhost mod.factory('$rdbHttp', ['$http', '$q', RdbhostAngular.makeRdbHttp]); // Change $httpProvider -remove unwanted headers add response transform mod.config(['$httpProvider', RdbhostAngular.providerInit]); mod.controller('DvdPreauthController', ['$scope', '$rdbHttp', '$resource', function($scope, $http, $resource) { var R = window.Rdbhost, RA = R.Angular; $scope.status = []; $scope.dvds = []; function add_status_line(ln) { $scope.status.push(ln); } var addDVDSQL = 'INSERT INTO sandbox.dvds_jsintro (name, rating) VALUES (%(name), %(rating)) '; var getDVDListSQL = 'SELECT name, rating FROM sandbox.dvds_jsintro'; var role = 'p'+('000000000'+ $.rdbHostConfig.opts.accountNumber).substr(-10); var listResource = $resource('https://:domain/db/:userName', { // default params userName: role, domain: $.rdbHostConfig.opts.domain }, { // actions query: { method: 'GET', params: { q: getDVDListSQL, format: 'json-easy' }, isArray: true, transformRequest: RA.rdbhostTransformRequest, transformResponse: RA.rdbhostTransformResponseFactory(true) }, save: { method: 'POST', params: { q: addDVDSQL, format: 'json-easy' }, transformRequest: RA.rdbhostTransformRequest, transformResponse: RA.rdbhostTransformResponseFactory(false) } } ); // updater for list $scope.listDVDs = function () { $scope.dvds = listResource.query({}, function() { add_status_line('DVDs listed'); }, function() { add_status_line('DVD List Error'); } ); return false; }; // submit handler for form $scope.addDVD = function () { var namedParams = { name: $scope.newTitle, rating: $scope.newRating }; listResource.save(namedParams, function() { add_status_line('DVD was added.'); }, function(err) { add_status_line(err.message); } ); return false; }; }]);
If you have your own response transform functions, be sure to include them in the transform function list ahead of the Rdbhost functions and do not serialize as part of your transformation, as the Rdbhost functions will serialize the response at the end of the transformation chain.
The Rdbhost Website
No comments:
Post a Comment