AngularJS Modules for Great Justice

First off I want to thank Joel Hooks of the DFW Area AngularJS Meetup Group for suggesting this topic and providing the title.javascript

「Modules provide an excellent mechanism for cleanly dividing up our code into functional areas. Angular allows us to use modules in many ways. Let’s take a look at modules and some of the approaches we can leverage to produce cleaner more manageable code.」 – Joel Hookshtml

So with that, let’s get started.

AngularJS modules are the core means by how you define the components in your app. Besides defining your components, modules provide a way to indicate the dependencies your components require and they help you organize your components to help you write modular code that can be re-used across applications.java

As long as I have been developing with AngularJS there has always been the great best practices debate over how to structure your application. Do you use a package by feature or package by layer approach?git

Both have their advantages and disadvantages so let’s take a quick look at each before we get into how to implement each using AngularJS.angularjs

Package by Feature

Package by Feature became popular in the Java development camp a few years back. The main tenet is that by keeping all the source code related to a specific feature in a single package, it was easier to develop and promote modularity across your source code. Entire features could be lifted from applications and re-used in others by just taking the source package, there was no need to bring in other packages since all the code for the feature was in one place.github

Package-by-feature uses packages to reflect the feature set. It places all items related to a single feature (and only that feature) into a single directory. This results in packages with high cohesion and high modularity, and with minimal coupling between packages. Items that work closely together are placed next to each other. They aren’t spread out all over the application.web

A lot of developers feel that by using Package by Feature, development on large projects is also easier since all the code you need to deal with is in one place and as the project grows following the pattern keeps everything well organized.express

A good example of Package by Feature, is the angular-sprout seed project. All of the JavaScript files for a particular feature are included in the same directory. So, if we have three different views in our application; User, Movie, and Rating, we’ll have three directories in our source code. Each folder would include the source for the controllers, services, models and templates for the specific feature.bootstrap

Package by Layer

If you come from a object-oriented development background, you probably are more familiar with the Package by Layer approach. The basic tenets of object-oriented design is to design by layers; data access, business logic, business entities, and user interface. This way your code is organized in layers that communicate with each other via specific interfaces. Depending on how rigid your design and coding rules are, layers only interacted with adjacent layers and never called across boundaries.app

In package-by-layer, the highest level packages reflect the various application 「layers」, instead of features. Each feature has its implementation spread out over multiple directories, over what might be loosely called 「implementation categories」. Each directory contains items that usually aren’t closely related to each other. This results in packages with low cohesion and low modularity, with high coupling between packages.

Package by Layer works great when it comes to code that implements cross cutting concerns, however when you are developing features that span multiple layers development becomes harder since you have to deal with source files in various places. Package by Layer also promotes cross package dependencies which some feel are a bad thing.

A good example of Package by Layer, is the angular-seed project. All of the JavaScript files for a particular type of AngularJS component are included in the same directory or file. So, our application would be divided across the following source folders; controllers, directives, services, models, filter, etc.

So What’s All This Got to do with AngularJS Code

So, you’re probably wondering, 「Why should I care, I write web apps?」  Actually how you partition your code has a lot to do with how you organize your project source code and how much easier it is to maintain as your project grows.

Depending on the size of your project your are going to have to decide on how you want your code structured. Do you keep everything in a single JavaScript file, do you break your files based on layers, do you break your files based on features, or do you structure your code directories by features and structure your source files by layers?

Tiny Projects

For me, if I have a very small or tiny project that does one thing, I might keep all of my source code in a single file using single module. Below is a picture of the project’s directory structure. Notice we have a single file, app.js which holds our source code and our index.html file which loads and bootstraps our app.

