summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGeorge Crawford <g.o.crawford@gmail.com>2013-10-08 10:34:53 +0100
committerGeorge Crawford <g.o.crawford@gmail.com>2013-10-08 10:34:53 +0100
commit3d33c839cdb7ce38ba5b0f887814279181d663fb (patch)
tree135371dc1f732485eb59281c9110585d791292f8
downloadfastdom-3d33c839cdb7ce38ba5b0f887814279181d663fb.zip
fastdom-3d33c839cdb7ce38ba5b0f887814279181d663fb.tar.gz
fastdom-3d33c839cdb7ce38ba5b0f887814279181d663fb.tar.bz2
First pages commit
-rw-r--r--.gitignore2
-rw-r--r--.npmignore3
-rw-r--r--.travis.yml7
-rw-r--r--History.md14
-rw-r--r--README.md141
-rw-r--r--bower.json15
-rw-r--r--component.json15
-rw-r--r--examples/animation.html120
-rw-r--r--examples/aspect-ratio.html96
-rw-r--r--index.js289
-rw-r--r--package.json25
-rw-r--r--test/index.html28
-rw-r--r--test/setup.js18
-rw-r--r--test/test.clear.js97
-rw-r--r--test/test.defer.js50
-rw-r--r--test/test.set.js187
16 files changed, 1107 insertions, 0 deletions
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..28f1ba7
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,2 @@
+node_modules
+.DS_Store \ No newline at end of file
diff --git a/.npmignore b/.npmignore
new file mode 100644
index 0000000..a806a83
--- /dev/null
+++ b/.npmignore
@@ -0,0 +1,3 @@
+/node_modules/
+/examples/
+/test/ \ No newline at end of file
diff --git a/.travis.yml b/.travis.yml
new file mode 100644
index 0000000..6212d53
--- /dev/null
+++ b/.travis.yml
@@ -0,0 +1,7 @@
+script:
+ - "npm test"
+
+language: node_js
+
+node_js:
+ - "0.10" \ No newline at end of file
diff --git a/History.md b/History.md
new file mode 100644
index 0000000..68eeb5f
--- /dev/null
+++ b/History.md
@@ -0,0 +1,14 @@
+
+0.7.1 / 2013-10-05
+==================
+
+ * fix memory leaks with undeleted refs
+ * fix context not being passed to `.defer` jobs
+
+0.7.0 / 2013-10-05
+==================
+
+ * add `FastDom#clear` clears read, write and defer jobs by id
+ * remove `FastDom#clearRead`
+ * remove `FastDom#clearWrite`
+ * change directory structure by removing `/lib` \ No newline at end of file
diff --git a/README.md b/README.md
new file mode 100644
index 0000000..bd01a18
--- /dev/null
+++ b/README.md
@@ -0,0 +1,141 @@
+# fastdom [![Build Status](https://travis-ci.org/wilsonpage/fastdom.png?branch=master)](https://travis-ci.org/wilsonpage/fastdom)
+
+Eliminates layout thrashing by batching DOM read/write operations (~750 bytes gzipped).
+
+```js
+var fastdom = new FastDom();
+
+fastdom.read(function() {
+ console.log('read');
+});
+
+fastdom.write(function() {
+ console.log('write');
+});
+
+fastdom.read(function() {
+ console.log('read');
+});
+
+fastdom.write(function() {
+ console.log('write');
+});
+```
+
+Outputs:
+
+```
+read
+read
+write
+write
+```
+
+## Examples
+
+- [Aspect ratio example](http://wilsonpage.github.io/fastdom/examples/aspect-ratio.html)
+
+## Installation
+
+FastDom is CommonJS and AMD compatible, you can install it in one of the following ways:
+
+```
+$ npm install fastdom
+```
+```
+$ bower install fastdom
+```
+```
+$ component install wilsonpage/fastdom
+```
+or [download](http://github.com/wilsonpage/fastdom/raw/master/index.js).
+
+## How it works
+
+FastDom works as a regulatory layer between your app/library and the DOM. By batching DOM access we **avoid unnecessary document reflows and speed up layout perfomance dramatically**.
+
+Each read/write job is added to a corresponding read/write queue. The queues are emptied (reads, then writes) at the turn of the next frame using [`window.requestAnimationFrame`](https://developer.mozilla.org/en-US/docs/Web/API/window.requestAnimationFrame).
+
+FastDom aims to behave like a singleton across *all* modules in your app. When any module requires `'fastdom'` they get the same instance back, meaning FastDom can harmonize DOM access app-wide.
+
+Potentially a third-party library could depend on FastDom, and better integrate within an app that itself uses it.
+
+## API
+
+### FastDom#read(callback[, context])
+
+Schedules a job for the 'read' queue. Returns a unique ID that can be used to clear the scheduled job.
+
+```js
+fastdom.read(function() {
+ var width = element.clientWidth;
+});
+```
+
+### FastDom#write(callback[, context])
+
+Schedules a job for the 'write' queue. Returns a unique ID that can be used to clear the scheduled job.
+
+```js
+fastdom.write(function() {
+ element.style.width = width + 'px';
+});
+```
+
+### FastDom#defer(frames, callback[, context])
+
+Defers a job for the number of frames specified. This is useful is you have a particualrly expensive piece of work to do, and don't want it to be done with all the other work.
+
+For example; you are using third party library that doesn't expose an API that allows you split DOM read/write work, `fastdom.defer()` will push this work futher into the future and prevent it from disrupting other carefully batched work.
+
+```js
+fastdom.defer(3, expensiveStuff);
+```
+
+### FastDom#clear(id)
+
+Clears **any** scheduled job by id.
+
+```js
+var read = fastdom.read(function(){});
+var write = fastdom.write(function(){});
+var defer = fastdom.defer(4, function(){});
+
+fastdom.clear(read);
+fastdom.clear(write);
+fastdom.clear(defer);
+```
+
+## Tests
+
+#### With PhantomJS
+
+```
+$ npm install
+$ npm test
+```
+
+#### Without PhantomJS
+
+Open `test/index.html` in your browser.
+
+## Author
+
+- **Wilson Page** - [@wilsonpage](http://github.com/wilsonpage)
+
+## Contributors
+
+- **Wilson Page** - [@wilsonpage](http://github.com/wilsonpage)
+- **George Crawford** - [@georgecrawford](http://github.com/georgecrawford)
+
+## License
+
+(The MIT License)
+
+Copyright (c) 2013 Wilson Page <wilsonpage@me.com>
+
+Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the 'Software'), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. \ No newline at end of file
diff --git a/bower.json b/bower.json
new file mode 100644
index 0000000..d318c23
--- /dev/null
+++ b/bower.json
@@ -0,0 +1,15 @@
+{
+ "name": "fastdom",
+ "description": "Eliminates layout thrashing by batching DOM read/write operations",
+ "version": "0.7.1",
+ "main": "index.js",
+ "scripts": [
+ "index.js"
+ ],
+ "ignore": [
+ "examples/",
+ "test/",
+ "README.md"
+ ],
+ "license": "MIT"
+}
diff --git a/component.json b/component.json
new file mode 100644
index 0000000..82aa51f
--- /dev/null
+++ b/component.json
@@ -0,0 +1,15 @@
+{
+ "name": "fastdom",
+ "description": "Eliminates layout thrashing by batching DOM read/write operations",
+ "version": "0.7.1",
+ "main": "index.js",
+ "scripts": [
+ "index.js"
+ ],
+ "ignore": [
+ "examples/",
+ "test/",
+ "README.md"
+ ],
+ "license": "MIT"
+} \ No newline at end of file
diff --git a/examples/animation.html b/examples/animation.html
new file mode 100644
index 0000000..9f9b80c
--- /dev/null
+++ b/examples/animation.html
@@ -0,0 +1,120 @@
+<!DOCTYPE html>
+<html>
+ <head>
+ <title>FastDom: Animation Example</title>
+ <script type="text/javascript" src="../index.js"></script>
+ </head>
+ <body>
+
+ <style>
+ .mover {
+ background: url("");
+ height: 100px;
+ width: 100px;
+ position: absolute;
+ }
+
+ button.active {
+ color: red;
+ }
+
+ </style>
+
+ <label>Number of elements <input id="count" type="text" value="400" /></label>
+ <button id="sync" class="active">Forced synchronous layout</button>
+ <button id="async">Run with FastDom</button>
+ <button id="noread">Avoid DOM read</button>
+ <button id="toggle">Start</button>
+
+ <div id='test'></div>
+
+ <script>
+
+ var moveMethod = 'sync',
+ count = document.getElementById('count'),
+ test = document.getElementById('test'),
+ timestamp, raf, movers;
+
+ var mover = {
+ sync: function(m) {
+ mover.setLeft(m, movers[m].offsetTop);
+ },
+ async: function(m) {
+ fastdom.read(function() {
+ var top = movers[m].offsetTop;
+ fastdom.write(function() {
+ mover.setLeft(m, top);
+ });
+ });
+ },
+ noread: function(m) {
+ mover.setLeft(m, m);
+ },
+ setLeft: function(m, top) {
+ movers[m].style.left = ((Math.sin(top + timestamp/1000)+1) * 500) + 'px';
+ }
+ };
+
+ function update(thisTimestamp) {
+ timestamp = thisTimestamp;
+ for (var m = 0; m < movers.length; m++) {
+ mover[moveMethod](m);
+ }
+ raf = window.requestAnimationFrame(update);
+ }
+
+ function toggleAnim(e) {
+
+ var html, num;
+
+ if (raf) {
+
+ window.cancelAnimationFrame(raf);
+ raf = false;
+ e.currentTarget.innerHTML = 'Start';
+ count.disabled = false;
+
+ } else {
+
+ html = '';
+ num = count.value;
+
+ for (i = 0; i < num; i++) {
+ html += '<div class="mover"></div>';
+ }
+ test.innerHTML = html;
+
+ movers = test.querySelectorAll('.mover');
+ movers[0].style.top = '50px';
+ for (var m = 1; m < movers.length; m++) {
+ movers[m].style.top = (m * 20) + 'px';
+ }
+
+ raf = window.requestAnimationFrame(update);
+ e.currentTarget.innerHTML = 'Stop';
+ count.disabled = true;
+ }
+ }
+
+ function setMethod(method) {
+ document.getElementById(moveMethod).classList.remove('active');
+ document.getElementById(method).classList.add('active');
+ moveMethod = method;
+ }
+
+ document.getElementById('toggle').addEventListener('click', toggleAnim);
+ document.getElementById('sync').addEventListener('click', function() {
+ setMethod('sync');
+ });
+ document.getElementById('async').addEventListener('click', function() {
+ setMethod('async');
+ });
+ document.getElementById('noread').addEventListener('click', function() {
+ setMethod('noread');
+ });
+
+ </script>
+
+
+ </body>
+</html>
diff --git a/examples/aspect-ratio.html b/examples/aspect-ratio.html
new file mode 100644
index 0000000..25ee73e
--- /dev/null
+++ b/examples/aspect-ratio.html
@@ -0,0 +1,96 @@
+<!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<title>FastDom: Aspect Ratio Example</title>
+<style>
+
+ * {
+ box-sizing: border-box;
+ }
+
+ div {
+ float: left;
+ background: silver;
+ border: solid 2px white;
+ }
+
+</style>
+</head>
+<body>
+ <label>Number of elements <input id="input" type="text" value="100" /></label>
+ <button id="withoutFastDom">Run without FastDom</button>
+ <button id="withFastDom">Run with FastDom</button>
+ <button id="resetbtn">reset</button>
+ <section id="perf"></section>
+ <section id="container"></section>
+ <script type="text/javascript" src="../index.js"></script>
+ <script>
+ var n = input.value;
+ var start;
+ var divs;
+
+ // Setup
+ function reset() {
+ divs = [];
+ container.innerHTML = '';
+
+ for (var i = 0; i < n; i++) {
+ var div = document.createElement('div');
+ div.style.width = Math.round(Math.random() * window.innerWidth) + 'px';
+ container.appendChild(div);
+ divs.push(div);
+ }
+ }
+
+ function setAspect(div, i) {
+ var aspect = 9/16;
+ var isLast = i === (n - 1)
+ var h = div.clientWidth * aspect;
+
+ div.style.height = h + 'px';
+
+ if (isLast) {
+ displayPerf(performance.now() - start);
+ }
+ }
+
+ function setAspectFastDom(div, i) {
+ var aspect = 9/16;
+ var isLast = i === (n - 1)
+
+ // READ
+ fastdom.read(function() {
+ var h = div.clientWidth * aspect;
+
+ // WRITE
+ fastdom.write(function() {
+ div.style.height = h + 'px';
+
+ if (isLast) {
+ displayPerf(performance.now() - start);
+ }
+ });
+ });
+ }
+
+ function displayPerf(ms) {
+ perf.textContent = ms + 'ms';
+ }
+
+ withoutFastDom.onclick = function() {
+ reset();
+ start = performance.now();
+ divs.forEach(setAspect);
+ };
+
+ withFastDom.onclick = function() {
+ reset();
+ start = performance.now();
+ divs.forEach(setAspectFastDom);
+ };
+
+ resetbtn.onclick = reset;
+ </script>
+</body>
+</html> \ No newline at end of file
diff --git a/index.js b/index.js
new file mode 100644
index 0000000..6be22f8
--- /dev/null
+++ b/index.js
@@ -0,0 +1,289 @@
+
+/**
+ * DOM-Batch
+ *
+ * Eliminates layout thrashing
+ * by batching DOM read/write
+ * interactions.
+ *
+ * @author Wilson Page <wilsonpage@me.com>
+ */
+
+;(function(fastdom){
+
+ 'use strict';
+
+ // Normalize rAF
+ var raf = window.requestAnimationFrame
+ || window.webkitRequestAnimationFrame
+ || window.mozRequestAnimationFrame
+ || window.msRequestAnimationFrame
+ || function(cb) { return window.setTimeout(cb, 1000 / 60); };
+
+ // Normalize cAF
+ var caf = window.cancelAnimationFrame
+ || window.cancelRequestAnimationFrame
+ || window.mozCancelAnimationFrame
+ || window.mozCancelRequestAnimationFrame
+ || window.webkitCancelAnimationFrame
+ || window.webkitCancelRequestAnimationFrame
+ || window.msCancelAnimationFrame
+ || window.msCancelRequestAnimationFrame
+ || function(id) { window.clearTimeout(id); };
+
+ /**
+ * Creates a fresh
+ * FastDom instance.
+ *
+ * @constructor
+ */
+ function FastDom() {
+ this.lastId = 0;
+ this.jobs = {};
+ this.mode = null;
+ this.pending = false;
+ this.queue = {
+ read: [],
+ write: []
+ };
+ }
+
+ /**
+ * Adds a job to
+ * the read queue.
+ *
+ * @param {Function} fn
+ * @api public
+ */
+ FastDom.prototype.read = function(fn, ctx) {
+ var job = this.add('read', fn, ctx);
+ this.queue.read.push(job.id);
+ this.request('read');
+ return job.id;
+ };
+
+ /**
+ * Adds a job to
+ * the write queue.
+ *
+ * @param {Function} fn
+ * @api public
+ */
+ FastDom.prototype.write = function(fn, ctx) {
+ var job = this.add('write', fn, ctx);
+ this.queue.write.push(job.id);
+ this.request('write');
+ return job.id;
+ };
+
+ /**
+ * Removes a job from
+ * the 'reads' queue.
+ *
+ * @param {Number} id
+ * @api public
+ */
+ FastDom.prototype.clear = function(id) {
+ var job = this.jobs[id];
+ if (!job) return;
+
+ // Clear reference
+ delete this.jobs[id];
+
+ // Defer jobs are cleared differently
+ if (job.type === 'defer') {
+ caf(job.timer);
+ return;
+ }
+
+ var list = this.queue[job.type];
+ var index = list.indexOf(id);
+ if (~index) list.splice(index, 1);
+ };
+
+ /**
+ * Makes the decision as to
+ * whether a the frame needs
+ * to be scheduled.
+ *
+ * @param {String} type
+ * @api private
+ */
+ FastDom.prototype.request = function(type) {
+ var mode = this.mode;
+ var self = this;
+
+ // If we are currently writing, we don't
+ // need to scedule a new frame as this
+ // job will be emptied from the write queue
+ if (mode === 'writing' && type === 'write') return;
+
+ // If we are reading we don't need to schedule
+ // a new frame as this read will be emptied
+ // in the currently active read queue
+ if (mode === 'reading' && type === 'read') return;
+
+ // If we are reading we don't need to schedule
+ // a new frame and this write job will be run
+ // after the read queue has been emptied in the
+ // currently active frame.
+ if (mode === 'reading' && type === 'write') return;
+
+ // If there is already a frame
+ // scheduled, don't schedule another one
+ if (this.pending) return;
+
+ // Schedule frame (preserving context)
+ raf(function() { self.frame(); });
+
+ // Set flag to indicate
+ // a frame has been scheduled
+ this.pending = true;
+ };
+
+ /**
+ * Generates a unique
+ * id for a job.
+ *
+ * @return {Number}
+ * @api private
+ */
+ FastDom.prototype.uniqueId = function() {
+ return ++this.lastId;
+ };
+
+ /**
+ * Calls each job in
+ * the list passed.
+ *
+ * If a context has been
+ * stored on the function
+ * then it is used, else the
+ * current `this` is used.
+ *
+ * @param {Array} list
+ * @api private
+ */
+ FastDom.prototype.flush = function(list) {
+ var id;
+ while (id = list.shift()) {
+ this.run(this.jobs[id]);
+ }
+ };
+
+ /**
+ * Runs any read jobs followed
+ * by any write jobs.
+ *
+ * @api private
+ */
+ FastDom.prototype.frame = function() {
+
+ // Set the pending flag to
+ // false so that any new requests
+ // that come in will schedule a new frame
+ this.pending = false;
+
+ // Set the mode to 'reading',
+ // then empty all read jobs
+ this.mode = 'reading';
+ this.flush(this.queue.read);
+
+ // Set the mode to 'writing'
+ // then empty all write jobs
+ this.mode = 'writing';
+ this.flush(this.queue.write);
+
+ this.mode = null;
+ };
+
+ /**
+ * Defers the given job
+ * by the number of frames
+ * specified.
+ *
+ * @param {Number} frames
+ * @param {Function} fn
+ * @api public
+ */
+ FastDom.prototype.defer = function(frames, fn, ctx) {
+ if (frames < 0) return;
+ var job = this.add('defer', fn, ctx);
+ var self = this;
+
+ (function wrapped() {
+ if (!(frames--)) {
+ self.run(job);
+ return;
+ }
+
+ job.timer = raf(wrapped);
+ })();
+
+ return job.id;
+ };
+
+ /**
+ * Adds a new job to
+ * the given queue.
+ *
+ * @param {Array} list
+ * @param {Function} fn
+ * @param {Object} ctx
+ * @returns {Number} id
+ * @api private
+ */
+ FastDom.prototype.add = function(type, fn, ctx) {
+ var id = this.uniqueId();
+ return this.jobs[id] = {
+ id: id,
+ fn: fn,
+ ctx: ctx,
+ type: type
+ };
+ };
+
+ /**
+ * Called when a callback errors.
+ * Overwrite this if you don't
+ * want errors inside your jobs
+ * to fail silently.
+ *
+ * @param {Error}
+ */
+ FastDom.prototype.onError = function(){};
+
+ /**
+ * Runs a given job.
+ * @param {Object} job
+ * @api private
+ */
+ FastDom.prototype.run = function(job){
+ var ctx = job.ctx || this;
+
+ // Clear reference to the job
+ delete this.jobs[job.id];
+
+ // Call the job in
+ try { job.fn.call(ctx); } catch(e) {
+ this.onError(e);
+ }
+ };
+
+ // We only ever want there to be
+ // one instance of FastDom in an app
+ fastdom = fastdom || new FastDom();
+
+ /**
+ * Expose 'fastdom'
+ */
+
+ if (typeof module !== 'undefined' && module.exports) {
+ module.exports = fastdom;
+ } else if (typeof define === "function" && define.amd) {
+ define(function(){ return fastdom; });
+ } else {
+ window['fastdom'] = fastdom;
+ }
+
+})(window.fastdom);
diff --git a/package.json b/package.json
new file mode 100644
index 0000000..e4ea0f3
--- /dev/null
+++ b/package.json
@@ -0,0 +1,25 @@
+{
+ "name": "fastdom",
+ "description": "Eliminates layout thrashing by batching DOM read/write operations",
+ "version": "0.7.1",
+ "main": "index.js",
+ "scripts": {
+ "test": "./node_modules/.bin/mocha-phantomjs test/index.html"
+ },
+ "homepage": "https://github.com/wilsonpage/fastdom",
+ "author": {
+ "name": "Wilson Page",
+ "email": "wilsonpage@me.com"
+ },
+ "repository": {
+ "type": "git",
+ "url": "git://github.com/wilsonpage/fastdom.git"
+ },
+ "license": "MIT",
+ "devDependencies": {
+ "mocha": "~1.12.0",
+ "sinon": "~1.7.3",
+ "chai": "~1.7.2",
+ "mocha-phantomjs": "~3.1.2"
+ }
+}
diff --git a/test/index.html b/test/index.html
new file mode 100644
index 0000000..4257f62
--- /dev/null
+++ b/test/index.html
@@ -0,0 +1,28 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Mocha Tests</title>
+ <link rel="stylesheet" href="../node_modules/mocha/mocha.css" />
+</head>
+<body>
+ <div id="mocha"></div>
+ <script src="../node_modules/mocha/mocha.js"></script>
+ <script src="../node_modules/chai/chai.js"></script>
+ <script src="../node_modules/sinon/lib/sinon.js"></script>
+ <script src="../node_modules/sinon/lib/sinon/match.js"></script>
+ <script src="../node_modules/sinon/lib/sinon/spy.js"></script>
+ <script>mocha.setup('tdd')</script>
+ <script src="../index.js"></script>
+ <script src="setup.js"></script>
+ <script src="test.set.js"></script>
+ <script src="test.clear.js"></script>
+ <script src="test.defer.js"></script>
+ <script>
+ mocha.checkLeaks();
+
+ if (window.mochaPhantomJS) mochaPhantomJS.run();
+ else mocha.run();
+ </script>
+</body>
+</html> \ No newline at end of file
diff --git a/test/setup.js b/test/setup.js
new file mode 100644
index 0000000..0844695
--- /dev/null
+++ b/test/setup.js
@@ -0,0 +1,18 @@
+
+// RequestAnimationFrame Polyfill
+var raf = window.requestAnimationFrame
+ || window.webkitRequestAnimationFrame
+ || window.mozRequestAnimationFrame
+ || function(cb) { window.setTimeout(cb, 1000 / 60); };
+
+// Make constructor
+var FastDom = fastdom.constructor;
+
+// Alias chai.assert
+var assert = chai.assert;
+
+function objectLength(object) {
+ var l = 0;
+ for (var key in object) l++;
+ return l;
+} \ No newline at end of file
diff --git a/test/test.clear.js b/test/test.clear.js
new file mode 100644
index 0000000..ee60f2e
--- /dev/null
+++ b/test/test.clear.js
@@ -0,0 +1,97 @@
+
+suite('Clear', function(){
+
+ test("Should not run 'read' job if cleared (sync)", function(done) {
+ var fastdom = new FastDom();
+ var read = sinon.spy();
+
+ var id = fastdom.read(read);
+ fastdom.clear(id);
+
+ raf(function() {
+ assert(!read.called);
+ done();
+ });
+ });
+
+ test("Should fail silently if job not found in queue", function(done) {
+ var fastdom = new FastDom();
+ var read = sinon.spy();
+ var read2 = sinon.spy();
+
+ var id = fastdom.read(read);
+ fastdom.clear(id);
+
+ raf(function() {
+ assert(!read2.called);
+ done();
+ });
+ });
+
+ test("Should not run 'write' job if cleared (async)", function(done) {
+ var fastdom = new FastDom();
+ var read = sinon.spy();
+ var write = sinon.spy();
+
+ var id = fastdom.write(write);
+ fastdom.read(function() {
+ fastdom.clear(id);
+
+ raf(function() {
+ assert(!read.called);
+ done();
+ });
+ });
+ });
+
+ test("Should not run 'write' job if cleared", function(done) {
+ var fastdom = new FastDom();
+ var write = sinon.spy();
+ var id = fastdom.write(write);
+
+ fastdom.clear(id);
+
+ raf(function() {
+ assert(!write.called);
+ done();
+ });
+ });
+
+ test("Should not run 'defer' job if cleared", function(done) {
+ var fastdom = new FastDom();
+ var write = sinon.spy();
+ var id = fastdom.defer(3, write);
+
+ fastdom.clear(id);
+
+ raf(function() {
+ raf(function() {
+ raf(function() {
+ raf(function() {
+ assert(!write.called);
+ done();
+ });
+ });
+ });
+ });
+ });
+
+ test("Should remove reference to the job if cleared", function(done) {
+ var fastdom = new FastDom();
+ var write = sinon.spy();
+ var id = fastdom.defer(2, write);
+
+ fastdom.clear(id);
+
+ raf(function() {
+ raf(function() {
+ raf(function() {
+ assert(!write.called);
+ assert(!fastdom.jobs[id]);
+ done();
+ });
+ });
+ });
+ });
+
+}); \ No newline at end of file
diff --git a/test/test.defer.js b/test/test.defer.js
new file mode 100644
index 0000000..c1ae179
--- /dev/null
+++ b/test/test.defer.js
@@ -0,0 +1,50 @@
+
+suite('defer', function(){
+
+ test("Should run the job after the specified number of frames", function(done) {
+ var fastdom = new FastDom();
+ var job = sinon.spy();
+
+ fastdom.defer(4, job);
+
+ raf(function() {
+ assert(!job.called);
+ raf(function() {
+ assert(!job.called);
+ raf(function() {
+ assert(!job.called);
+ raf(function() {
+ assert(job.called);
+ done();
+ });
+ });
+ });
+ });
+ });
+
+ test("Should call a deferred callback with the given context", function(done) {
+ var fastdom = new FastDom();
+ var cb = sinon.spy();
+ var ctx = { foo: 'bar' };
+
+ fastdom.defer(2, function() {
+ assert.equal(this.foo, 'bar');
+ done();
+ }, ctx);
+ });
+
+ test("Should remove the reference to the job once run", function(done) {
+ var fastdom = new FastDom();
+ var callback = sinon.spy();
+ var id = fastdom.defer(2, callback);
+
+ raf(function() {
+ raf(function() {
+ raf(function() {
+ assert(!fastdom.jobs[id]);
+ done();
+ });
+ });
+ });
+ });
+});
diff --git a/test/test.set.js b/test/test.set.js
new file mode 100644
index 0000000..a1dd83c
--- /dev/null
+++ b/test/test.set.js
@@ -0,0 +1,187 @@
+
+suite('Set', function() {
+
+ test("Should run reads before writes", function(done) {
+ var fastdom = new FastDom();
+
+ var read = sinon.spy(function() {
+ assert(!write.called);
+ });
+
+ var write = sinon.spy(function() {
+ assert(read.called);
+ done();
+ });
+
+ fastdom.read(read);
+ fastdom.write(write);
+ });
+
+ test("Should call all reads together, followed by all writes", function(done) {
+ var fastdom = new FastDom();
+ var read1 = sinon.spy();
+ var read2 = sinon.spy();
+ var write1 = sinon.spy();
+ var write2 = sinon.spy();
+
+ // Assign unsorted
+ fastdom.read(read1);
+ fastdom.write(write1);
+ fastdom.read(read2);
+ fastdom.write(write2);
+
+ // After the queue has been emptied
+ // check the callbacks were called
+ // in the correct order.
+ raf(function() {
+ assert(read1.calledBefore(read2));
+ assert(read2.calledBefore(write1));
+ assert(write1.calledBefore(write2));
+ done();
+ });
+ });
+
+ test("Should call a read in the same frame if scheduled inside a read callback", function(done) {
+ var fastdom = new FastDom();
+ var cb = sinon.spy();
+
+ fastdom.read(function() {
+
+ // Schedule a callback for *next* frame
+ raf(cb);
+
+ // Schedule a read callback
+ // that should be run in the
+ // current frame checking that
+ // the RAF callback has not
+ // yet been fired.
+ fastdom.read(function() {
+ assert(!cb.called);
+ done();
+ });
+ });
+ });
+
+ test("Should call a write in the same frame if scheduled inside a read callback", function(done) {
+ var fastdom = new FastDom();
+ var cb = sinon.spy();
+
+ fastdom.read(function() {
+
+ // Schedule a callback for *next* frame
+ raf(cb);
+
+ // Schedule a read callback
+ // that should be run in the
+ // current frame checking that
+ // the RAF callback has not
+ // yet been fired.
+ fastdom.write(function() {
+ assert(!cb.called);
+ done();
+ });
+ });
+ });
+
+ test("Should call a read in the *next* frame if scheduled inside a write callback", function(done) {
+ var fastdom = new FastDom();
+ var cb = sinon.spy();
+
+ fastdom.write(function() {
+
+ // Schedule a callback for *next* frame
+ raf(cb);
+
+ // Schedule a write that should be
+ // called in the next frame, meaning
+ // the test callback should have already
+ // been called.
+ fastdom.read(function() {
+ assert(cb.called);
+ done();
+ });
+ });
+ });
+
+ test("Should call a 'read' callback with the given context", function(done) {
+ var fastdom = new FastDom();
+ var cb = sinon.spy();
+ var ctx = { foo: 'bar' };
+
+ fastdom.read(function() {
+ assert.equal(this.foo, 'bar');
+ done();
+ }, ctx);
+ });
+
+ test("Should call a 'write' callback with the given context", function(done) {
+ var fastdom = new FastDom();
+ var cb = sinon.spy();
+ var ctx = { foo: 'bar' };
+
+ fastdom.write(function() {
+ assert.equal(this.foo, 'bar');
+ done();
+ }, ctx);
+ });
+
+ test("Should have empty job hash when batch complete", function(done) {
+ var fastdom = new FastDom();
+
+ fastdom.read(function(){});
+ fastdom.read(function(){});
+ fastdom.write(function(){});
+ fastdom.write(function(){});
+
+ // Check there are four jobs stored
+ assert.equal(objectLength(fastdom.jobs), 4);
+
+ raf(function() {
+ assert.equal(objectLength(fastdom.jobs), 0);
+ done();
+ });
+ });
+
+ test("Should maintain correct context if single method is registered twice", function(done) {
+ var fastdom = new FastDom();
+ var ctx1 = { foo: 'bar' };
+ var ctx2 = { bar: 'baz' };
+
+ function shared(){}
+
+ var spy1 = sinon.spy(shared);
+ var spy2 = sinon.spy(shared);
+
+ fastdom.read(spy1, ctx1);
+ fastdom.read(spy2, ctx2);
+
+ raf(function() {
+ assert(spy1.calledOn(ctx1));
+ assert(spy2.calledOn(ctx2));
+ done();
+ });
+ });
+
+ test("Should call a registered onError handler when an error is thrown inside a job", function(done) {
+ var fastdom = new FastDom();
+ var err1 = { some: 'error1' };
+ var err2 = { some: 'error2' };
+
+ fastdom.onError = sinon.spy();
+
+ fastdom.read(function() {
+ throw err1;
+ });
+
+ fastdom.write(function() {
+ throw err2;
+ });
+
+ raf(function() {
+ assert(fastdom.onError.calledTwice);
+ assert(fastdom.onError.getCall(0).calledWith(err1));
+ assert(fastdom.onError.getCall(1).calledWith(err2));
+ done();
+ });
+ });
+}); \ No newline at end of file