Skip to content

karlvr/consistent.js

Repository files navigation

Consistent.js

Consistent is a small and simple Javascript framework to enable an abstract model and controller to be connected to the DOM.

Build Status

Introduction

Use Consistent to create a scope, and then bind DOM nodes to it. Consistent inspects the DOM nodes (and their children) to learn how to relate them to the scope.

The scope starts with no properties. You add properties to the scope and then apply them to the DOM. In your HTML markup you add ct... attributes to declare to Consistent how to use the scope’s properties. You can also use data-ct... instead of ct... if you prefer.

The scope can be created programmatically, or automatically using declarations in your markup.

The scope contains a $ property in which Consistent keeps its functions and scope. For example, when you want to apply the scope you call scope.$.apply(). This $ property separates the properties you add to the scope from Consistent, so you can add properties with any other name. Note that this $ is nothing to do with jQuery and doesn’t interfere with it.

The scope may contain scalar values, such as booleans, strings and numbers, and also value functions that return a calculated value.

The scope has a controller object that contains any event handler functions you create. The controller can be a custom class, or you can add functions to the default empty controller.

Consistent includes a jQuery plugin, and the examples below show this approach. Consistent does not however require jQuery and can be used without it.

<script src="src/consistent.js"></script>
<script src="src/jquery.consistent.js"></script>

Or use a minified and combined version. The minified and combined script for Consistent and the jQuery plugin is around 9KB.

<script src="lib/consistent-for-jquery.min.js"></script>

Consistent is designed with security in mind. Be sure to read the Security section, so you understand the potential attacks and how Consistent mitigates them.

License

Consistent is released under the Apache License, Version 2.0.

Browser compatibility

Consistent works in all modern browsers. It has also been tested, and works in its entirety, in IE 6 and later.

Consistent.js uses BrowserStack for cross-browser testing.

Features

Substitution

Set the contents of an h1 element with the title property in the scope.

<h1 ct="title"></h1>

Now create a scope using the jQuery plugin, and assign a value to it.

var scope = $("h1").consistent();
scope.title = "Consistent.js";
scope.$.apply();

The h1 element will now have its body changed to "Consistent.js".

Notice that after changing properties in the scope you need to call apply to instruct Consistent to update the DOM. You can also apply changes to the scope like this, which is equivalent:

var scope = $("h1").consistent();
scope.$.apply(function() {
	this.title = "Consistent.js";
});

Or even the following, as most things in Consistent are chainable:

$("h1").consistent().$.apply(function() {
	this.title = "Consistent.js";
});

Note that if a scope property is undefined, Consistent does not change the DOM.

How Consistent applies the scope value to the DOM depends upon the element. For most elements the innerHTML is set to the scope value. There are the following exceptions:

  • <input>, select and textarea elements have their value set, or are checked, as appropriate. See Forms below.
  • <img> has its src attribute set

Templating

Consistent supports pluggable templating engines. The examples use Hogan. Any templating engine that provides compile(string) and render(object) methods will work.

<h1 ct-tmpl="Welcome to {{name}}"></h1>

Now configure Consistent to use Hogan as its templating engine, and populate the scope.

Consistent.defaultOptions.templateEngine = Hogan;
var scope = $("h1").consistent();
scope.name = "Consistent.js";
scope.$.apply();

You can also reference templates by an id, rather than writing them inline:

<h1 ct-tmpl-id="h1-template"></h1>

<script id="h1-template" type="text/x-hogan-template">
	Welcome to {{name}}
</script>

Note that Consistent will re-render the templates and thus recreate the DOM nodes each time the scope is applied.

If you need to create a large DOM structure and then have it bound to a scope, consider creating it first using templating and then binding it with Consistent.