Below is the app.js file, I’ve defined a single module called myApp and used the controller and directive definition methods to define my controller and directive used by my app. Everything is in one place and easy to maintain, which is all you need when writing a very simple application.

 
  1.         angular.module('myApp', [])


  2.         // register the controller for the app

  3.         .controller('myController',['$scope', function($scope){

  4.             $scope.format = 'M/d/yy h:mm:ss a';

  5.         }])


  6.         // Register the 'myCurrentTime' directive factory method.

  7.         // We inject $timeout and dateFilter service since the factory method is DI.

  8.         .directive('myCurrentTime', function($timeout, dateFilter) {

  9.             // return the directive link function. (compile function not needed)

  10.             return function(scope, element, attrs) {

  11.                 var format,  // date format

  12.                     timeoutId; // timeoutId, so that we can cancel the time updates


  13.                 // used to update the UI

  14.                 function updateTime() {

  15.                     element.text(dateFilter(new Date(), format));

  16.                 }


  17.                 // watch the expression, and update the UI on change.

  18.                 scope.$watch(attrs.myCurrentTime, function(value) {

  19.                     format = value;

  20.                     updateTime();

  21.                 });


  22.                 // schedule update in one second

  23.                 function updateLater() {

  24.                     // save the timeoutId for canceling

  25.                     timeoutId = $timeout(function() {

  26.                         updateTime(); // update DOM

  27.                         updateLater(); // schedule another update

  28.                     }, 1000);

  29.                 }


  30.                 // listen on DOM destroy (removal) event, and cancel the next UI update

  31.                 // to prevent updating time after the DOM element was removed.

  32.                 element.bind('$destroy', function() {

  33.                     $timeout.cancel(timeoutId);

  34.                 });


  35.                 updateLater(); // kick off the UI update process.

  36.             }

  37.         });

Below is the index.html file that instantiates the app and displays an input field that allows you to modify the time format that is used by the myCurrentTime directive to display the current date and time to the user.

 
  1.         <!DOCTYPE html>

  2.         <html ng-app="myApp">

  3.         <head>

  4.             <title></title>

  5.             <script src="lib/angular/angular.js" type="text/javascript"></script>

  6.             <script src="app/app.js" type="text/javascript"></script>

  7.         </head>

  8.         <body>

  9.         <div ng-controller="myController">

  10.             <input type="text" ng-model="format"/>

  11.             <div>Time format is: {{format}}</div>

  12.             <div>Current time is: <span my-current-time="format"></span></div>

  13.         </div>

  14.         </body>

  15.         </html>

Small Projects

As your project starts to get a little more complicated you might want to start organizing your code into multiple modules, but still keep it in a single file. Below is a picture of the project’s directory structure. Notice we still have a single file, app.js which holds our source code and our index.html file which loads and bootstraps our app.

Below is the app.js file, I’ve defined multiple modules one for the controllers, a second for the directives and a third for the app. I have again used the controller and directive definition methods to define my controller and directive used by my app, but now they are defined on specific modules defined for each type of AngularJS component type. I have also added dependencies to the application’s module definition so everything get injected properly. Everything is in one place and easy to maintain, but is ready for when you need to separate your code into multiple files when they get too big to maintain.

 
  1.         angular.module('myControllers', [])

  2.             // register the controller for the app

  3.             .controller('myController',['$scope', function($scope){

  4.                 $scope.format = 'M/d/yy h:mm:ss a';

  5.             }]);


  6.         angular.module('myDirectives', [])

  7.             // Register the 'myCurrentTime' directive factory method.

  8.             // We inject $timeout and dateFilter service since the factory method is DI.

  9.             .directive('myCurrentTime', function($timeout, dateFilter) {

  10.                 // return the directive link function. (compile function not needed)

  11.                 return function(scope, element, attrs) {

  12.                     var format,  // date format

  13.                         timeoutId; // timeoutId, so that we can cancel the time updates


  14.                     // used to update the UI

  15.                     function updateTime() {

  16.                         element.text(dateFilter(new Date(), format));

  17.                     }


  18.                     // watch the expression, and update the UI on change.

  19.                     scope.$watch(attrs.myCurrentTime, function(value) {

  20.                         format = value;

  21.                         updateTime();

  22.                     });


  23.                     // schedule update in one second

  24.                     function updateLater() {

  25.                         // save the timeoutId for canceling

  26.                         timeoutId = $timeout(function() {

  27.                             updateTime(); // update DOM

  28.                             updateLater(); // schedule another update

  29.                         }, 1000);

  30.                     }


  31.                     // listen on DOM destroy (removal) event, and cancel the next UI update

  32.                     // to prevent updating time after the DOM element was removed.

  33.                     element.bind('$destroy', function() {

  34.                         $timeout.cancel(timeoutId);

  35.                     });


  36.                     updateLater(); // kick off the UI update process.

  37.                 }

  38.             });


  39.         angular.module('myApp', ['myControllers', 'myDirectives']);

