If you've been keeping up to date on the RhoMoible blogs, then you've seen the recent post about all the benefits of using Ionic in your RhoMobile application. A big part of what makes Ionic so extraordinary is that it is built on top of AngularJS.


While many developers know about the power of AngularJS and its data-binding abilities, few might know about Angular's impressive finer points: directives and services. Directives are one of Angular’s most overlooked, yet most powerful features.

 

What is a Directive?

At its most basic level, a directive is a marker on a DOM element (an attribute, element name, comment, or CSS class) that tells the Angular compiler in your HTML to attach a specified behavior to the marked DOM element. This allows you to take a lot of repetitive code out of your HTML files. You can name your directives based on their purpose, such as “displayImages” or “showWeeklyWeather.” This gives the added bonus of making your code clearer and easier to read.


On top of this, Angular lets you have custom directives and create your own reusable functionality. This is where Angular directives really shine.


Below is a custom Directive that changes the background color of “Hello World” to match the color in the input box.



HTML Code:


<html>

  <head>
    <meta charset="utf-8" />
    <script data-require="angular.js@1.2.x" src="http://code.angularjs.org/1.2.7/angular.js" data-semver="1.2.7"></script>
    <script src="js/app.js"></script>
  </head>


  <body ng-controller="MainCtrl" ng-app="myapp">
  <input type="text" ng-model="color" placeholder="Enter a color" />
  <hello-world/>
</body>
</html>










JavaScript Code:


var app = angular.module('myapp', []);


app.controller('MainCtrl', function($scope) {
  $scope.name = 'World';
});


app.directive('helloWorld', function() {
  return {
    restrict: 'AE',
    replace: true,
    template: '<p style="background-color:{{color}}" >Hello World </p>',
    link: function(scope, elem, attrs) {
      elem.bind('click', function() {
        elem.css('background-color', 'white');
        scope.$apply(function() {
          scope.color = "white";
        });
      });
      elem.bind('mouseover', function() {
        elem.css('cursor', 'pointer');
      });
    }
  };
});









This might look light a bit of code, but if you want this functionality in multiple places on a web page, all you have to do is drop the hello world directive into the HTML “<hello-world/>” and watch the magic happen.


The “restrict” and “replace” properties from the code above are optional.


Restrict lets your HTML know how you are going to reference your directives and can be set to:

'A' - only matches attribute name

'E' - only matches element name

'C' - only matches class name

or any combination of the above:

'AEC' - matches either attribute or element or class name

If you don’t set restrict to anything, its default status will be 'AE'.


“Replace: true” means that the content of the directive template will replace the element that the directive is declared on, in this case, the </hello-world> tag.


Another important thing to notice in the code above is that the directive is named “helloWorld” in the JavaScript file, but is referenced as “hello-world” in the HTML code. This is not a typo; your HTML file knows that hello-world = helloWorld. It is important to follow this pattern as HTML files expect “snake-case” variables and JavaScript files expect “camelCase” variables.


There’s a lot of potential in custom directives and they are becoming increasingly popular.


Angular Google Maps is a great example of how to take advantage of custom-made directives. Angular Google Maps is a set of directives (part of angular-ui) written in CoffeeScript and JavaScript which integrate Google Maps in an AngularJS application. There are directives for most of the widely-used Google Maps objects, including markers, windows, lines and shapes, and coming up is integration with more layer objects, including the HeatMap.


Screen Shot 2015-07-30 at 3.52.27 PM.png


UI-Bootstrap is a useful directive that comes with many of the great Bootstrap features that you’ve grown to love, but without any dependency on jQuery or Bootstrap’s JavaScript. Bootstrap offers a powerful set tools that can take you far and wide. (The website below was created with Bootstrap)


Screen Shot 2015-07-30 at 3.57.04 PM.png


Check out more of the gorgeous designs people have created.


If you are in the market for more basic features, UI-Calendar lets you implement a fully functioning Arshaw FullCalendar or UI-Router, which provides many routing options that don’t inherently come with Angular.


Screen Shot 2015-07-30 at 4.06.53 PM.png


Whatever you need, there’s probably a directive for that


Directive Deep Dive

The Ionic blog post touched on earlier had a great feature where a user's profile would load with a generic profile picture and then clicking on the picture would let you use RhoMobile's Camera API to open your device's camera and take a new profile picture.


Screen Shot 2015-07-30 at 4.29.59 PM.png


This feature will come in handy, so let's turn it into a directive for future use.


