angular-bp
v0.1.9
Published
A boilerplate resulting from the study of AngularJS boilerplates, modularity and application architectures
Downloads
13
Maintainers
Readme
#AngularJS boilerplate (study)
This study looks at AngularJS boilerplates to find the best approaches so you don't have to. Although plenty of choices are available, they will do justice for only some types of applications. There is no one-size-fits all, universal boilerplate. Plan your app well, prototype it and then refactor. If you still think it's a good idea to use a boilerplate, choose one and adapt it to your needs.
See a live app built with this boilerplate.
##Getting started
Install npm
and bower
dependencies and run grunt
, that's it. You are good to go.
$ sudo npm install
$ bower install
$ grunt
After running grunt, the source files will be built into ./build
.
When the app is ready for production, compile the app into ./application
by running:
$ grunt compile
###Requirements
Grunt
, grunt-cli
and Bower
are required to build the application.
Read more about these packages on the Grunt or Bower documentation pages.
##Table of contents
- Philosophy
- Application structure
- Build and compile * Build * Compile
- Core components
- Third party components
- Conventions and best practices
- Testing * Unit testing * End to end testing
##Philosophy
The modular nature of AngularJS (mainly) facilitated the development of numerous boilerplates. Based on the insanely comprehensive analysis done by Dan Cancro (link), it's noticeable that boilerplates exist for certain purposes and are not in fact the result of different approaches to achieve the same exact thing. There is no question that they do have the common purpose of quickly starting off development. At the same time, they are forcing some patterns/concepts that their creators think help in some way. This is where it's easy to see why they are different. There are many nuances to some way, it can be:
- flexibility
- performance
- learnability/documentation
- efficiency (as in increased dev. speed)
- testability
- architecture (structure/overview/understanding)
- maintainability, etc
Some developers will value one of them more than the others. This doesn't mean that if testability is considered to be the most important, the others will be neglected. In fact, you probably won't be able to address one issue without affecting the others. It's easier to get this if you look at what this man had to say back when 'personal' and 'computer' where two unrelated terms:
"The benefits expected of modular programming are: managerial-development time should be shortened because separate groups would work on each module with little need for communication: product flexibility-it should be possible to make drastic changes to one module without a need to change others; comprehensibility-it should be possible to study the system one module at a time. The whole system can therefore be better designed because it is better understood."
Parnas, D. L. On the Criteria To Be Used in Decomposing Systems into Modules, 1972
Modularity's umbrella can cover most of the decisions we consider when creating a boilerplate. Therefore, that's the first thing this study focuses on: getting modularity right.
Performance and comprehensibility are chosen as being the focus areas of this boilerplate. The two don't usually play well, that's why this makes for a somewhat noble goal. On one hand, performance decisions can be measured and justified. That's going to make some implementations easier to digest. On the other hand, comprehensibility is more personal and hard to measure. That's why it will require more research & testing. (more on this in the Conventions and best practices chapter)
Finally, the result of this study embraces the idea of certain boilerplates work best for certain applications
. It's based on a real project with clear demands, a specific development setup, a given team & a specific set of tools (technologies).
##Application structure
root/
+—— src/
| +—— app/
| | +—— _app-main/ # Root module -> application bootstrap
| | | +—— _app-main.config.js
| | | +—— _app-main.controller.js
| | | +—— _app-main.init.js
| | | +—— _app-main.spec.js
| | +—— module/ # Any custom module
| | | +—— module.config.js # Registering a module
| | | +—— module.controller.js
| | | +—— module.e2e.js # Module E2E test
| | | +—— module-page.html # Module views
| | | +—— module.spec.js # Module Unit test
| | +—— feature/ # Any custom feature (multiple modules)
| | | +—— module-one/
| | | | +—— module-one.config.js
| | | | +—— module-one.controller.js
| | | | +—— module-one-page.html
| | | | +—— module-one.spec.js
| | | +—— module-two/
| | | | +—— module-two.config.js
| | | | +—— module-two.controller.js
| | | | +—— module-two-page.html
| | | | +—— module-two.spec.js
| | | +—— feature.config.js
| | | +—— feature.e2e.js
| | +—— shared/ # Shared modules
| | | +—— directive/ # Any shared directive
| | | | +—— directive.directive.js
| | | | +—— directive.html
| | | +—— layout/ # Shared layout directives
| | | | +—— footer.directive.js
| | | | +—— footer.html
| | | | +—— layout.config.js
| | | | +—— layout.controller.js
| | | | +—— shell.directive.js
| | | | +—— shell.html
| | | +—— model/ # Any shared model
| | | | +—— model.config.js
| | | | +—— model.service.js
| +—— assets/ # All application assets
| | +—— css/
| | +—— fonts/
| | +—— images/
| | +—— less/
| +—— common/ # Common modules
| | +—— loggly/
| | | +—— loggly.config.js
| | | +—— loggly-exception-decorator.service.js
| | | +—— loggly-exception-logging.service.js
| | | +—— loggly-trace.service.js
| +—— test/ # Protractor helper methods (e2e tests)
| | | +—— helpers.protractor.js
| +—— index.html # Application index file
Browse the ./src
directory for examples.
Go to src/app/shared for more info on shared modules.
Go to src/common for more info on common modules.
##Build and compile
A 2-stage Grunt 'strategy' is implemented.
For development, files will go to ./build
.
For production, files will be concatenated, uglyfied / minified and moved to ./application
. (1 css, 1 js)
###Build
The build task will do all the work for development and create files in -> ./build
$ grunt build
You can use the following command to start the build http server (port 8008 default):
$ grunt dev
###Compile
The compile task will do all the work for production and create files in -> ./application
$ grunt compile
You can use the following command to start the build http server (port 8009 default):
$ grunt prod
###Watch
The watch
task builds JavaScript, HTML and all assets (including tests and the gruntfiles themselves) when any of them are changed.
##Core components
The only 'core' AngularJS modules are:
_app-main
: contains logic for manually bootstrapping the application, sets up a module for$templateCache
, allows defining 3rd party dependencies and exposes a method to register modules:pushAfterBootstrap
.layout
: (insideapp/shared
) contains theng-view
directive to display content based on the current route (ui-router
specific directive).layout
contains other directives for footer & navigation, but they can be removed. Theshell
directive is the one responsible forng-view
.
To start clean, you can remove the directories of sample modules or features: /app/about
, /app/home
, /app/profile-feature
and app/shared/profile-model
can be safely removed.
Note: app/shared/profile-model
is a dependency of /app/home
and /app/profile-feature
. Removing the modules also requires the navigation directive in /app/shared/layout/
to be updated.
# For the lazy developer, just run the following to
# remove the sample modules:
$ grunt clean:clean_samples
##Third party components
With the exception of ui-router
, third party components that are used to build the sample application are just a recommendation. They can be replaced with any other similar components or completely removed.
See the builds for this repository on Travis CI
To set up Sauce Labs with Travis CI you have to provide a SauceKey and SauceUsername securely in ./.travis.yml (you can remove all 3 secure env variables). See how to do it in SauceLabs' docs.
###Grunt 'contrib' tasks
Grunt tasks are used in order to automate the build and compilation of the app code.
Open ./package.json
to see the installed Grunt tasks. An external configuration file is used to define file locations in ./config/grunt.conf.js
. Both the config file and ./grunfile.js
are well commented. Reading it would be a good way to fully understand how the Grunt tasks work, but you don't have to.
jit-grunt
and grunt-newer
are used to speed up tasks.
##Conventions and best practices
###Workflow conventions These conventions should be followed during normal development.
####Working with Grunt
As a rule, Grunt tasks (./grunfile.js
) shouldn't require any alteration. Changes should be made in ./config/grunt.conf.js
. Before adding/changing a task, it's always a good idea to go through all tasks and check if it's not already there.
####Adding a third party dependency
Grunt 'glues together' the various application components and moves Bower dependencies where they belong (i. e. ../bootstrap/dist/css/bootstrap.min.css
-> ../assets/css
). Due to the way Bower dependencies are organized (well they aren't, really), this process cannot be fully automated. To add a third party dependency, you need to:
Step 1: Install it
$ bower install <component-name>
Step 2: Update the build object in ./config/grunt.conf.js
...
build: {
vendor_js: {
...
'./vendor/component-name/dist/js/component.js' //not minified
...
}
}
...
Step 3: Update the compile object in ./config/grunt.conf.js
...
compile: {
vendor_min_js: {
...
'./vendor/component-name/dist/js/component.min.js' //minified
...
}
}
...
Step 4 (optional) : If it's an Angular core component (such as ui-route
), it should be added to the appMainVendorDependencies
Array in the root module's init file (./src/app/_app-main/_app-main.init.js
).
Providing different source files for build and compile phases is done for a couple reasons:
- During build (aka. development time) meaningful error messages/exceptions should be provided for debugging.
- During build, it's useful to have the code 'beautified' (line numbers) and not 'mangled' (full variable names)
- There is no reason to minify a script if it's already minified by a vendor
- Attempting to minify a vendor script can break it (it can have it's own minifying settings)
For a more in-depth look, ./config/grunt.conf.js
contains useful comments.
####Naming conventions
#####Files and directories Lower case, separated by a dash
- Directory:
src/directory-name/
- Filename:
file-name.js
#####AngularJS directories
- Module dirs:
<module-name>/
- Feature dirs:
<feature-name>-feature/
(contains multiple modules) - Model dirs:
<module-name>-model/
(contains services)
#####AngularJS component filenames
- Init/globals:
<module-name>.init.js
- Config blocks:
<module-name>.config.js
- Run blocks:
<module-name>.run.js
- Services:
<module-name>.service.js
(including factories, providers, values) - Controllers:
<module-name>.controller.js
- Directives:
<module-name>.directive.js
- Filters:
<module-name>.filter.js
- Views:
<view-name>.html
#####Other files
- E2E tests:
<module-name>.e2e.js
,<feature-name>.e2e.js
(depending on the case) - E2E helpers:
<name>.protractor.js
- Unit tests:
<module-name>.spec.js
#####Angular modules
- Module names:
<namespace>.<moduleName>
- Feature names (collection of modules):
<namespace>.<featureName>Feature
- Nested modules (inside a feature):
<namespace>.<featureName>Feature.<moduleName>
(i.e. abs.about
, abs.profileFeature
, abs.profileFeature.providerPage
)
- Core modules:
<namespace>.core<ModuleName>
- Core features:
<namespace>.core<FeatureName>Feature
- Core nested module:
<namespace>.core<FeatureName>Feature.core<ModuleName>
(i.e. abs.coreLayout
)
- Common modules:
<namespace>.common<ModuleName>
- Common features:
<namespace>.common<FeatureName>Feature
- Common nested:
<namespace>.common<FeatureName>Feature.common<ModuleName>
(i.e. abs.commonLoggly
)
- Model names:
<namespace>.<modelName>Model
- Nested models:
<namespace>.<parentModel>Model.<modelName>Model
(i.e. abs.profileModel
, abs.profileModel.customerModel
)
#####Angular module components
- Controllers:
<ControllerName>Controller
- Models:
<ModelName><ModelType>
(ModelType could be service, factory, value) - Directives:
<directiveName>
- Filters:
<filterName>
####Creating a module To create a module, make a directory in a corresponding location:
./src/app/
for any basic module./src/app/shared/
for any shared module (i.e. directive, shared layout)./src/common/
for any common module
After making the directory, create a config file (<module-name>.config.js
). In this file the module needs to be registered as a dependency of the application root.
absConfig.pushAfterBootstrap('<namespace>.<module-name>');
That's it, use it as any other Angular module.
Here's a more practical example:
absConfig.pushAfterBootstrap('abs.home');
angular.module('abs.home').config(homeConfig);
homeConfig.$inject = ['$log'];
function homeConfig($log){
$log.warn('I am being practical');
}
####Component dependency injection
Dependencies are injected into components by creating a $inject property on the component function:
angular.module('user').controller('UserController', userController);
userController.$inject = ['$scope', '$stateParams', 'ProviderModelService'];
function userController($scope, $stateParams, ProviderModelService){
//controller logic
}
Note on annotation:
First of all automatic annotation is not implemented, although it can be achieved with ng-annotate
. This is so because every now and then ng-annotate
backfires. After many hours of debugging, you'll promise yourself to never use it and avoid such situations. Therefore, the 'old-school' annotation is used, but with a twist: it's not done inline with the component (in this case controller) definition. This makes it easy to have an overview of dependencies and easily manage them while being minification-proof.
####File locations during build/compile
Note: compiling will combine all shared views in a module and use $templateCache to define their location. The directory above won't exist in the compile phase, but the views will still be accessible in the same location.
####Routes
Routes are defined in module configuration blocks. In the case of features, routes are defined in the feature configuration block. The idea behind keeping the route logic in each module or feature is to encourage a higher modularity of components. Moreover, keeping module logic
Examples are available in the configuration block of the home module and profile feature.
####Performance & Misc Other style considerations:
- Only perform DOM manipulations with directives
- Use ng-bind, avoid interpolations
- Use the bind-once syntax to reduce watchers where possible
- Always use the 'controller as' syntax
- Each file holds one component (be it controller, directive, etc)
- Used data-* for attributes
- Use ui-sref instead of href for internal anchors
- Try to use 'ng-repeat as' while filtering to display a 'no results' message
- Don't use inline filters and ng-repeats if there's another way to achieve the same thing
##Testing
To run all tests use:
$ grunt test
Note: you should have a selenium server running, see end to end testing
Note: this won't build or compile, but just run all unit and end to end tests for the build binaries, which should be present in ./build
.
###Unit
To run unit tests it will come in handy to install karma-cli (if you don't have it yet).
Unit tests will run on PhantomJS, but if you have binaries for other browsers you are encouraged to add them in karma.conf.js
$ npm install -g karma-cli
$ grunt test:unit
You can use the continuous task while writing tests to auto refresh your changes
$ grunt karma:continuous
Unit tests will run in the grunt default task (which builds first)
$ grunt
###End to end
Protractor is used for running end to end tests. Before running the tests be sure the webdriver-manager is updated.
Protractor's base url is http://localhost:8008/#!
, but can be changed in protractor.conf.js
- just make sure the application port matches the base url.
In order to run Protractor tests, a selenium server needs to be running (default at http://localhost:4444/wd/hub
):
$ node_modules/protractor/bin/webdriver-manager start
$ grunt test:e2e
If you are running tests for the first time don't forget to update the webdriver
$ node_modules/protractor/bin/webdriver-manager update
##What's coming next
- Add
grunt-perfbudget
for better performance monitoring, perhaps together withgrunt-pagespeed
- Add
uncss
in the workflow (withphantomcss
) - Switch to Angular dependency 'strict mode'
- Extend documentation: provider visual resources for build & sample application structure; add a list of 'features' in the boilerplate about page
- Create unit tests for the home, customer-index & provider-index modules
- Extend E2E tests to at least cover all routes
Other future additions:
- Include
phantomas
in the workflow - make perf reporting trends ? - Reduce repetition in CSS - use
parker
,colorguard
? - Use debug: false in production ?
- Try
useApplyAsync
for httpProvider ? - Implement a form and animation example ?
- Change remaining inline filters to controller filters ?
- Work on Windows support (shell commands) ?
- Move dev/prod demo to github pages ?
##Mentions
Special thanks to the people at fadeit.dk for their patience and to Niels Henrik Juul Hansen for guiding me.
##License
Copyright (c) Dan Mindru [email protected]
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.