Notice that the index.html file used in our example below hasn’t changed.

 
  1.         <!DOCTYPE html>

  2.         <html ng-app="myApp">

  3.         <head>

  4.             <title></title>

  5.             <script src="lib/angular/angular.js" type="text/javascript"></script>

  6.             <script src="app/app.js" type="text/javascript"></script>

  7.         </head>

  8.         <body>

  9.         <div ng-controller="myController">

  10.             <input type="text" ng-model="format"/>

  11.             <div>Time format is: {{format}}</div>

  12.             <div>Current time is: <span my-current-time="format"></span></div>

  13.         </div>

  14.         </body>

  15.         </html>

Medium Projects

As your projects get larger and more complex, you might want to follow the angular-seed approach and split your source code across layers. In the sample code, I’ve broken the code out across multiple source files by layer, there is one file for the app, one for the controllers and one for the directives. Depending on how you want to structure your code, you can use a single module or multiple modules. In this example I use a single module that is assigned to a variable that is referenced by the other source files.

Our source code structure has changed. Now, instead of just an app directory there is now a controllers and directives directory.

Below is the contents of the app.js file. All we are doing is declaring the module that will be used by app:

        var myModule = angular.module('myApp', []);

Again, I am using a single module across my entire app so I am assigning it to variable, which I will reference in the other source files to define my controller and directive.

Below is the contents of the controllers.js file. Here we are using the myModule variable to define the controller.

 
  1.         // register the controller for the app

  2.         myModule.controller('myController',['$scope', function($scope){

  3.             $scope.format = 'M/d/yy h:mm:ss a';

  4.         }]);

Below is the contents of the directives.js file. Again we are using the myModule variable to define the directive.

 
  1.         myModule.directive('myCurrentTime', function($timeout, dateFilter) {

  2.             // return the directive link function. (compile function not needed)

  3.             return function(scope, element, attrs) {

  4.                 var format,  // date format

  5.                     timeoutId; // timeoutId, so that we can cancel the time updates


  6.                 // used to update the UI

  7.                 function updateTime() {

  8.                     element.text(dateFilter(new Date(), format));

  9.                 }


  10.                 // watch the expression, and update the UI on change.

  11.                 scope.$watch(attrs.myCurrentTime, function(value) {

  12.                     format = value;

  13.                     updateTime();

  14.                 });


  15.                 // schedule update in one second

  16.                 function updateLater() {

  17.                     // save the timeoutId for canceling

  18.                     timeoutId = $timeout(function() {

  19.                         updateTime(); // update DOM

  20.                         updateLater(); // schedule another update

  21.                     }, 1000);

  22.                 }


  23.                 // listen on DOM destroy (removal) event, and cancel the next UI update

  24.                 // to prevent updating time after the DOM element was removed.

  25.                 element.bind('$destroy', function() {

  26.                     $timeout.cancel(timeoutId);

  27.                 });


  28.                 updateLater(); // kick off the UI update process.

  29.             }

  30.         });

Finally below is the index.html file used to load and instantiate the app. Notice how we now need to include the new source files in order to load all of our code.

 
  1.         <!DOCTYPE html>

  2.         <html ng-app="myApp">

  3.         <head>

  4.             <title></title>

  5.             <script src="lib/angular/angular.js" type="text/javascript"></script>

  6.             <script src="app/app.js" type="text/javascript"></script>

  7.             <script src="controllers/controllers.js" type="text/javascript"></script>

  8.             <script src="directives/directives.js" type="text/javascript"></script>

  9.         </head>

  10.         <body>

  11.         <div ng-controller="myController">

  12.             <input type="text" ng-model="format"/>

  13.             <div>Time format is: {{format}}</div>

  14.             <div>Current time is: <span my-current-time="format"></span></div>

  15.         </div>

  16.         </body>

  17.         </html>

