generator-thorax
v0.0.8
Published
A Thorax generator for Yeoman
Downloads
37
Maintainers
Readme
Thorax Generator
Generate a new Thorax application. Includes some configurable options and some opinions.
$ npm install -g yo generator-thorax
$ yo thorax desired-application
$ cd desired-application
$ grunt
For a snazzier development environment, it's recommended that you install the Thorax Inspector Chrome Extension before getting started.
Stack
- Thorax, a framework for building better Backbone applications
- Handlebars, a framework that works out of the box with Thorax for writing dynamic JavaScript views in HTML, variables are inserted using the
{{ handlebars }}
syntax - Grunt, a framework for declaratively defining CLI tasks, e.g., for running tests, compiling stylesheets, creating a production build of your app.
- Bower, a package manager for the browser making it easy to pull in 3rd party libraries
- Your choice of LESS/Stylus/SASS for more productive stylesheets
- Automated JavaScript testing stack:
- Mocha w/ Chai - for BDD style tests
- Karma, a framework for piping test results to the command line for a rapid testing feedback loop
- Built in support for Travis CI
- Require JS, a framework for working with modules in the browser
hbs!
Require JS plugin for 'just in time' compiling of Handlebars templates for rapid feedback during development. Templates will be pre-compiled when creating a production build via the r.js optimizer provided as a grunt taskcs!
Require JS plugin for 'just in time' compiling of CoffeeScript files, allowing you to mix CoffeeScript and JavaScript files at will, enjoy rapid feedback during development, including fast tests. CoffeeScript will be pre-compiled to JavaScript when creating a production build via the r.js optimizer provided as a grunt task
- Yoeman Generators
- generating a new application
- generate a new router, view, model, collection or helper with an example test written in Mocha + Chai
- Live Reload, for automated browser reloading during development
- Built in deployment strategies for Heroku and Nodejitsu, for quick and efficient deployment of your app to the cloud.
Directory Structure
bower_components
Installed third party libraries. Managed by Bower, do not modify directly.css
Regardless of whether you are using a CSS superset (LESS / Stylus / SASS) or plain CSS, this is where your CSS should live. This will get copied or compiled topublic/css
when runninggrunt build
js
collections
models
routers
templates
views
collection-view.js
collection.js
layout-view.js
require-config.js
Require JS data-main entry pointmodel.js
view.js
node_modules
public
The connect server will serve this directory at/public
css
Generated or copied CSS from thecss
directory, do not modify.fonts
Any web fonts for your application, safe to modify.img
Any images, safe to modifyindex.html
Do not modify, and add new libraries to/require-config.js
tasks
Put your custom grunt tasks in hereoptions
Installed grunt options are looked up here, by file name
bower.json
Third party library management via BowerGruntfile.js
Grunt build file.grunt
will run the default task specified in this file.package.json
Anynpm
dependencies needed to build / compile your application.README.md
A copy of this very file!
Grunt Tasks
Every generated application comes fully loaded with an amazing development and testing environment. Furthermore, when you're ready to show the world what you've made, building a production version of your application and deploying it to the cloud is painless and even testable in multiple browsers.
Development
** Run grunt
**
The best way to get up and running in development is to run grunt
from the command line. This will boot a connect server and open up a browser to http://localhost:8000/
.
Connect will also serve your tests at http://localhost:8000/test
.
LiveReload will reload both /
and /test
when changes are made to any file within your directory while grunt
is running in the background.
Testing
** Run grunt autotest
**
The easiest way to run your tests is perhaps just to stick with the test runner page provided when running the default grunt
task discussed above.
However, every application also comes with karma setup to run your mocha tests inside your terminal window out of the box.
To try this out run grunt autotest
from the command line. By default karma is setup to run tests in Chrome which is the most reliable and fastest way to gain rapid feedback from a real browser during development.
Advanced Testing
** Run grunt
in one window and karma start
in a second**
Another option for running your tests is to skip grunt altogether and simply run karma start
from your terminal window. This option will provide the fastest feedback loop during development and is recommended once your comfortable running your tests completely on the command line.
You still may want to run grunt
in a second terminal window to hook up livereload to your application when adjusting styles or manually clicking through your app, but running karma start
will shave roughly a second off of each test run making for a very tight feedback loop.
Debugging Karma
To gain access to the console and debugging tools provided by Chrome, click on the Debug
button at the top of the window booted by karma. Within here you'll have full access to your Thorax application, even though the page itself is white.
Production
** Run grunt production
**
When you're ready to build concatenated and minified versions of your css and JavaScript source run grunt production
. The output can be found in dist/
and a browser window will be booted so you can play with the production version of your application before deployment.
Test Your Production Build in Multiple Browsers
** Run grunt deploy
**
Before deploying your application run grunt deploy
to run your tests against the built version of your application inside the following browsers:
- Chrome
- Firefox
- Safari
- PhantomJS
To test your application in IE you'll need a Windows box. Run npm install karma-ie-launcher --save-dev
and add IE
to the array of browsers within the deploy task inside tasks/options/karma.js
. However, using Sauce Labs in conjunction with the Karma Sauce Launcher is the recommended and effective solution to this common problem.
Phantom JS with Karma or Mocha-PhantomJS
If you prefer to manually run your tests via the command line during development every generated app also provides to ways to accomplish this:
grunt test
will run your tests with Phantom JS via karmagrunt test-mocha-phantomjs
will run your tests with Mocha PhantomJS, which also outputs to your terminal but provides a nicer UI than Karma in a format that reads like documentation, similar to what you'll find in the browser window when runninggrunt
or when running mocha tests in node js.
Continuous Integration with Travis CI
Use github, flip on Travis CI switch
If you host your application as a repo on github(public or private), you can optionally turn on Travis CI support. Every application is generated with a .travis.yml
file, so once the switch has been turned on, it'll work out of the box.
Deployment
Every generated application comes setup ready to deploy to either Heroku and/or nodejitsu.
To deploy to heroku:
- Make sure you have the Heroku Toolbelt installed and you have logged into your account via
heroku login
. - Build your app:
grunt production
. This will output a minified version of your app todist/
which heroku will serve up. - Make sure you have committed your changes to git:
git add .
git commit -m "commit message here"
- Run
heroku create
from the command line. - Run
git push heroku master
- Run
heroku open
to see your app live on the web!
Note: During deployment run foreman start
to test your application. Open your browser to port 5000
to view your app.
To deploy to nodejitsu:
- Install the tool:
npm install jitsu -g
- Create an account:
jitsu signup
- Login:
jitsu login
- Build your application:
grunt production
- Deploy:
jitsu deploy
That's it. Your package.json
file already has a scripts
option named start
that nodejitsu will use to boot the application.
CoffeeScript Support
Out of the box all generators can optionally output CoffeeScript.
When generating a new application, the generator will simply ask you if you want CoffeeScript or JavaScript and act accordingly by outputting the generated application with the files of the chosen format.
Running a sub generator, like yo thorax:router
, will look for the presence of at least one .coffee
file within js/
and choose whether to output CoffeeScript or JavaScript accordingly.
To force CoffeeScript output, append --coffee
to any of the following commands.
To force JavaScript output, append --js
to any of the following commands.
Requiring a CoffeeScript Module
Regardless of what a generator outputs, .coffee
files are always supported out of the box.
To require a CoffeeScript module instead of a JavaScript module simply prefix the module name with cs!
. For example: (inside a JavaScript module or CoffeeScript module)
require(['cs!my-module'], function(myModule) { ... })
Prefixing a module name with cs!
will compile the CoffeeScript file on the fly removing the need for pre-compilation during development.
When creating a production build with grunt production
however, the r.js optimizer will pre-compile the CoffeeScript files to JavaScript before doing concatenation and minification.
Handlebars Support
Similar to how CoffeeScript files are compiled on the fly using the cs!
Require JS plugin, Handlebars files ending in .hbs
also benefit from not having to run through a compile phase.
To require a handlebars template for use in a view, prefix the name of the module with hbs!
. For example, inside a view or collection view:
require(['hbs!my-template'], function(template) { ... })
Handlebars files will be treated as JavaScript templates therefore leave off the trailing .hbs
when requiring a template within a view or collection-view.
One current caveat with this approach is that the template
option within a view or collection-view will always need to be manually set.
Generators
Application Generator:
Generate a new Thorax Application using the software stack mentioned above:
$ yo thorax app-name
Sub Generators:
Use these within an existing Thorax application:
$ yo thorax:router name
$ yo thorax:view name
$ yo thorax:model name
$ yo thorax:collection name
$ yo thorax:collection-view name
The name
argument may include a directory path, such as todo-list/index
:
$ yo thorax:router todo-list
$ yo thorax:view todo-list/index
From Zero to Todos
Install the generator
$ npm install -g yo generator-thorax
Generate your application
$ yo thorax todo-list
[?] Would you like to generate the app in a new directory? Yes
[?] Choose a css pre-processor: (Use arrow keys)
❯ Less with bootstrap (default choice)
Sass
Stylus
Plain CSS
[?] Would you like to use CoffeeScript? No
[?] Would you like to use Zepto in place of jQuery (Zepto is best for mobile apps)
[?] Would you like to setup your project with a sample application? (Use arrow keys)
Hello World
Todo List
❯ None
...and then $ cd todo-list
.
Note that had you chosen Todo List
for the sample application above, you would have generated the completed version of the app we're about to build. You may want to do this when we're finished to check your work.
Sub Generators
A sub-generator is a generator that you use within your app, after it has been initially created.
Both application and test files are output from all sub-generators and CoffeeScript files will be output in place of JS files if one or more CoffeeScript files are present within your application. To override this settings pass --js
or --coffee
at the end any sub-generator command.
In the following section, we'll build a Todo App using sub-generators. Note that test files will be omitted for brevity.
Generate a View:
$ yo thorax:view todo-list/index
This generates two new files, a view and a matching template:
create js/views/todo-list/index.js
create js/templates/todo-list/index.hbs
views/todo-list/index.js
should contain the following code:
define([
'view',
'hbs!templates/todo-list/index' // IMPORTANT, prefix with hbs! when requiring a template
], function (View, template) {
return View.extend({
name: 'todo-list/index',
template: template // passed as `template` arg above, must be set
});
});
Had we been using CoffeeScript in our application the output would instead look like:
define [
'cs!view', # IMPORTANT, prefix with cs! when requiring a module written in CS
'hbs!templates/todo-list/index'
], (View, template) ->
View.extend
name: 'todo-list/index'
template: template
Those familiar with RequireJS will be thrilled to see the define() call above, and those who aren't should read the section about require jsfor short summary.
Generate a Router:
$ yo thorax:router todo-list
This will generate one new file...
create js/routers/todo-list.js
...into which the following code will be inserted:
define([
'backbone'
'views/root',
], function (Backbone, RootView) {
return Backbone.Router.extend({ //plain Backbone. Thorax doesn't touch the router.
routes: {
}
});
});
Now that we have our files, we can start editing them. Let's first get something up on the screen. We'll add our index route to js/routers/todo-list.js
, and an index
function Backbone will fire for us when that route is hit. Inside of that function we'll create an instance of our view, which we'll be able to access because we've passed it into scope by way of the dependencies array...
define([
'backbone', //included because we're calling the Backbone object below. While Backbone depends on Underscore, we wouldn't pass that in unless we were going to use "_" inside of our callback.
'views/root', //this is what's going to get attached to the DOM. More on that soon.
'views/todo-list/index' //this is the view class we're going to instantiate below...
], function (Backbone, RootView, TodoListIndexView) { //...but to make it available we're going to need to add it and pass it into our callback, here we've named it TodoListIndexView
return Backbone.Router.extend({
routes: {
"": "index" //add an index route
},
index: function(){
var view = new TodoListIndexView({}); //Hey! I'm a view getting instantiated! My template will be rendered
RootView.getInstance().setView(view); //Nuke whatever was in the {{layout-element}} element in root.hbs (and do memory management), replace it with the template rendered by the line above.
}
});
});
...and then give templates/todo-list/index.handlebars
something to render:
<p> Arrrr! I'm a pirate with a handlebar mustache. </p>
Finally, edit js/main.js
. To the list of dependencies required by this module, add 'routers/todo-list'
. Add it before 'helpers'
. Also make sure to inject the dependency into the module as an argument to the factory function. Then instantiate the router inside the initialize function, before calling next();
.
The final changes should look like:
require([
'jquery',
'backbone',
'views/root',
'routers/todo-list', // ADDED
'helpers',
], function ($, Backbone, RootView, TodoListRouter) { // MODIFIED
// Initialize your routers here
new TodoListRouter(); // ADDED
...snip...
Run grunt
to build and view your application
Now that we have something on screen, let's get some data on the screen and finish the rest of our todo list.
Render a Collection
To implement a todo list, we need to create a collection and set it on the view. Unlike a Backbone.View
instance, a Thorax.View
instance does not have an options
object. All properties passed to the constructor are set on the instance and also become available inside of the handlebars template. We'll now update js/routers/todo-list.js
...
define([
'backbone',
'collection', // ADDED
'views/root',
'views/todo-list/index'
], function(Backbone, Collection, RootView, TodoListIndexView) { //and also add the 'Collection' arg here
return Backbone.Router.extend({
routes: {
"": "index"
},
index: function() {
var collection = new Collection([{ //here we instantiate the collection and populate it with a single model with two properties, 'title' and 'done'
title: 'First Todo',
done: true
}]);
var view = new TodoListIndexView({
collection: collection //here we set the 'collection' property of our view to the collection we just instantiated
});
RootView.getInstance().setView(view);
}
});
});
To display the collection we will edit templates/todo-list/index.hbs
and use the collection
handlebars helper to loop through each model in the collection:
{{#collection}}
render the code between the opening and
closing collection tags for each model
in the collection
{{/collection}}
Beautifully, all of the properties of the associated model are available in the helpers (see {{title}}
below). A tag
option may be specified to define what type of HTML tag will be used when creating the collection element:
{{#collection tag="ul"}}
<li>{{title}}</li>
{{/collection}}
Since we want to be able to mark our todos as done and add new ones, we will add a checkbox to each item in the collection and a form to make new items at the bottom. Our templates/todo-list/index.hbs
should now look like:
{{#collection tag="ul"}}
<li {{#done}}class="done"{{/done}}>
<input type="checkbox" {{#done}}checked{{/done}}>
{{title}}
</li>
{{/collection}}
<form>
<input name="title">
<input type="submit" value="Add">
</form>
We'll also create a style sheet called css/todo-list.css
, which will be automatically applied ONLY to the view with the same filename. Populate it with the following code:
.done {
text-decoration: line-through;
}
To get our new css file to load we will add the link to our public/index.html file. It should look like this:
<head>
<meta charset="utf-8">
<title>Thorax-Todo</title>
<link rel="stylesheet" type="text/css" href="css/base.css">
<link rel="stylesheet" type="text/css" href="css/todo-list.css"> <!-- Add this line to reference our custom stylesheet. -->
</head>
<body>
View Behaviors
In order to add new items to the list we should listen to the submit
event on form
elements in our view. We can use the events hash in js/views/todo-list/index.js
:
events: {
"submit form": function(event) {
event.preventDefault();
var attrs = this.serialize();
this.collection.add(attrs)
this.$('input[name="title"]').val('');
},
The serialize
method will return key value pairs of all attributes in form elements on the page. Since we had an input with a name of title
attrs will be set to: {title: "your todo"}
. When using the collection
helper or a CollectionView
, Thorax adds, removes and updates views in the collection as appropriate. When we add
a new model to the collection the view will automatically update.
'change input[type="checkbox"]': function(event) {
var model = $(event.target).model();
model.set({done: event.target.checked});
}
We also need to listen for a change in a checkbox so we can mark a model as done. Thorax extends the jQuery or Zepto $
object with three methods: $.view
, $.model
and $.collection
. They will retrieve closest bound object to an element. In this case, a model was automatically bound to the li
tag passed into the collection
helper in the template. Now that we have a reference to the model
we can update it and the view will automatically update.
Our finished js/views/todo-list/index.js
file should look like:
define([
'view',
'templates/todo-list/index'
], function(View, template) {
return View.extend({
name: "todo-list/index",
template: template,
events: {
"submit form": function(event) {
event.preventDefault();
var attrs = this.serialize();
this.collection.add(attrs);
this.$('input[name="title"]').val('');
},
'change input[type="checkbox"]': function(event) {
var model = $(event.target).model();
model.set({done: event.target.checked});
}
}
});
});
And that's a finished non-persistent todo list application! For more complex examples and tutorials using the thorax framework, see the tutorials on the Thorax homepage
Require JS
Instead of compiling requiring script tags in your HTML, Require JS is used to load AMD modules instead of plain JavaScript files.
Adding a New Dependency to Your App
To add a new library to your app. Here are the steps:
- Install the library via bower by running
bower install library-name --save-dev
. If the library is not updated regularly from bower try pointing the version to a github tag, for example:"library-name": "some-github-profile/library-name#v0.1"
inside of yourbower.json
file. - Open
require-config.js
at the root of your application and add apath
that points the the AMD module version of the library installed by bower. If the library does not have an AMD module version(many don't) you'll need to setup ashim
settings, which tells Require JS to wrap a normal JS file as a module you can require. Also make sure to set thedeps
array for any libraries the library depends on. For an example, take a look at how Thorax itself is setup withinrequire-config.js
. - Inside an existing or new module, require the library by adding an item to the array of required modules - the name you'll use here is the name of the
path
you setup in step 2 above. Then, pass a new argument into the callback function, name the argument whatever you'd like, but remember that uppercase names should be reserved for modules that export constructor functions. When all the modules that this module depends on resolve, the callback function will be called and the argument representing the exported object from the module will be available for use inside.
For in depth information on Require JS make sure to check out http://requirejs.org/.