Skip to content
unscriptable edited this page Sep 3, 2011 · 9 revisions

Plugins

The AMD specification allows loading of non-module javascript or other resources such as html templates, css files, and json files. It does this via plugins. Plugin dependencies are delineated by an exclamation character ("!") as follows (without the spaces):

plugin-id ! resource-id

"resource-id" typically looks very module-like and the loader will interpret it as a module id (unless the plugin tells the loader it shouldn't). Some typical plugin references:

text!mylib/page-template.html
link!mylib/css/reset.css
css!http://ajax.googleapis.com/ajax/libs/dojo/1.6/dojo/resources/dojo.css

curl.js <3 plugins!

curl.js supports the AMD specification for plugins. This means you can use any compatible plugin module and use it with curl.js. curl.js also comes with its own collection of excellent plugins. The following plugins are included with curl.js 0.5.x and more are in the works:

  • [[js!|plugins/js]] - loads non-AMD javascript files (non-blocking!)
    • supports ordering of javascript files
    • can prefetch, but not execute a file for later execution
  • link! - loads css files using a LINK element (non-blocking!)
    • ultra-fast and light
  • text! - loads text files using XHR (non-blocking!)
    • for cross-domain text resouces, consider using JSON-P
  • domReady! - waits for the dom to be ready for manipulation
    • there is also a "curl/domReady" module if you prefer to receive a callback directly
  • async! - defers resolution of an AMD module that returns a promise
    • module does not have to be a true promise: any module that returns a then(callback, errback) method will work
  • css! - a more flexible css loader
    • allows css to be built into javascript (when using cram)
    • avoids 31-stylesheet limit in IE6-8
    • normalizes some common, cross-browser stylesheet methods
    • waits for css files to be active before proceeding (option)

An example of a simple module using plugins:

define(
	['mylib/BaseView', 'css!mylib/CoolView.css', 'text!mylib/CoolView.html'],
	function (BaseView, stylesheet, template) {
		// This is our custom view, CoolView, that extends BaseView
		// Tt comes with it's own stylesheet (CoolView.css) and
		// template (CoolView.html).
		function CoolView (domNode) {
			this.template = template;
			this.stylesheet = stylesheet;
			BaseView.apply(this, domNode);
		}
	}
);

Plugins may also be used during the bootstrap phase of your web app:

curl({ baseUrl: '/js/' })
	// load our non-module javascript and css
	.next(['js!libs/jQuery1.6/jQuery.js!order', 'js!libs/jQueryUI1.3/jQueryUI.js!order', 'link!css/screen.css'])
	// load our bootstrap module and wait for dom-ready
	.next(['mylib/boot', 'domReady!'], function (boot) {
		// go!
		boot.init();
	});

Notice that the css and domReady dependencies aren't referenced in the callback function. This is typical with css resources and domReady. It's unusual to need a reference to a stylesheet. domReady! returns nothing (undefined). Therefore, these can be safely omitted from the dependency list.

Plugin options

Some plugins have optional features. These may be applied where the plugin resource is defined or may be defined globally. Global options are specified in a "plugins" property of the curl config object as follows:

curl({
	plugins: {
		// js! plugin global options
		js: {
			prefetch: false
		},
		// css! plugin global options
		css: {
			nowait: true
		}
	},
	baseUrl: '/path/to/myapp/'
});

Please refer to each plugin for information about its options.

Build you own plugins!

It's incredibly easy to create your own plugins. Depending on the complexity of your plugin and whether you want it to build resources into the compiled javascript files, you'll only need to implement up to 4 API methods. TODO: page about creating plugins

Plugin namespacing

Plugins are just regular AMD modules that expose a handful of standard methods. However, in the examples above, the plugins aren't namespaced. The plugins reside in curl's folder hierarchy at curl/plugin/. So why don't we have to reference plugins using their full namespace like in the following define()?

define(['curl/plugin/text!mylib/CoolView.html'], function (template) {});

curl.js has a convenience config property, "pluginPath", that defaults to curl/plugin/. All "naked" plugins (un-namespaced) are assumed to be located in the pluginPath folder. Not only is this convenient, but it also provides an implmentation-agnostic method for referencing plugins.

Not all AMD loaders that support plugins will have a "pluginPath" config property. They should, however, have a way to support un-namespaced plugins.

If you create your own plugins and want to reference them "naked", you have several options:

  1. place your plugin in the same folder as curl.js's plugins
  2. change "pluginPath" to point at your plugins folder (and move curl.js's plugins there if you use them)
  3. place your plugins' modules at the root of your application (i.e. where baseUrl points)
  4. use the "paths" config property to translate plugin paths like this:
curl({
	paths: {
		myplugin: 'mylib/plugins/myplugin'
	}
}, ['myplugin!path/to/cool/resource'], bootFunc);

## ### 1.

Clone this wiki locally