Large to Very Large Projects

As your application starts to grow and include more and more features, it is best to re-structure your source code’s directory structure by using the Package by Feature pattern and then use the Package by Layer pattern for your source files that relate to the feature. Below is a picture of the Large project’s structure. Now, we have created the folder structure based on the features in the app and the source files are based on the different AngularJS layers.

Below is the contents of the app.js file. We are again defining a module for our app. Notice that we have included the names of the other modules which define our features.

        angular.module('myApp', ['my-view-controller', 'time']);

Below is contents of the my-view-controller.js app, which defines our controller for the feature.

 
  1.         // register the controller for the app

  2.         angular.module('my-view-controller', []).controller('myController',['$scope', function($scope){

  3.             $scope.format = 'M/d/yy h:mm:ss a';

  4.         }]);

Since we are using the Package by Feature pattern to structure our code, we have now created a template that will be used to instantiate our controller. The reason for breaking out the HTML for the feature is because we want to have a fully re-usable package that we can pick up and drop into another application without having to drag other parts of the source application into your new app.

Below is the contents of my-view-partial.html:

 
  1.         <div ng-controller="myController">

  2.             <input type="text" ng-model="format"/>

  3.             <div>Time format is: {{format}}</div>

  4.             <div>Current time is: <span my-current-time="format"></span></div>

  5.         </div>

We have also moved our directive to it’s own feature directory giving us another module that we can reuse in other applications.

 
  1.         // Register the 'myCurrentTime' directive factory method.

  2.         // We inject $timeout and dateFilter service since the factory method is DI.

  3.         angular.module('time', []).directive('myCurrentTime', function($timeout, dateFilter) {

  4.             // return the directive link function. (compile function not needed)

  5.             return function(scope, element, attrs) {

  6.                 var format,  // date format

  7.                     timeoutId; // timeoutId, so that we can cancel the time updates

  8.        

  9.                 // used to update the UI

  10.                 function updateTime() {

  11.                     element.text(dateFilter(new Date(), format));

  12.                 }

  13.        

  14.                 // watch the expression, and update the UI on change.

  15.                 scope.$watch(attrs.myCurrentTime, function(value) {

  16.                     format = value;

  17.                     updateTime();

  18.                 });

  19.        

  20.                 // schedule update in one second

  21.                 function updateLater() {

  22.                     // save the timeoutId for canceling

  23.                     timeoutId = $timeout(function() {

  24.                         updateTime(); // update DOM

  25.                         updateLater(); // schedule another update

  26.                     }, 1000);

  27.                 }

  28.        

  29.                 // listen on DOM destroy (removal) event, and cancel the next UI update

  30.                 // to prevent updating time after the DOM element was removed.

  31.                 element.bind('$destroy', function() {

  32.                     $timeout.cancel(timeoutId);

  33.                 });

  34.        

  35.                 updateLater(); // kick off the UI update process.

  36.             }

  37.         });

Finally, we have our index.html file which loads our source and instantiates our app. Notice that we have a script tag for each of our source files and that we are using the ng-include directive to load our view’s template.

 
  1.         <!DOCTYPE html>

  2.         <html ng-app="myApp">

  3.         <head>

  4.             <title></title>

  5.             <script src="lib/angular/angular.js" type="text/javascript"></script>

  6.             <script src="app/app.js" type="text/javascript"></script>

  7.             <script src="myView/my-view-controller.js" type="text/javascript"></script>

  8.             <script src="time/time-directive.js" type="text/javascript"></script>

  9.         </head>

  10.         <body>

  11.         <div ng-include src="'myView/my-view-partial.html'"></div>

  12.         </body>

  13.         </html>

Source Code

https://github.com/lavinjj/angularjs-modules-for-great-justice

相關文章
相關標籤/搜索