JSKit is a tiny library to end the problem of jQuery soup and $(document).ready()
mess. JSKit introduces a simple, clean, and easily testable way to architect basic javascript enhanced web pages. Based on a simple event system, JSKit allows your back-end application to seamlessly integrate your javacsript code with minimal coupling.
- lodash or underscore
Make sure that lodash or underscore is included and available globally.
<script type="text/javascript" src="path/to/lodash.js"></script>
-- or --
<script type="text/javascript" src="path/to/underscore.js"></script>
- Download the latest version
- Include Script
- include the
jskit.js
file in your application
- include the
<script type="text/javascript" src="path/to/jskit.js"></script>
The Application
is the interface to your JSKit components. To create an application you simply call createApplication
:
var App = JSKit.createApplication();
Every application has a Dispatcher
that allows registry and triggering of events throughout your JSKit application. In fact, JSKit is basically a thin wrapper around the Dispatcher
that allows you to coordinate your javascript with minimal friction.
App.Dispatcher.on("some-event-name", function() {
// handle some-event-name
});
App.Dispatcher.trigger("some-event-name");
Generally you will not have to manually register for events (App.Dispatcher.on
) since JSKit Controllers
register their own actions automatically.
The basic component of a JSKit application is a Controller. Controllers are basically objects that map actions to events. To create a controller, call createController
:
App.createController("Posts", {
actions: ["index", "show", "new", "edit"],
index: function() {
// handle index action
},
show: function() {
// handle show action
}
new: function() {
// handle new action
}
edit: function() {
// handle edit action
}
})
When the Application
creates a controller, it handles injecting the Controller with its Dispatcher
, creating the Controller's constructor, and instantiating an instance of the controller.
Controller instances are stored in Application.Controllers
by name. Looking at the "Post" controller example above we know that there is an instance of a "Post" Controller in
App.Controllers.Post
Controller constructors are stored on the Application
object itself with the suffix "Controller".
and the PostController
class constructor is at
App.PostController
The Controller instance is what Application
will use at runtime. The Controller's constructor is simply a convenience for creating clean states for testing.
Controllers are the main component of a JSKit Application
. Controllers accept a single argument of a Dispatcher. The Dispatcher is passed automatically when a controller is created with the Application.createController
method. Generally, you will only instantiate Controller
s in your tests. Having the Dispatcher injected makes testing them in isolation much easier.
Actions define to which events your controller responds. The Controller
uses the actions
array to map its methods to the Dispatcher's events. Actions are automatically mapped to the Dispatcher when the Controller
is instantiated. There are two ways to define an action mapping in your Controller
s:
To map an action by name, simply provide the method name as a string in the actions array:
App.createController("Posts", {
actions: ["foo"],
foo: function() {
// handle "controller:posts:foo" event
}
});
Sometimes you may want to map an action to a specific method, or you may want to bind two actions to the same method, to do so simply provide an object keyed by the action name with value of the method name:
App.createController("Posts", {
actions: [{ new: "setupForm" }, { edit: "setupForm" }],
setupForm: function() {
// handle "controller:posts:new" and "controller:posts:edit" events
}
});
When an action is mapped, it will be registered on the dispatcher for a specific event. The event that is registered depends on a few properties of the Controller
: namespace
, channel
, name
, and action
. The default values for these are:
namespace = ""
channel = "controller"
controllerEventName = "<Controller.name>" (lowercase and underscored)
action = "<action>
eventSeperator = ":"
So using the above examples, the event maps for the PostsController
are:
controller:posts:foo -> Controller.foo
controller:posts:new -> Controller.setupForm
controller:posts:edit -> Controller.setupForm
Note: undefined/empty values effectively removes that segment from the event name
By default, Controller
s have an empty namespace. If you wish to prefix the events the Controller
registers for with an namespace, set this property:
App.createController("Posts", {
namespace: "admin"
...
});
This will register all events with the namespace
prefix:
admin:controller:posts:foo -> Controller.foo
admin:controller:posts:new -> Controller.setupForm
admin:controller:posts:edit -> Controller.setupForm
The default channel for a Controller
is controller
. To change this simply set the channel:
App.createController("Posts", {
channel: "pages"
...
});
This will register all events with the channel
:
pages:posts:foo -> Controller.foo
pages:posts:new -> Controller.setupForm
pages:posts:edit -> Controller.setupForm
The Controller
is given a name
property by the Application.createController(name, attributes)
method. The controllerEventName
is automatically created by lowercasing and underscoring the name
to normalize event names. The PostsController
example has the name
"Posts" and a controllerEventName
of "posts". CamelCased names will have underscores between each uppercased word:
App.createController("CamelCase");
This would register all events with the controller controllerEventName
of camel_case
:
controller:camel_case:<action> -> Controller.<action>
The eventSeparator
property defines how the namespace
, channel
, name
, and action
will be joined to create an event name. You can change this by setting the eventSeparator
on the Controller
:
App.createController("Posts", {
eventSeparator: "."
...
});
This would register all events using "." as a separator:
controller.posts.foo -> Controller.foo
controller.posts.new -> Controller.setupForm
controller.posts.edit -> Controller.edit
Every controller has an automatically wired all
action. Other than being automatically wired, it is a simple action->event map like any other:
controller:posts:all -> Controller.all
The actionEventName
method allows you to get the full event name of a given action. It returns the namespace
, channel
, controllerEventName
, and the action
joined by the eventSeperator
:
var controller = App.createController("Posts");
controller.actionEventName("index"); // controller:posts:index
controller.actionEventName("show"); // controller:posts:show
controller.actionEventName("new"); // controller:posts:new
controller.actionEventName("edit"); // controller:posts:edit
controller.actionEventName(); // controller:posts
Every Controller has a className
. The className
property is generated by capitalizing the name
and adding the suffix "Controller". This property is used to display the Controller's constructor name in error output. Given the "Posts" Controller, the controllerName
would be: PostsController
.
JSKit is all about making testing javascript easier. JSKit itself comes with some handy tools for testing JSKit applications. When you create a controller with the Application.createController
method. The created Controller's constructor will be available on the Application
object using the Controller's className
. Using the "Posts" controller example:
App.PostsController
ALWAYS use the provided constructors when testing your Controllers to ensure you don't pollute your tests with mutated state.
The TestDispatcher
is used to help test JSKit Controllers. You should always use the TestDispatcher
when testing Controllers. To create a controller with the TestDispatcher
simply pass it to the Controller:
describe("PostsController", function() {
var subject;
var dispatcher;
beforeEach() {
dispatcher = new JSKit.TestDispatcher;
subject = new App.PostsController(dispatcher);
}
...
};
When your Controller registers events on the TestDispatcher
, your Controller's action methods will be enhanced with properties to determine if and how they were called:
...
it("calls index when the index event is triggered", function() {
dispatcher.trigger(controller.actionEventName("index"));
expect(controller.index.called).to.equal(true);
expect(controller.index.callCount).to.equal(1);
expect(controller.index.calls[0].args).to.be.an("Array");
expect(controller.index.calls[0].args).to.be.empty();
});
...
};
Most of the time, you will not need the extra functionality provided by spies and you simply want to see if a Controller's actions are wired up correctly. You can do this simply by calling hasAction
with the controller and action you wish to test:
...
var subject;
var dispatcher;
beforeEach() {
dispatcher = new JSKit.TestDispatcher;
subject = new App.PostsController(dispatcher);
}
it("has an index action", function() {
expect(dispatcher.hasAction(subject, "index")).to.equal(true);
});
...
- Fork it
- Clone it locally
- Set the upstream remote (
git remote add upstream git@github.com:daytonn/jskit.git
) - Install the dependencies with npm (
npm install
) - Run the specs with
npm test
- Run the example app with
npm start
- Create your feature branch (
git checkout -b my-new-feature
) - Commit your changes (
git commit -am 'Add some feature'
) - Ensure remote is updated (
git remote update
) - Rebase from upstream (
git rebase upstream/master
) - Push to the branch (
git push origin my-new-feature
) - Create new Pull Request