Security warning: The text inside the <script> element above is parsed by the templating engine used by Consistent. Therefore if you inject unsafe content, such as user generated content, you need to ensure that it is properly escaped so that it is not interpreted by the templating engine. In the case of Hogan you cannot easily escape content. You need to ensure that unsafe content does not contain {{s. Fortunately this issue only exists for Consistent in this referenced template case, as all other parsed areas are inside ct and ct-* attributes. Please read the Security section, and please keep this issue in mind.

Visibility

Consistent can show and hide nodes based on the scope.

<h1 ct-show="showTitle">My title</h1>

You can also use ct-hide to hide the element when the scope property is truthy.

Now create a scope and set the showTitle property. Consistent will show or hide the element using a display:none style. Consistent also restores the old value of display when re-showing, in case it was set to something specifically.

var scope = $("h1").consistent();
scope.showTitle = true;
scope.$.apply();

Note: There is an exception relating to handling of undefined with ct-show as of v0.14. If the scope property referenced in ct-show is undefined, Consistent will hide the element.

Animation

You can override the behaviour of showing and hiding elements. For example, you may want to fade elements in and out. See the Options section for more information.

Value functions

As well as adding scalar values to the scope, you can also add value functions. In this case the function is executed each time the scope is applied and its return value is used. Value functions allow a scope property to be calculated dynamically.

<div id="container">
	<p>The number of people is <span ct="numberOfPeople">&nbsp;</span>.</p>
</div>

The &nbsp; above is sometimes necessary for IE 6 (is anyone still?) which will otherwise collapse the whitespace around the <span> and you may get strange spacing.

var scope = $("#container").consistent();
var people = [ "Albert", "Bob", "Carl", "Donald" ];
scope.numberOfPeople = function() {
	return people.length;
};
scope.$.apply();

The value function gets called with this set to the scope it is declared in, and the first argument is the scope where the request for the value originates, which is important when using parent and child scopes. If the value function returns undefined then no changes will be made to the DOM, as for other undefined scope properties.

Updating value functions

When the scope is populated from the DOM using the scope.$.update function, or when a scope property is set manually using the scope.$.set function, and the scope contains a value function for the affected property; the value function is called with two arguments, the scope where the “set” originates, and the new value. Your value function can simply ignore this form if it doesn’t support updates.

var numberOfPeople = 5;
scope.numberOfPeople = function(localScope, newValue) {
	if (typeof newValue !== "undefined") {
		numberOfPeople = parseInt(newValue);
	} else {
		return numberOfPeople;
	}
}

Form elements

Form elements are automatically associated with the scope property with the same name as the element, and Consistent updates the form field’s value from the scope.

<input type="text" name="email">
<input type="checkbox" name="optin">

Note that we don’t explicitly specify the scope property, it defaults to the name. You can explicitly specify the scope property using the ct attribute.

Now create a scope and set the elements’ values.

var scope = $("input").consistent();
scope.email = "example@example.com";
scope.optin = true;
scope.$.apply();

All form elements are supported, including:

  • text fields
  • checkboxes
  • radio buttons
  • select (single and multiple selection)
  • textareas

Checkboxes are usually represented by a boolean value in the scope. For groups of checkboxes with the same name they can be thought of as an array of values (the value attribute from the checkbox). If you have a boolean value in the scope, Consistent will not change it, but you may get some strange results if you have multiple checkboxes bound to the same property (with the same name with the default property binding). If the scope contains an array or no value for a checkbox’s bound property then Consistent will create an array. Consistent can also cope with scalar values for a checkbox, in which case they are matched to the checkbox’s value. Note that in this case Consistent may convert the property to an array if there are multiple checkboxes bound to the same property.

For <select> elements that can have multiple options selected, the scope property can be an array.

Controlling updating

Consistent automatically listens to the change event on form elements. When the change event fires, Consistent updates the scope with that element and then applies the scope. Note that the update is just for the element that fired the change event, it is not for all of the scope’s DOM nodes as it is if you call scope.$.update(). You can turn off this behaviour by setting autoListenToChange to false in the options object, either when the scope is created or when you bind the form elements.

You can control updating on an element by adding a ct-update attribute. Valid values are auto (the default), auto-nokey, noauto, disabled:

  • auto listen to the change event and update the scope
  • auto-nokey the same as auto, except don't update on key events
  • noauto don't listen to change events, but still update when the scope is updated
  • disabled don't listen to change events, and do not update when the scope is updated
<input type="text" name="email" ct-update="auto">

Disabled and Read only

You can control the disabled and readOnly properties of form elements.

<input type="text" name="email" ct-disabled="locked">
scope.locked = true;

It is often useful to use a value function in this case, so that the disabled state of the form element is calculated dynamically each time the scope is applied.

The above shows how to control the disabled property, the full list is:

  • ct-disabled
  • ct-enabled
  • ct-readonly
  • ct-readwrite

Note that for each of disabled and readonly there is the opposite so that you can best fit the option to the model.

Select options

You can set the options array for a <select> element from the scope.

<select name="product" ct-options="products"></select>

Now set the options array either as an array of scalar values, such as strings:

scope.products = [ "", "Lamp", "Bucket", "Axe" ];

Or as an array of objects that separate the text and value:

scope.products = [
	{},
	{ text: "Lamp", value: "lamp" },
	{ text: "Bucket", value: "bucket" },
	{ text: "Axe", value: "axe" }
];

You can also include label and disabled properties in the objects to set those properties in the created options.

You can of course bind the selected option as well, e.g. scope.product = "bucket";.

Events

Consistent can add event listeners to DOM nodes which call functions in the scope’s controller. The controller is an object that holds the scope’s event handler functions. See the Controllers section below for more information on controllers.

<a href="#" ct-on-click="handleClick">Click me</a>

Now create a scope and add the click handler.

var scope = $("a").consistent();
scope.$.controller("handleClick", function(scope, ev) {
	ev.preventDefault();
	alert("Click!");
});

Shortcut

There is a shortcut for binding events, the ct-do declaration. It behaves like ct-on-..., but binds a default event based on the type of element. It chooses the click event for most elements, e.g.:

<a href="#" ct-do="handleClick">Click me</a>

But the following special cases apply:

  • <input>, <textarea> and <select> elements bind the change event
  • <form> elements bind the submit event

Event handler functions

Event handler functions are called with this set to the controller object.

The event handler function arguments are:

  • The scope in which the event occurred
  • The Javsacript event object
  • The DOM element that is the source of the event

The scope in which the event occurred is important when you use child and parent scopes.

scope.$.controller("handleClick", function(scope, ev, dom) {
	scope.clickCount++;
	scope.$.apply();
});

This handler function makes a change to the scope and then calls apply to apply the scope to the DOM.

A nice style to use in event handlers is to nest scope modifications inside the optional function argument to scope.$.apply(). Inside that function this will be the scope, which may be tidier (especially if you’re changing lots of values), e.g.:

scope.$.controller("handleClick", function(scope) {
	scope.$.apply(function() {
		this.clickCount++;
	});
});

Note that we don’t need to call apply on the scope after adding handlers to the controller, as event listeners are added when the DOM nodes are bound to the scope based on the declarations in the DOM; just make sure the handler functions are defined by the time they are invoked.

The scope parameter is the scope in which the event occurred. This may not be the controller’s scope, it may be a child scope. In the event handler you can decide whether you want to operate on the child scope or not. You can always get a reference to the controller’s scope using this.$.scope(), see Controllers below.

Controllers

Each scope has a controller. When you create a scope you can specify a constructor function to create the controller, otherwise an empty object is used.

The $ object from the scope is always added to the controller, so the controller can access and control the scope. The best way for the controller to access the scope is using this.$.scope(). The controller can tell the scope to apply by simply calling this.$.apply().

You can access the controller using the scope.$.controller() function. To add a function to the controller, use the scope.$.controller(name, function) function, e.g. scope.$.controller("handleClick", function() { ... });.

You can add anything to the controller. It is a good place to encapsulate all of the code and state related to the scope, but that is not involved in binding to the DOM.

Custom controller classes

The custom controller class is specified using a constructor function that takes one argument, the scope.

function MyController(scope) {
	scope.title = "My title";
}

MyController.prototype.handleClick = function(scope, ev) {
	scope.title = "I’ve been clicked!";
	scope.$.apply();
	ev.preventDefault();
};

var scope = $("div").consistent(MyController);

Repeating blocks

If your scope contains array values, you can repeat blocks of DOM nodes to represent them.

<ul>
	<li ct-repeat="people" ct="name"></li>
</ul>
var scope = $("ul").consistent();
scope.people = [
	{ name: "Alfred" },
	{ name: "Bob" },
	{ name: "Carl" }
];
scope.$.apply();

This will result in a list containing three <li> elements, one for each of the people in the scope.people array. If you change the array, the DOM will be updated.

scope.people.push({ name: "Donald" });
scope.$.apply();

Or remove an item:

scope.people.shift();
scope.$.apply();

Consistent only creates new nodes when new items are added to the array. So any changes you make to the DOM outside of Consistent will be preserved, such as applying classes.

Consistent creates a child scope for each repeated block, and the object in the array becomes its scope. Therefore each object in the array will have a $ property added containing Consistent’s scope functionality. As the objects in the array are the child scopes, you can access the child scopes if you need to via the original array in the original scope.

You don’t need to call apply on the child scopes created for repeated blocks, as they are automatically applied when the parent is applied. You can use these child scopes as you would any other scope.

See the Parent scopes section for essential information about parent and child scopes.

Repeating clones the repeated element, including all of its children:

<table>
	<tr ct-repeat="people">
		<td>Person #<span ct="index"></span></td>
		<td ct="name"></td>
		<td ct="address"></td>
	</tr>
</table>
scope.index = function(childScope) {
	return childScope.$.index;
};

Note above that the scope contains a property scope.$.index that contains the 0-based index of the current repeated block. You can’t access this property directly from the DOM as it is inside the $ object, but you can use a value function to access it (and to add 1 to it if you want the index to be 1-based!).

Another interesting thing is happening in this example, which will be clearer after reading the Parent scopes section. The index value function is added to the parent scope. Each repeating block gets a child scope, which when it looks for the index property will fall back to the parent scope. When a value function is called the first argument is the scope asking for the value, which in this case is the child scope. So return childScope.$.index above returns the index from the child scope! If we’d written this.$.index it would have tried to find the index of the parent scope, which would not exist in this case.

It is also possible to repeat a collection of elements. See Repeating multiple elements in the Advanced section.

Expressions

You can use an expression in place of a scope property in a number of declarations. The expression is evaluated each time the scope is applied.

<h1 ct-show="showTitle and enabled">My title</h1>

You may use and and or in place of && and || to avoid the need to escape ampersands and to be more conversational. Also supported is not, gt, ge, lt, le, eq and ne.

Expressions can also be used to set the value or body of an element:

<h1 ct="title + (titleSuffix ? ': ' + titleSuffix : '')"></h1>

Expressions are supported in most declarations. The exceptions are repeat declarations, declarations that specify ids and template declarations.

Statements

While expressions can be used in place of a scope property, statements can be used in place of an event handler function.

<button ct-do="numberOfClicks++">Count</button>

Multiple statements can be combined using the ; separator.

<button ct-do="clicked=true; numberOfClicks++">Count</button>

Automatic scope creation

The above examples all create the scope explicitly in Javascript. You can also declare where you want scopes to be created using declarations in your markup, and Consistent will automatically create them.

Consistent uses its jQuery plugin to auto create scopes when the onDOMReady event fires. If you are not using Consistent with the jQuery plugin, you will need to call Consistent.autoCreateScopes() yourself at the appropriate time.

If you want to trigger the auto scope creation yourself, set Consistent.settings.autoCreateScopes = false before the onDOMReady event, then call Consistent.autoCreateScopes() when you’re ready.

The simplest way to declare a scope is to add an empty ct-scope attribute. This tells Consistent to create a scope with the given root node, and to call scope.$.update() followed by scope.$.apply(). This will populate the scope from the DOM and then apply the state back to the DOM. You can get the scope in Javascript by finding the scope from the DOM node.

<div ct-scope>
	<p ct="body"></p>
</div>
var scope = $("div").consistent();

Named scopes

You can also declare scopes with a name, and then you can fetch the scope by name in Javascript:

<div ct-scope="myScope"></div>
var scope = Consistent("myScope");

Controllers

You can declare the constructor function to use to create the controller for the scope using the ct-controller attribute.

<body ct-controller="MyPageController">
	<h1 ct="title"></h1>
</body>
function MyPageController(scope) {
	scope.title = "My page title";
}

The controller function name can also be nested, e.g.:

<body ct-controller="MyApp.MyPageController">
	<h1 ct="title"></h1>
	<button ct-do="handleClick">Change</button>
</body>
var MyApp = (function() {
	function MyPageController(scope) {
		scope.title = "My page title";
	}

	MyPageController.prototype.handleClick = function() {
		this.$.apply(function() {
			this.title = "My clicked page title";
		});
	};

	return {
		MyPageController: MyPageController

	};
})();

Initialising the scope

You can declare how you want to initialise the scope, and provide initialisation statements and functions, using the ct-init attribute. You can name a function to use to init the scope using the ct-init-func attribute.

The following values of ct-init are supported:

  • update - the scope will be updated, init function called if declared, and then the scope will be applied.
  • none - the scope will not be updated or applied, but the init function will be called if declared.
  • Any other value is interpreted as a statement, and the scope values are set by it, e.g. ct-init="title='My title'; subtitle='My subtitle'". The init function will be called if declared. Then the scope will be applied.

If ct-init is missing or empty, the scope will default to update if there is no init function and no controller declared. Otherwise it will not update, call the init function, if applicable, and then apply the scope.

<div ct-scope ct-init="title='My title'; subtitle='My subtitle'"></div>

The init function should exist in the global scope. You can use nested property names. The init function will be called with this set to the scope.

<div ct-scope ct-init-func="MyInitFunc">
function MyInitFunc() {
	this.title = "My title";
}

Attributes

You can set DOM element attributes from the scope.

<h1 ct-attr-title="headingTitle">Consistent.js</h1>

Now create a scope and set the heading’s title attribute.

var scope = $("h1").consistent();
scope.headingTitle = "Welcome";
scope.$.apply();

If you want to set DOM element properties, see the Properties section below.

Class attributes

Class attributes can be bound by declaring a ct-attr-class attribute, as above, however there is a shortcut for classes: ct-class.

<h1 ct-class="headingClass">Consistent.js</h1>
scope.headingClass = "heading";
scope.headingClass = "heading another-class";
scope.headingClass = [ "heading", "another-class" ];

Note that as well as being a shortcut, the ct-class supports array values, which are automatically converted to a space-separated string. If the scope contains an array value, update will convert the classes into an array when setting the scope.

The alternative ct-add-class attribute preserves existing classes. When the scope is applied the scope property determines the set of classes to add to the element in addition to its existing classes. The following example code is applied in order and shows how the class attribute changes.

<h1 class="heading" ct-add-class="headingClass">Consistent.js</h1>
scope.headingClass = "another-class";
scope.$.apply(); // class attribute is now "heading another-class"

scope.headingClass = "one two";
scope.$.apply(); // "heading one two"

scope.headingClass = null;
scope.$.apply(); // "heading"

scope.headingClass = "heading";
scope.$.apply(); // "heading"

If you add classes to an element independent of Consistent, it will treat those classes as if they were existing and will not remove them when the scope is next applied, so everyone can play nicely together.

Templating

You can also use templates to update attributes.

<h1 ct-tmpl-attr-title="This is a story about {{subject}}">Title</h1>

<h1 ct-tmpl-id-attr-title="h1-title-template">Title</h1>
<script id="h1-title-template" type="text/x-hogan-template">This is a story about {{subject}}</script>

Binding the scope to the DOM

In the examples above we’ve specifically bound the example nodes by their element name, this isn’t very realistic in practice. When you bind a DOM node to Consistent, all of its child nodes are bound as well, so you typically bind a container element:

<div id="container">
	<h3 ct="name"></h3>
	<p ct="body"></p>
</div>

Now bind the scope.

$("#container").consistent();

Often you will have multiple blocks on the page and you’ll need to have an individual scope for each of them.

<div class="container">
	<p ct="body"></p>
</div>
<div class="container">
	<p ct="body"></p>
</div>

Now bind each to a new scope.

$(".container").each(function() {
	var scope = $(this).consistent();
	scope.body = "Lorem ipsum";
	scope.$.apply();
});

Getting the scope for a DOM node

If you need to get the existing scope for a node, you can follow the exact same pattern. Calling .consistent() again will return the existing scope.

$(".container").each(function() {
	var scope = $(this).consistent();
	scope.body = "Change the body";
	scope.$.apply();
});

You can also call the Consistent.findScopeForNode(node) function, if you just want to check if there’s a scope rather than create one.

Updating the scope from the DOM

Consistent can inspect the DOM to populate the scope.

var scope = $("#container").consistent();
scope.$.update();

Note this doesn’t work for any properties that are using templates.

Watching for changes in the scope

Register a handler function to watch for changes to a particular property, or to the scope as a whole. Watch handler functions are called when apply is called on the scope, before the DOM has been updated.

scope.$.watch("title", function(scope, property, newValue, oldValue) {
	scope.shortTitle = this.title.substring(0, 10);
});

scope.$.watch(["title", "title2"], function(scope, property, newValue, oldValue) {
	scope.shortTitle = this.title.substring(0, 10);
});

scope.$.watch(function(scope, changedProperties, snapshot, oldSnapshot) {
	scope.changeSummary = "The following properties were changed: " + changedProperties;
});

The snapshots passed to the watch handler function for the whole scope are created using the scope.$.snapshot function, and therefore do not have the $ object, and value functions have been replaced with their value.

Notice that you do not need to call apply if you change the scope inside a watch handler. A watch handler may be called multiple times in a single apply if the scope is changed by other watch handlers.

Value functions are watched based on their returned value. If the value returned by a value function changes between one apply and the next, the watch handler function will be called.

It is possible for watch handlers to cause an infinite loop, if the scope does not reach a steady state. This is especially likely if you use value functions that return a new value each time they are evaluated. Consistent detects excessive looping through the watch handler list and throws an exception to break it. The number of loops is set in Consistent.settings.maxWatcherLoops; the default should be good enough.

The scope parameter contains the scope in which the property changed. This is important when using parent and child scopes. this is set to the scope where the watch is declared.

Nested properties

In the case of nested properties, the property argument will be the path to the property, separated by .s.

Watch handlers registered for a property name will be called for any nested properties, e.g. if you register a watch on todos, the watch handler will fire when anything inside the todos object, or array, is changed.

scope.todos = [ "Learn Portugese", "Wash dog", "Wash car" ];
scope.$.watch("todos", function(scope, property) {
	console.log("Todos changed");
});
scope.$.apply(); // Fires the watch handler as the todos array is added to the scope

scope.todos[0] = "Wash cat";
scope.$.apply(); // Fires the watch handler again as todos has changed.

Populating the scope from an object

Often you’ll receive data from an Ajax JSON response as a Javascript object. You can merge these into the scope using the merge function.

var scope = $("#item").consistent();
$.ajax({
	success: function(data) {
		scope.$.merge(data);
		scope.$.apply();
	}
})

Note that the merge is a shallow merge. For each property in the given object it adds it to the scope, replacing and values that are already there. If your scope has nested objects, they are replaced rather than merged.

Exporting the scope to a Javascript object

The scope contains one extra property required for Consistent, the $ property, where all of Consistent’s functionality lives (e.g. scope.$.apply()). It also contains value functions.

Snapshot

Use the snapshot function to obtain a Javascript object without the $ property, and with value functions evaluated and replaced with their value. This provides a snapshot of the state of the scope, and can be used to inspect values without being concerned with value functions, or for submitting back to a server using Ajax.

scope.title = "My title";
scope.subtitle = function() { return "My subtitle" };

console.log(scope.$.snapshot());
{
	title: "My title",
	subtitle: "My subtitle"
}

A snapshot is used when applying the scope to the DOM, and in the watcher functions.

The snapshot function includes properties from parent scopes. If you don’t want to include parent scopes, pass false for the optional includeParents parameter, e.g. snapshot(false).

An example using jQuery and Ajax:

var scope = $("#item").consistent();
scope.$.update();
$.ajax({
	data: scope.$.snapshot()
});

Security

Security is critically important for any web application. The major security vulnerability for Javascript frameworks like Consistent is XSS or cross-site scripting. This occurs when a user is able to inject content into a web page, and that content is then executed by the browser in some form. Sometimes that execution can be harmful, other times it can just be unwanted. We want to prevent all of it.

Web frameworks escape <>& in unsafe content, to prevent an attacker from being able to insert arbitrary markup into your page. As the developer you also need to make sure you don't allow any user generated content into unsafe HTML attributes, such as onclick, and when you allow user generated content into a safe HTML attribute, you need to make sure you escape quotes so an attacker can't end the current attribute value and create a new, unsafe, attribute.

With the addition of Javascript frameworks that perform additional parsing of your page, you need to understand where this parsing occurs and whether there are additional risks.

Consistent is driven by the contents of ct and ct-* attributes on HTML tags. This is by design, so that security with Consistent is achieved using the same techniques already in use. Prevent attackers from creating HTML elements and attributes.

If you allow users to create some HTML markup, such as a whitelist of "safe" elements, you need to make sure the users cannot add attributes, or can only add whitelisted attributes. I don't believe blacklisting attributes is a safe approach, but if you do, you need to ensure that the Consistent attributes are included.

Consistent does support templated content in external elements (see Templating), therefore you need to avoid outputting unsafe content in these areas, or ensure that you escape the content correctly for the templating engine in use.

Consistent does not parse the text content of your HTML page, so there is no need to escape anything additional, as you should already be doing, in those areas. Consistent does not parse the values of any attributes except ct and ct-*, so there is no need to escape anything additional in other attribute values.

Principles

Undefined

If a scope property is not defined then Consistent will not change the DOM.

Advanced

Nested properties

You can use nested properties in the scope.

<h1 ct="person.fullName"></h1>
var scope = $("h1").consistent();
scope.person = {
	fullName: "Nathanial Hornblower"
};
scope.$.apply();

Watch handler functions will be called with the property attribute set to the nested property name, eg. person.fullName. For convenience the scope declares two functions for working with nested property names.

var nestedPropertyName = "person.fullName";
scope.$.get(nestedPropertyName);
scope.$.set(nestedPropertyName, value);

If the appropriate intermediate objects don’t exist, when calling set, they are created and added to the scope for you.

Note that get will fall back to a parent scope, if there is one. See below for Parent scopes. If you don’t want to fall back to a parent scope pass false for the optional includeParents parameter, e.g. get(property, false) instead.

Parent scopes

You can create parent and child scopes. Child scopes will look to their parent if they don’t contain a value for a given property, and so on up the parent chain. When a snapshot is created of a scope, it will include all of the properties in its parent, and its parent’s parent, and so on. As snapshots are used to apply the scope to the DOM, the combined properties of all of the scopes are available to be applied to the DOM.

When apply() is called on a child scope, and the child scope needs to be applied, it automatically calls apply() on its parent scope after applying itself. Similarly, when apply() is called on a parent scope, and the parent scope needs to be applied, it automatically calls apply() on its child scopes after applying itself. Note that in both of these cases, if the scope doesn’t need to be applied, i.e. there are no changes, it does not cascade to parents or children. So always call apply() on the scope that you’ve changed.

var rootScope = Consistent(); /* Create the root scope */
var childScope = Consistent(rootScope); /* Create a child scope */
$("#item").consistent(childScope); /* Bind a DOM node to the child scope */

Note that here we call the Consistent function, whereas previously we’ve used the jQuery plugin to create the scope, as the jQuery plugin does not support the creation of child scopes. If you pass a scope as a parameter to the jQuery plugin it treats that as the scope to bind to. The Consistent function also has an alias created by the jQuery plugin at $.consistent.

Now the following will work:

<div id="item">
	<h2 ct="title"></h2>
</div>
rootScope.title = "Default title";
childScope.$.apply();

Then if you add a title to the childScope and apply it again, it will override the title property in the parent.

Value functions

When a snapshot is created, the value functions are executed and the snapshot will contain their return value rather than the function itself. When a value function in a parent scope is executed for a child scope, the first argument to the value function will be the child scope. Value functions can therefore decide whether to access the scope in which they were declared (this), or the scope in which they are accessed (the first argument).

rootScope.title = function(childScope) {
	return childScope.myTitle;
};
childScope.myTitle = "Title from the child";

Event handlers

If a scope’s controller doesn’t contain the named event handler, the parent scope’s controller will be searched, and so on up the parent chain. Similar to value functions, event handlers are always invoked with this set to the controller in which they are declared. Event handlers’ first argument is the scope in which the event occurred. The function can use that value, if necessary, to operate on the scope where the event occurred.

<div id="item">
	<h2 ct="title" ct-on-click="handleClick"></h2>
</div>
rootScope.$.controller("handleClick", function(childScope, ev, dom) {
	childScope.title += ".";
};

Watch handler functions

Watch handler functions added to parent scopes will be fired for changes in child scopes. Note that this inside the watch function will always be the scope where the watch function is declared, and the first argument will be the scope where the change occurred; in this case, the child scope. In this way the watch function can access both the scope where the change occurred and the scope where the watch function was declared.

Getting the nodes bound to a scope

To get an array of DOM nodes that have been bound to a scope, and that have declared bindings (e.g. have ct... attributes), you can use the nodes() function. Even if a node has been passed to a Consistent scope’s bind function, if a node doesn’t declare bindings then it will not be included in the result from nodes().

nodes() includes any bound nodes in child scopes as well. If you don’t want to include child scopes, pass true for the optional includeParents parameter, e.g. nodes(false).

Note that DOM nodes that define a repeating section (i.e. have a ct-repeat declaration) are not included in the result from nodes(), as those nodes no longer exist in the DOM. However, as nodes from child scopes are included the result may include the repeated nodes if they declare bindings.

To get an array of the DOM nodes that have been passed to a scope’s bind function use the roots() function. Note that the root nodes do not need to have declared Consistent bindings.

$(scope.$.nodes()).addClass("found");
$(scope.$.roots()).addClass("found");

Repeating multiple root elements

The Repeating blocks section above introduces repeating. In that example you can only repeat a single root element, such as an <li> or a <tr>. Consistent also supports repeating a block of multiple root elements, which is useful if you want to add multiple table rows to a table for each block.

<table>
	<tr ct-repeat="people" ct-repeat-container-id="rows"></tr>
</table>

<table style="display:none">
	<tbody id="rows">
		<tr>
			<td>Name</td>
			<td ct="name"></td>
		</tr>
		<tr>
			<td>Address</td>
			<td ct="address"></td>
		</tr>
	</tbody>
</table>

Using the ct-repeat-container-id attribute you can identify nodes elsewhere in the DOM that should be cloned and used in the repeating block. Note that tables automatically get a <tbody> element created, even if it isn’t in the markup, therefore you should attach the id to an explicit <tbody> otherwise if the id is on the <table>, the repeating block will include the automatically created <tbody>.

Properties

You can set DOM element properties from the scope. Properties are DOM node Javascript properties, as opposed to attributes which are declared in the markup. The most common property to use is the style property, which exposes an object containing the DOM element’s style.

<p ct-prop-style-display="showHide">Lorem ipsum</p>

Note that properties may be nested, as in the case of style.display above, and we can specify this by - separating the property name when we declare the ct-prop- attribute.

var scope = $("p").consistent();
scope.showHide = "none";
scope.$.apply();

This sets the style.display property of the <p> element to "none", causing it not to be displayed.

See the Visibility section above for a better way to show and hide elements.

Options

Visibility animation

Often you want to use animation to show or hide elements. You can override the behaviour of showing and hiding by specifying options when you create a scope, or bind a node.

var scope = $("h1").consistent({
	$: {
		show: function(dom) {
			// jQuery fade
			$(dom).fadeIn();
		},
		hide: function(dom) {
			// jQuery fade
			$(dom).fadeOut();
		}
	}
});

You could also specify the show / hide implementation for a specific apply:

scope.$.apply({
	$: {
		show: function(dom) {
			// jQuery fade
			$(dom).fadeIn();
		}
	}
});

Repeating blocks animation

If you want to animate the appearance and disappearance of blocks in a repeating section, you can override the behaviour after adding nodes, and for removing nodes.

var options = { $: {} };
options.$.added = function(dom) {
	// jQuery hide and fadeIn
	$(dom).hide().fadeIn();
};
options.$.remove = function(dom) {
	// jQuery fade then remove
	$(dom).fadeOut(function() {
		$(this).remove();
	});
};
var scope = $("#container").consistent(options);

Change prefix for event handler and value functions

If you’re adding existing objects to your scopes that use naming conventions that don’t fit with Consistent; when Consistent makes a snapshot of the scope (which occurs whenever you apply the scope), you may get unexpected results such as functions in your scope being called unexpectedly. This is because Consistent has interpreted those functions as value functions.

To solve this issue you can pass options to the scope to change the way Consistent identifies value functions and event handler functions. You instruct Consistent to add a prefix to the name before looking in the scope or controller.

By default there are no prefixes. If you apply prefixes, you still refer to the thing without the prefix. It only when you add it to the scope or controller directly that you need to include the prefix.

When you set an event handler prefix ending with a letter, e.g. "do", Consistent will expect the property to be camel-cased and will look for an event handler function specified as ct-do="click" in the property doClick.

You can change the value function prefix by setting the option valueFunctionPrefix. When there is a valueFunctionPrefix set, Consistent will only call functions that match the prefix. Any functions that don’t match the value function prefix will be left untouched. The result of the value will appear in the snapshot without its prefix.

When you use prefixes you must not include the prefix when declaring in the DOM. The value function prefix is removed when the snapshot is created, and event handler prefixes are added when Consistent looks for an event handler.

<div id="container">
	<h1 ct="title"></h1>
	<button ct-do="click">Button</button>
</div>
var options = {
	eventHandlerPrefix: "do",
	valueFunctionPrefix: "get"
};
var scope = $("#container").consistent(options);
scope.getTitle = function() {
	return "Consistent.js"
};
scope.$.controller().doClick = function() {
	alert("Click!");
};

Note that we accessed the controller directly so we had to include the prefix. If we use the scope.$.controller(name, function) appraoch we do not include the prefix as Consistent will include it automatically.

See Merging only specified properties below for an alternative to this approach.

Merging only specified properties

The merge function provides an easy way to merge properties from existing objects into the scope. It also has an optional argument, properties, which is an array of strings. This enables you to pick and choose which properties from your existing objects you merge into the scope.

The properties array supports nested properties using . separators, e.g. person.name.

var object = {
	title: "Consistent.js",
	subtitle: "A JavaScript framework",
	person: {
		name: "Arthur",
		age: 4,
		gender: "m"
	},
	location: {
		city: "Auckland",
		country: "New Zealand"
	},
	friends: [
		"Bob",
		"Carl"
	]
};

scope.merge(object, [ "title", "person.name", "person.age", "location", friends" ]);

The above results in the title, the person’s name and age, and the location and friends arrays all being copied into the scope. The subtitle and the person’s gender are not copied.

Expressions and Statements

Consistent supports expressions and statements for writing simple functionality into the DOM declarations. Note the term “expressions” is sometimes used to refer to both expressions and statements.

Expressions are parsed then reformed into safe Javascript, ensuring that expressions can only access values in the scope, and then compiled into Javascript functions for quick reuse.

Expressions and statements work with value functions as a consequence of accessing a snapshot (where value functions are replaced by their value) or using scope.$.get (which also evaluates value functions), but they cannot themselves call functions. That is, an expression of the form myProperty() will not compile.

Expressions enable you to inline simple logic and changes to your scope. However, don’t overuse expressions; they can result in an application that is harder to maintain if the application logic is spread between HTML and Javascript files.

Reference

DOM declarations

By default, DOM attributes are used to declare the binding between DOM nodes and the scope. The preferred attributes style starts with ct. You can also use data-ct instead of ct.

Binding values

  • ct the name of a property in the scope to use to set the value of this element. Where setting the value means setting the innerHTML, or other properties as appropriate to the element type.
  • ct-tmpl a template that will be rendered with the scope as its context, and then used to set the value of this element.
  • ct-tmpl-id the id of a DOM element that contains template text, e.g. a <script type="text/x-hogan-template"> element.

Form properties

  • ct-disabled make this element disabled when the named property in the scope is true.
  • ct-enabled make this element enabled when the named property in the scope is true.
  • ct-readonly make this element read-only when the named property in the scope is true.
  • ct-readwrite make this element not read-only when the named property in the scope is true.
  • ct-options set the <select> options array based on the array in the named property in the scope. The array should contain either scalar values, or objects each with text and value properties. Each object may also contain a disabled or label property to set those properties on the options.

Attributes and properties

The NAME segment in the following list represents the name of the attribute or property. In the case of properties, the name will have -s changed to .s to enable access to nested properties, e.g. ct-prop-style-display affects the style.display property.

  • ct-attr-NAME the name of a property in the scope to use to set the value of the given attribute on this element.

  • ct-prop-NAME the name of a property in the scope to use to set the value of the given property on this element.

  • ct-tmpl-attr-NAME a template that will be rendered with the scope as its context, and then used to set the value of the given attribute on this element.

  • ct-tmpl-id-attr-NAME the id of a DOM element that contains template text.

  • ct-attrs the name of an object property in the scope with properties and values mapping to attribute names and values. Note that for setting the attribute class you should instead use className as class is sometimes a reserved word.

  • ct-properties the name of an object property in the scope with properties and values mapping to properties, including support for nested properties.

  • ct-class the name of a property in the scope to use to set the value of the class attribute on this element. Supports string and array values.

  • ct-add-class the name of a property in the scope to use to add classes to the existing class attribute on this element. Supports string and array values.

Visibility

  • ct-show show this element when the named property in the scope is true, otherwise hide it.
  • ct-hide the opposite of show.

Event handlers

  • ct-do binds the default event for this element to the named event handler function in the scope. Event handler functions are called with the scope as the first argument, the second argument is the event object and the third argument is the DOM node source of the event.
  • ct-on-EVENT binds the event named EVENT for this element to the named event handler function in the scope.

Repeating blocks

  • ct-repeat repeats this element, and all of its children, for each item in the array in the named property in the scope.
  • ct-repeat-container-id the id of a DOM element that contains DOM nodes to be repeated in place of this element.

Miscellaneous

  • ct-nobind prevents Consistent from binding this element to a scope, and prevents Consistent from cascading the bind to this element’s children. This declaration can be used to fence off markup that should not acquire Consistent functionality (e.g. any unsafe user-generated markup). This attribute can be declared with no value, e.g. <div ct-nobind>, or with the value "true".
  • ct-update controls Consistent's behaviour when updating the scope from this element. Valid values are auto (the default), auto-nokey, noauto, disabled.

Scope functions

All scope functions are nested inside the $ object, and therefore you call them, e.g. scope.$.apply(). All functions without an explicitly documented return value will return the scope to enable chaining, e.g. scope.$.update().$.apply().

DOM

  • apply([options, ] [function, ] [includeChildren]) applies the scope to the DOM. If the optional options are provided they augment each node’s options before applying. If the function argument is provided, the function is called with this set to the scope before the scope is applied and the options as an argument. If the scope has child scopes, they are applied after the scope is applied. If the scope has a parent scope, it will also be applied. If includeChildren is false, child scopes will not be applied. Note that child scopes created for repeated blocks are always applied.
  • applyLater([options, ] [function, ] [includeChildren]) as for apply but rather than applying immediately it creates a setTimeout with a 0 time so it will be called after the current Javascript event handling finishes. The function, if supplied, is called immediately. It is safe to call this multiple times, the scope will only be applied once.
  • needsApply() returns true if the scope has been changed and needs to be applied to the DOM. Changes include properties changed in the scope or new nodes bound to the scope.
  • update([dom [, includeChildren]]) updates the scope by reading property values from the DOM. If the optional dom parameter is provided, only update the given node or array of nodes. If includeChildren is true, the update cascades to child nodes.
  • bind(dom [, options]) binds the given DOM node to the scope. See the options section for the optional options argument. The dom parameter may also be an array of nodes.
  • unbind(dom) unbinds the given DOM node from the scope. The dom parameter may also be an array of nodes.
  • nodes([includeParents]) returns an array of DOM nodes that have been bound to this scope and have bindings. Includes nodes in child scopes unless the optional includeParents parameter is false.
  • roots() returns an array of the DOM nodes explicitly bound to this scope, that is the nodes that were passed to the bind function.

Scope

  • snapshot([includeParents]) returns a Javascript object containing the scope’s model properties, excluding the Consistent $ object, and evaluating value functions and replacing with their current values. Includes properties in parent scopes unless the optional includeParents parameter is false.

  • merge([deep, ] object) merges properties from the given object into the scope. If deep is provided it is a boolean indicating whether to do a deep merge. A normal merge simply copies across all of the properties in object, replacing any existing objects, whereas a deep merge will merge objects.

  • merge(object, properties) merges the properties named in the properties array from the given object into the scope. The properties argument may be an array of property names or a single property, and may include nested properties using dot notation, e.g. [ "name", "address.street" ].

  • clear() removes all properties from the scope. This only leaves Consistent’s $ object.

  • scope() returns the scope itself.

  • scope(object) replaces the scope with the given object. The given object is actually used as the scope, and Consistent’s $ object is added into this object. The return value is the object given.

  • get(property [, includeParents]) returns the value in the scope for the given property. If the scope contains a value function for the given property (after adding the value function prefix, if any), the value function is evaluated and its result returned. If the scope contains an event handler for the given property (after adding the event handler prefix), the event handler function is returned. Supports nested properties (i.e. that contain dot notation) and falls back to parent scopes if the scope doesn’t have a property for the given property itself, unless the optional includeParents parameter is false. If no property with the given property is found it returns undefined.

  • set(property, value) sets the value in the scope for the given property. Supports nested properties. If the target property exists and contains a value function, the value function is called passing the value as the only argument. If no property exists in the scope for the given property, parent scopes are searched for a value function to call. If no value functions are found, a new property is created in the scope with the given value.

  • controller() returns the controller.

  • controller(object) sets the controller to the given object. Consistent’s $ object is added to this object. The return value is the scope.

  • controller(name) returns the value, usually an event handler function, with the given name in the scope’s controller. Supports nested names. If the value is a function, it is wrapped in an anonymous function that ensures this is bound to the controller when it is called.

  • controller(name, function) sets the function in the scope’s controller for the given name. Supports nested names.

  • fire(name [, arguments...]]) looks for a function in the scope’s controller for the given name (supports nested names), and call that function passing the optional additional arguments (in the case of event handler functions, these are the Javascript event object and the DOM node), and returning the result. Supports nested properties and falls back to parent scopes’ controllers. Note that the first argument to the controller functions is always the scope in which the event occurred. If no controller function is found this function has no effect and returns undefined.

  • getValueFunction(property [, includeParents]) returns the value function in the scope for the given property. Supports nested properties and falls back to parent scopes, unless the optional includeParents parameter is false.

  • setValueFunction(property, function) sets the value function in the scope for the given property. Supports nested properties.

Watch

  • watch([property,] function) adds the given handler function as a watch function to the property, if provided, otherwise to the whole scope. The property argument can be an array of property names.
  • unwatch([property,] function) unbinds the watch function. The property argument can be an array of property names.

Expressions and Statements

  • evaluate(expression) evaluates the given expression string in the context of the scope.
  • exec(statements) parses and executes the given statements string in the context of the scope.

General

  • parent() returns the parent scope, or null if there is no parent scope.
  • children() returns an array containing the immediate child scopes of this scope.
  • options([node]) returns the options object for the given node, or for the scope as a whole. Note that you can modify the returned options object, but changes to the scope’s options will not affect node options.

Scope properties

The scope exposes some properties inside the $ object, e.g. scope.$.index.

  • index the 0-based index of the given scope within a repeating section, or undefined if not in a repeating section.

Consistent functions

  • Consistent([options] [, controllerConstructor]) returns a new scope. If the options are provided the scope is initialised with them. If the controller function is provider, a new controller is created from it.
  • Consistent(parentScope [, options] [, controllerFunction]) returns a new scope and sets its parent scope. If the options are provided the scope is initialised with them. If the controller function is provider, a new controller is created from it.
  • Consistent(node) returns the scope the DOM node is bound to, or null.
  • Consistent(name) returns the named scope with the given name, or null. As for Consistent.findScopeByName(name).
  • Consistent.findScopeByName(name) returns the named scope with the given name, or null.
  • Consistent.isScope(object) returns true if the given object is a Consistent scope.

jQuery plugin

  • $.consistent is synonymous with the Consistent function above and can be used in the same way.
  • $(selector).consistent() checks the selected elements to see if they have been bound to a scope. If they’ve all been bound to the same scope, it returns that scope. If they’ve been bound to different scopes (or some have been bound and some haven’t) this throws an exception. If they haven’t been bound to a scope a new scope is created, the elements are bound and the scope is returned.
  • $(selector).consistent(scope [, options]) binds the selected nodes to the given scope, with the options if provided and returns the scope.
  • $(selector).consistent([options] [, controllerConstructor]) creates a new scope with the given arguments, binds the selected elements to it and returns the scope.

What Consistent doesn’t do

Consistent doesn’t create DOM nodes. There are great tools for creating DOM nodes, such as simply using jQuery or using a templating engine such as Mustache or Hogan (which I’ve used in the examples). You can easily create new DOM nodes and then bind a new Consistent scope to them. Note that Consistent does in fact create DOM nodes if you create them in repeat blocks or templates; however see the templating section for advice about that.

Consistent doesn’t do any Ajax. Consistent scopes can be easily populated from an Ajax JSON response, and their data can be easily exported for sending to a server. Look at the scope.$.merge(object) and scope.$.snapshot() functions, respectively.

Troubleshooting

Functions in objects in the scope are called unexpectedly

See the section Change prefix for event handler and value functions above for an explanation and a solution to this problem.

About

Consistent is a small and simple Javascript framework to enable an abstract model to be synced with the DOM.

Resources

Stars

Watchers

Forks

Packages

No packages published