Yeoman generator for AngularJS apps with optional Polymer support
This generator focuses on organizing Angular components by feature (home, about, video player, etc.) instead of by type (controller, service, directive, etc.) to encourage the development of self-contained, reusable components.
A typical workflow with this generator consists of creating an Angular module (ng-poly:module) and then generating controllers, directives, etc. for this module to create a new feature.
Polymer is just an added feature, but it isn't required to utilize this generator.
Install generator-ng-poly
:
npm install -g generator-ng-poly
Run yo ng-poly
Yeoman will then ask for an app name and language preferences.
Run gulp
to build and start the development environment.
Please feel free to ask any questions on our GitHub Issues or Google Group
Available generators:
- AngularJS
- Polymer
Languages and Features supported:
- Markup
- HAML, HTML, Jade
- Application scripting languages
- CoffeeScript †, JavaScript
- Testing scripting languages
- CoffeeScript, JavaScript
- Style languages
- CSS, LESS, SCSS, Stylus
- Routers
- Angular Route, UI Router
- Unit testing
- Jasmine (Karma as the test runner) for AngularJS
- Mocha with Chai (Karma as the test runner) for AngularJS
- e2e testing
- Protractor (Jasmine) for AngularJS
- Frameworks
- Bootstrap with AngularStrap
- Bootstrap with UI Bootstrap
- Foundation with Angular Foundation
- Polymer
- Core, Paper
- Task runners
- Gulp
- Other supported Bower packages:
- Angular Animate
- Angular Cookies
- Angular Resource
- Angular Sanitize
- Angular Touch
- Font Awesome
- Lo-Dash
- Restangular
†Code coverage isn't currently performed on CoffeeScript application source code
gulp
will start a localhost and open in the default browser
Using --stage prod
will concat and minify HTML, CSS, and Angular modules.
gulp build
will compile the assets
gulp dev
will call the build task and setup the development environment
gulp unitTest
will run unit tests via Karma and create code coverage reports
gulp webdriverUpdate
will download the Selenium server standalone and Chrome driver for e2e testing
gulp e2eTest
will run e2e tests via Protractor (must start a localhost before running gulp e2eTest
)
All generators ask for a module name except app and element. All generators except app take a name as an argument. A name can be written with CamelCase or hyphens.
Generators requiring a module can take a module option to bypass the prompt:
yo ng-poly:view newView --module=home/kitchen
A module value of app
will add the new components to the module defined in app.js or app.coffee.
Examples are shown with HTML, LESS, JavaScript, Jasmine, and UI Router as the app configuration.
Asks for application name and language preferences to scaffold out an application with a home module. It will also ask if tests should be placed in the app/
or tests/
directory. It'll ask for some additional Bower dependencies and then install npm and Bower dependencies.
Example:
yo ng-poly
[?] What is the app's name?
[?] What host should the app run on?
[?] Which port should the app run on?
[?] Which folder should the app be developed in?
[?] Which is the preferred markup language?
[?] Which is the preferred application scripting language?
[?] Want to use Controller As syntax?
[?] Should functions be defined and passed instead of defined inline (in callbacks)?
[?] Want to use named functions instead of anonymous?
[?] Where should unit tests be saved?
[?] Which is the preferred test scripting language?
[?] Which is the preferred unit testing framework?
[?] Which is the preferred style language?
[?] Should Polymer support be enabled?
[?] Should a framework be setup?
[?] Should ngRoute be used instead of UI Router?
[?] Which additional Bower components should be installed?
Produces:
root/
├── app/
│ ├── fonts/ (empty)
│ ├── home/
│ │ ├── home.{coffee,js}
│ │ ├── home.{css,less,scss,styl}
│ │ ├── home.tpl.{haml,html,jade}
│ │ ├── home-controller.{coffee,js}
│ │ └── home-controller_test.{coffee,js}
│ ├── images/ (empty)
│ ├── app.{coffee,js}
│ └── index.{haml,html,jade}
├── bower_components/
├── e2e/
│ └── home/
│ ├── home.po.{coffee,js}
│ └── home_test.{coffee,js}
├── gulp/
│ ├── analyze.js
│ ├── build.js
│ ├── test.js
│ └── watch.js
├── node_modules/
├── .bowerrc
├── .editorconfig
├── .jscsrc
├── .jshintrc
├── .yo-rc.json
├── bower.json
├── build.config.js
├── Gulpfile.js
├── karma.config.js
├── package.json
├── protractor.config.js
└── README.md
Generates a constant and its test.
Example:
yo ng-poly:constant theHero
[?] Which module is this for?
Produces app/module/the-hero-constant.js
:
'use strict';
/**
* @ngdoc service
* @name module.constant:TheHero
*
* @description
*
*/
angular
.module('module')
.constant('TheHero', 0);
Produces app/module/the-hero-constant_test.js
:
/*global describe, beforeEach, it, expect, inject, module*/
'use strict';
describe('TheHero', function () {
var constant;
beforeEach(module('module'));
beforeEach(inject(function (TheHero) {
constant = TheHero;
}));
it('should equal 0', function () {
expect(constant).toBe(0);
});
});
Genrates a controller and its test.
Example:
yo ng-poly:controller micro
[?] Which module is this for?
Produces app/module/micro-controller.js
:
'use strict';
/**
* @ngdoc object
* @name module.controller:MicroCtrl
* @requires $scope
*
* @description
*
*/
angular
.module('module')
.controller('MicroCtrl', function ($scope) {
$scope.ctrlName = 'MicroCtrl';
});
Produces app/module/micro-controller_test.js
:
/*global describe, beforeEach, it, expect, inject, module*/
'use strict';
describe('MicroCtrl', function () {
var scope;
beforeEach(module('module'));
beforeEach(inject(function ($rootScope, $controller) {
scope = $rootScope.$new();
$controller('MicroCtrl', {$scope: scope});
}));
it('should have ctrlName as MicroCtrl', function () {
expect(scope.ctrlName).toEqual('MicroCtrl');
});
});
Generates a directive, its template, and its test.
Example:
yo ng-poly:directive fancy-button
[?] Which module is this for?
Produces app/module/fancy-button-directive.js
:
'use strict';
/**
* @ngdoc directive
* @name module.directive:fancyButton
* @restrict EA
* @element
*
* @description
*
* @example
<example module="module">
<file name="index.html">
<fancy-button></fancy-button>
</file>
</example>
*
*/
angular
.module('module')
.directive('fancyButton', function () {
return {
restrict: 'EA',
scope: {},
templateUrl: 'module/fancy-button-directive.tpl.html',
replace: false,
link: function (scope, element, attrs) {
element.text('fancyButton\n' + scope + '\n' + attrs);
}
};
});
Produces app/module/fancy-button-directive.tpl.html
:
<div></div>
Produces app/module/fancy-button-directive_test.js
:
/*global describe, beforeEach, it, expect, inject, module*/
'use strict';
describe('fancyButton', function () {
var scope;
var element;
beforeEach(module('module', 'module/fancy-button-directive.tpl.html'));
beforeEach(inject(function ($compile, $rootScope) {
scope = $rootScope.$new();
element = angular.element('<fancy-button></fancy-button>');
$compile(element)($rootScope);
}));
it('should have correct text', function () {
scope.$digest();
expect(element.html()).toEqual('fancyButton\n[object Object]\n[object Object]');
});
});
The directive's template (HAML, HTML, or Jade) is converted to a temporary module automatically for testing.
Generates a factory and its test.
Example:
yo ng-poly:factory cake
[?] Which module is this for?
Produces app/module/cake-factory.js
:
'use strict';
/**
* @ngdoc service
* @name module.factory:Cake
*
* @description
*
*/
angular
.module('module')
.factory('Cake', function () {
var CakeBase = {};
CakeBase.someValue = 'Cake';
CakeBase.someMethod = function () {
return 'Cake';
};
return CakeBase;
});
Produces app/module/Cake-factory_test.js
:
/*global describe, beforeEach, it, expect, inject, module*/
'use strict';
describe('Cake', function () {
var factory;
beforeEach(module('module'));
beforeEach(inject(function (Cake) {
factory = Cake;
}));
it('should have someValue be Cake', function () {
expect(factory.someValue).toEqual('Cake');
});
it('should have someMethod return Cake', function () {
expect(factory.someMethod()).toEqual('Cake');
});
});
Generates a filter and its test.
Example:
yo ng-poly:filter coffee
[?] Which module is this for?
Produces app/module/coffee-filter.js
:
'use strict';
/**
* @ngdoc filter
* @name module.filter:coffee
*
* @description
*
* @param {Array} input The array to filter
* @returns {Array} The filtered array
*
*/
angular
.module('module')
.filter('coffee', function () {
return function (input) {
var temp = [];
angular.forEach(input, function (item) {
if(item > 3) {
temp.push(item);
}
});
return temp;
};
});
Produces app/module/coffee-filter_test.js
:
/*global describe, beforeEach, it, expect, inject, module*/
'use strict';
describe('coffee', function () {
beforeEach(module('module'));
it('should filter our numbers not greater than 3', inject(function ($filter) {
expect($filter('coffee')([1,2,3,4])).toEqual([4]);
}));
});
Generates a new module and create a new route. Updates parent module's dependencies.
Top Level Example:
yo ng-poly:module top
Produces app/top/top.js
:
'use strict';
/* @ngdoc object
* @name top
*
* @description
*
*/
angular
.module('top', [
'ui.router'
]);
angular
.module('top')
.config(function ($stateProvider) {
$stateProvider
.state('top', {
url: '/top',
templateUrl: 'top/top.tpl.html',
controller: 'TopCtrl'
});
});
Produces app/top/top-controller.js
, app/top/top-controller_test.js
, app/top/top.tpl.html
, app/top/top.less
, e2e/top/top.po.js
, e2e/top/top_test.js
Updates app/app.js
:
'use strict';
/* @ngdoc object
* @name module
* @requires $urlRouterProvider
*
* @description
*
*/
angular
.module('module', [
'ui.router',
'home',
'top'
]);
angular
.module('module')
.config(function ($urlRouterProvider) {
$urlRouterProvider.otherwise('/home');
});
Deep Level Example:
yo ng-poly:module top/bottom
Produces app/top/bottom/bottom.js
, app/top/bottom/bottom-controller.js
, app/top/bottom/bottom-controller_test.js
, app/top/bottom/bottom.tpl.html
, app/top/bottom/bottom.less
, e2e/bottom/bottom.po.js
, e2e/bottom/bottom_test.js
Updates app/top/top.js
:
'use strict';
/* @ngdoc object
* @name top
* @requires $stateProvider
*
* @description
*
*/
angular
.module('top', [
'ui.router',
'top.bottom'
]);
angular
.module('top')
.config(function ($stateProvider) {
$stateProvider
.state('top', {
url: '/top',
templateUrl: 'top/top.tpl.html',
controller: 'TopCtrl'
});
});
Notice the module in app/top/bottom/
is called 'top.bottom'. All tests in this directory use this nomenclature, as well.
Deeper Level Example:
yo ng-poly:module top/bottom/bottomest
Produces 'bottom.bottomest' module, a controller, controller test, style, and a view in app/top/bottom/bottomest/
Updates 'top.bottom' module with the new 'bottom.bottemest' module as a dependency.
Deeperestier Level Example:
It just keeps going...
Empty modules
By running ng-poly:module newHome --empty
a module without a route will be created as such:
'use strict';
/* @ngdoc object
* @name newHome
*
* @description
*
*/
angular
.module('newHome', [
]);
angular
.module('newHome')
.config(function () {
});
It is still possible to add a route to this module via ng-poly:route. The route subgenerator will also add the ui.router dependency and $stateProvider paramater for the config function.
Generates a provider and its test.
Example:
yo ng-poly:provider bacon
[?] Which module is this for?
Produces app/module/bacon-provider.js
:
'use strict';
/**
* @ngdoc service
* @name module.provider:Bacon
*
* @description
*
*/
angular
.module('module')
.provider('Bacon', function () {
return {
$get: function () {
return 'Bacon';
}
};
});
Produces app/module/Bacon-provider_test.js
:
/*global describe, beforeEach, it, expect, inject, module*/
'use strict';
describe('Bacon', function () {
var provider;
beforeEach(module('module'));
beforeEach(inject(function (Bacon) {
provider = Bacon;
}));
it('should equal Bacon', function () {
expect(provider).toEqual('Bacon');
});
});
Adds a new route and generates a controller and view. The name provided is used as state name for UI Router and as the route URL for ngRoute. Yeoman will then ask for the module to add the route to, the URL for the route (UI Router only), and the templateUrl. It will also generate an e2e test and a Page Object model for the new route.
Example:
yo ng-poly:route your-place
[?] Which module is this for?
[?] What's the URL for this route? (UI Router only)
[?] What's the templateURL for this route?
Updates app/module/module.js
:
'use strict';
/* @ngdoc object
* @name module
* @requires $stateProvider
*
* @description
*
*/
angular
.module('module', [
'ui.router'
]);
angular
.module('module')
.config(function ($stateProvider) {
$stateProvider
.state('module', {
url: '/module',
templateUrl: 'module/module.tpl.html',
controller: 'ModuleCtrl'
})
.state('yourPlace', {
url: '/yourPlace',
templateUrl: 'module/your-place.tpl.html',
controller: 'YourPlaceCtrl'
});
});
Produces e2e/your-place/your-place.po.js
:
/*global element, by*/
'use strict';
var YourPlacePage = function () {
this.text = element(by.tagName('p'));
this.heading = element(by.tagName('h2'));
};
module.exports = YourPlacePage;
Produces e2e/your-place/your-place_test.js
:
/*global describe, beforeEach, it, browser, expect */
'use strict';
var buildConfigFile = require('findup-sync')('build.config.js')
, buildConfig = require(buildConfigFile)
, YourPlacePagePo = require('./your-place.po');
describe('Your place page', function () {
var yourPlacePage;
beforeEach(function () {
yourPlacePage = new YourPlacePagePo();
browser.driver.get(buildConfig.host + ':' + buildConfig.port + '/#/yourPlace');
});
it('should say YourPlaceCtrl', function () {
expect(yourPlacePage.heading.getText()).toEqual('yourPlace');
expect(yourPlacePage.text.getText()).toEqual('YourPlaceCtrl');
});
});
Produces app/module/your-place-controller.js
, app/module/your-place-controller_test.js
, app/module/your-place.tpl.html
, and app/module/your-place.less
Currently, the module must have an existing state for another to be added.
The route generator can take URL and templateUrl options, as well.
yo ng-poly:route yourPlace --url=yourPlace --template-url=your-place
The URL will automatically be prepended with /
and and the templateUrl will be appended with .tpl.html
.
Generates a service and its test.
Example:
yo ng-poly:service cheap-or-good
[?] Which module is this for?
Produces app/module/cheap-or-good-service.js
:
'use strict';
/**
* @ngdoc service
* @name home.service:CheapOrGood
*
* @description
*
*/
angular
.module('module')
.service('CheapOrGood', function CheapOrGood() {
var self = this;
self.get = function get() {
return 'CheapOrGood';
};
});
Produces app/module/cheap-or-good-service_test.js
:
/*global describe, beforeEach, it, expect, inject, module*/
'use strict';
describe('CheapOrGood', function () {
var service;
beforeEach(module('module'));
beforeEach(inject(function (CheapOrGood) {
service = CheapOrGood;
}));
it('should equal CheapOrGood', function () {
expect(service.get()).toEqual('CheapOrGood');
});
});
Generates a value and its test.
Example:
yo ng-poly:value morals
[?] Which module is this for?
Produces app/module/morals-value.js
:
'use strict';
/**
* @ngdoc service
* @name module.constant:Morals
*
* @description
*
*/
angular
.module('module')
.value('Morals', 0);
Produces app/module/Morals-value_test.js
:
/*global describe, beforeEach, it, expect, inject, module*/
'use strict';
describe('Morals', function () {
var value;
beforeEach(module('module'));
beforeEach(inject(function (Morals) {
value = Morals;
}));
it('should equal 0', function () {
expect(value).toBe(0);
});
});
Generates a view and its style.
Example:
yo ng-poly:view nice
[?] Which module is this for?
Produces app/module/nice-view.tpl.html
:
<h2>nice</h2>
<p>{{ctrlName}}</p>
Produces app/module/nice-view.less
:
@bg-color: #E5E5E5;
body {
background-color: @bg-color;
}
Generates a Polymer element.
Example:
yo ng-poly:element gold-silver
Produces app/components/gold-silver/gold-silver.less
:
:host {
height: 100px;
width: 100px;
display: inline-block;
}
Produces app/components/gold-silver/gold-silver.html
:
<link rel='import' href='../polymer/polymer.html'>
<polymer-element name='gold-silver'>
<template>
<link rel='stylesheet' href='gold-silver.css'>
<div>{{name}}</div>
</template>
<script src='gold-silver.js'></script>
</polymer-element>
Produces app/components/gold-silver/gold-silver.js
:
/*global Polymer*/
(function () {
'use strict';
var element = new Polymer('gold-silver', {
name: 'gold-silver',
domReady: function () {
console.log('gold-silver');
}
});
return element;
}());
It is possible to override the configurations initially specified when yo ng-poly
was ran.
Each generator is able to take the following arguments. For example, yo ng-poly:module test --controller-as=true --markup=jade
will override the configuration settings for everything generated by this command.
Option | Possible Values |
---|---|
app-script | coffee, js |
markup | haml, html, jade |
style | css, less, scss, styl |
test-script | coffee, js |
controller-as | true, false |
pass-func | true, false |
named-func | true, false |
ng-route | true, false |
It's not recommended to mix ngRoute and UI Router, but it's possible.
This generator has support for the Controller As syntax. Yeoman will ask if this should be enabled when ng-poly:app
is ran.
This will generate controllers like:
'use strict';
/**
* @ngdoc object
* @name home.controller:HomeCtrl
*
* @description
*
*/
angular
.module('home')
.controller('HomeCtrl', function () {
var vm = this;
vm.ctrlName = 'HomeCtrl';
});
...and their tests like:
/*global describe, beforeEach, it, expect, inject, module*/
'use strict';
describe('HomeCtrl', function () {
var ctrl;
beforeEach(module('home'));
beforeEach(inject(function ($rootScope, $controller) {
ctrl = $controller('HomeCtrl');
}));
it('should have ctrlName as HomeCtrl', function () {
expect(ctrl.ctrlName).toEqual('HomeCtrl');
});
});
It'll also modify the state's controller like:
'use strict';
/* @ngdoc object
* @name home
* @requires $stateProvider
*
* @description
*
*/
angular
.module('home', [
'ui.router'
]);
angular
.module('home')
.config(function ($stateProvider) {
$stateProvider
.state('home', {
url: '/home',
templateUrl: 'home/home.tpl.html',
controller: 'HomeCtrl as home'
});
});
Lastly, views will be generated like:
<h2>home</h2>
<p>{{home.ctrlName}}</p>
The generator will ask when ng-poly:app
is ran if it should pass defined functions instead of defining inline.
This is currently only supported in JavaScript files.
If enabled, the app source code will pass functions, such as:
(function () {
'use strict';
/**
* @ngdoc object
* @name home.controller:HomeCtrl
*
* @description
*
*/
angular
.module('home')
.controller('HomeCtrl', HomeCtrl);
function HomeCtrl() {
var vm = this;
vm.ctrlName = 'HomeCtrl';
}
})();
The generator will ask when ng-poly:app
is ran if it should use named functions or anonymous functions. Named functions create a stack trace that is easier to understand.
This is only supported in JavaScript files.
If enabled, the app source code will have named functions, such as:
(function () {
'use strict';
/**
* @ngdoc service
* @name module.factory:Cake
*
* @description
*
*/
angular
.module('module')
.factory('Cake', Cake);
function Cake() {
var CakeBase = {};
CakeBase.someValue = 'Cake';
CakeBase.someMethod = function someMethod() {
return 'Cake';
};
return CakeBase;
}
})();
MIT