.directive("profileDirective", function() {
    return {
        restrict: "AEC",
        template: '<div class="item item-image" ng-click="takePicture()"> <img ng-src={{dataUriModel}}> <div/>',
        link: function(scope, element) {
          scope.dataUriModel = "";
          scope.takePicture = function(){
          Rho.Camera.takePicture({
            outputFormat: Rho.Camera.OUTPUT_FORMAT_DATAURI
            },
            function(resp){
                scope.dataUriModel = resp.imageUri;
                scope.$apply();
          });
        }
        }
    };
});







We pass the original image (in dataUri format) into the scope of our directive, and then simply make the necessary API calls and store the new photo as our profile picture.


Any time you want to use this feature, you could drop "<profile-directive/>" into your HTML and reap all the benefits without any additional code.


Clean, simple, and easy to implement, directives are the way to go.


The Best in Service

Services are singletons, which are objects that are instantiated only once per app (by the $injector). They provide an interface to keep together methods that relate to a specific function. The $http service is an example of an AngularJS service that provides low-level access to the browser’s XMLHttpRequest object. Rather than needing to dirty the application with low-level calls to the XMLHttpRequest object, we can simply interact with the $http API.

Angular offers several useful services (like $http), but for most applications you'll also want to create your own. You can define your own service by registering the service's name and service factory function, with an Angular module. This function generates the single object or function that represents the service to the rest of the application. The object or function returned by the service is injected into any component (controller, service, filter or directive) that specifies a dependency on the service.

Services are registered to modules via the Module API. Typically you use the Module factory API to register a service:

var myModule = angular.module('myModule', []);
myModule.factory('serviceId', function() {
  var shinyNewServiceInstance;
  // factory function body that constructs shinyNewServiceInstance
  return shinyNewServiceInstance;
});





Note that you are not registering a service instance, but rather a factory function that will create this instance when called.

To use an Angular service, you add it as a dependency for the component (controller, service, filter or directive) that depends on the service. Angular's dependency injection subsystem takes care of the rest.


Have it Your Way

Just like with directives, we can write custom services. Let's take the code from the Ionic blog post, and add a custom service that uses RhoMobile's barcode API


Go into build.yml and add the  "- barcode" extension.


extensions:
  - rhoconnect-client
  - mediacapture
  - barcode





Next, add the factory method for the service into Controller.js and add the necessary dependencies to the Profile Controller:


.controller('ProfileCtrl', ['$scope', '$stateParams', '$zebraBarcodeScanner', function($scope, $stateParams,$zebraBarcodeScanner) {
  $scope.dataUriModel = "";
  $scope.barcode = "Scan";
  $scope.takePicture = function(){
    Rho.Camera.takePicture({
      outputFormat: Rho.Camera.OUTPUT_FORMAT_DATAURI
      },
      function(resp){
          $scope.dataUriModel = resp.imageUri;
          $scope.$apply();
    });
  }


  $scope.scanBarcode = function(){
    $zebraBarcodeScanner
      .scan()
      .then(function(barcodeData) {
        // Success! Barcode data is here
        $scope.barcode = barcodeData;
      }, function(error) {
        // An error occurred
        alert(error)
      });
  }


}])


.factory('$zebraBarcodeScanner', ['$q', function ($q) {


    return {
      scan: function (config) {
        var q = $q.defer();


        if(Rho.Barcode){
          Rho.Barcode.take({},
            function(resp){
              if(resp.status=='ok'){
                q.resolve(resp.barcode);
              }else{
                q.reject('user cancelled or timeout occured');
              }
            });
        }
        else{
          q.reject('API not found - Make sure to add barcode extension and perform clean build.')
        }


        return q.promise;
      }
    };
  }]);





Now if you go into profile.html inside of Public/templates and remove the ng-if statement from the page and compile your code, your profile page should have a spiffy new scanner!


</div>
    <div class="list">
    <div class="item">
    <button class="button button-full button-positive icon-left ion-ios-barcode-outline" ng-click="scanBarcode()">{{barcode}}</button>
    </div>
    </div>
  </ion-content>
</ion-view>





Screen Shot 2015-07-31 at 1.05.54 PM.png

 

As the code above suggests, services are used more in the controller layer then in presentation, while directives are used mostly to influence what your users see. Directives and Services are both great ways to tidy up your HTML and JavaScript files and to keep everything organized and effective. Start looking into these options as a way to bring your RhoMobile apps to the next level.