refactor redraw into pubsub and autoredraw
- pubsub is a basic pubsub impl - autoredraw is glue code to register callback to pubsub and onevent moved e.redraw to autoredraw
This commit is contained in:
parent
db46bb4414
commit
0005cf26ee
13 changed files with 149 additions and 105 deletions
17
api/autoredraw.js
Normal file
17
api/autoredraw.js
Normal file
|
|
@ -0,0 +1,17 @@
|
||||||
|
"use strict"
|
||||||
|
|
||||||
|
var throttle = require("../api/throttle")
|
||||||
|
|
||||||
|
module.exports = function(root, renderer, pubsub, callback) {
|
||||||
|
var run = throttle(callback)
|
||||||
|
renderer.setEventCallback(function(e) {
|
||||||
|
if (e.redraw !== false) run()
|
||||||
|
})
|
||||||
|
|
||||||
|
if (pubsub != null) {
|
||||||
|
if (root.redraw) pubsub.unsubscribe(root.redraw)
|
||||||
|
pubsub.subscribe(run)
|
||||||
|
}
|
||||||
|
|
||||||
|
return root.redraw = run
|
||||||
|
}
|
||||||
|
|
@ -1,18 +1,15 @@
|
||||||
"use strict"
|
"use strict"
|
||||||
|
|
||||||
var coreRenderer = require("../render/render")
|
var coreRenderer = require("../render/render")
|
||||||
var throttle = require("../api/throttle")
|
var autoredraw = require("../api/autoredraw")
|
||||||
|
|
||||||
module.exports = function($window, renderers) {
|
module.exports = function($window, pubsub) {
|
||||||
var renderer = coreRenderer($window)
|
var renderer = coreRenderer($window)
|
||||||
return function(root, component) {
|
return function(root, component) {
|
||||||
var run = throttle(function() {
|
var run = autoredraw(root, renderer, pubsub, function() {
|
||||||
renderer.render(root, {tag: component})
|
renderer.render(root, {tag: component})
|
||||||
})
|
})
|
||||||
|
|
||||||
renderer.setEventCallback(run)
|
|
||||||
|
|
||||||
renderers.push(run)
|
|
||||||
run()
|
run()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
15
api/pubsub.js
Normal file
15
api/pubsub.js
Normal file
|
|
@ -0,0 +1,15 @@
|
||||||
|
"use strict"
|
||||||
|
|
||||||
|
module.exports = function() {
|
||||||
|
var callbacks = []
|
||||||
|
function unsubscribe(callback) {
|
||||||
|
var index = callbacks.indexOf(callback)
|
||||||
|
if (index > -1) callbacks.splice(index, 1)
|
||||||
|
}
|
||||||
|
function publish() {
|
||||||
|
for (var i = 0; i < callbacks.length; i++) {
|
||||||
|
callbacks[i].apply(this, arguments)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {subscribe: callbacks.push.bind(callbacks), unsubscribe: unsubscribe, publish: publish}
|
||||||
|
}
|
||||||
|
|
@ -1,12 +0,0 @@
|
||||||
"use strict"
|
|
||||||
|
|
||||||
module.exports = function(renderers) {
|
|
||||||
return function() {
|
|
||||||
if (renderers.length === 0) return
|
|
||||||
if (renderers.length === 1) return renderers[0]()
|
|
||||||
|
|
||||||
for (var i = 0; i < renderers.length; i++) {
|
|
||||||
renderers[i]()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -2,9 +2,9 @@
|
||||||
|
|
||||||
var coreRenderer = require("../render/render")
|
var coreRenderer = require("../render/render")
|
||||||
var coreRouter = require("../router/router")
|
var coreRouter = require("../router/router")
|
||||||
var throttle = require("../api/throttle")
|
var autoredraw = require("../api/autoredraw")
|
||||||
|
|
||||||
module.exports = function($window, renderers) {
|
module.exports = function($window, pubsub) {
|
||||||
var renderer = coreRenderer($window)
|
var renderer = coreRenderer($window)
|
||||||
var router = coreRouter($window)
|
var router = coreRouter($window)
|
||||||
var route = function(root, defaultRoute, routes) {
|
var route = function(root, defaultRoute, routes) {
|
||||||
|
|
@ -13,10 +13,7 @@ module.exports = function($window, renderers) {
|
||||||
}, function() {
|
}, function() {
|
||||||
router.setPath(defaultRoute)
|
router.setPath(defaultRoute)
|
||||||
})
|
})
|
||||||
var run = throttle(replay)
|
autoredraw(root, renderer, pubsub, replay)
|
||||||
|
|
||||||
renderer.setEventCallback(run)
|
|
||||||
renderers.push(run)
|
|
||||||
}
|
}
|
||||||
route.link = router.link
|
route.link = router.link
|
||||||
route.prefix = router.setPrefix
|
route.prefix = router.setPrefix
|
||||||
|
|
|
||||||
|
|
@ -19,13 +19,15 @@
|
||||||
<script src="../../querystring/parse.js"></script>
|
<script src="../../querystring/parse.js"></script>
|
||||||
<script src="../../request/request.js"></script>
|
<script src="../../request/request.js"></script>
|
||||||
<script src="../../router/router.js"></script>
|
<script src="../../router/router.js"></script>
|
||||||
<script src="../redraw.js"></script>
|
<script src="../../api/throttle.js"></script>
|
||||||
<script src="../throttle.js"></script>
|
<script src="../../api/pubsub.js"></script>
|
||||||
<script src="../mount.js"></script>
|
<script src="../../api/autoredraw.js"></script>
|
||||||
<script src="../router.js"></script>
|
<script src="../../api/mount.js"></script>
|
||||||
|
<script src="../../api/router.js"></script>
|
||||||
|
|
||||||
<script src="./test-redraw.js"></script>
|
|
||||||
<script src="./test-throttle.js"></script>
|
<script src="./test-throttle.js"></script>
|
||||||
|
<script src="./test-pubsub.js"></script>
|
||||||
|
<script src="./test-autoredraw.js"></script>
|
||||||
<script src="./test-mount.js"></script>
|
<script src="./test-mount.js"></script>
|
||||||
<script src="./test-router.js"></script>
|
<script src="./test-router.js"></script>
|
||||||
|
|
||||||
|
|
|
||||||
63
api/tests/test-autoredraw.js
Normal file
63
api/tests/test-autoredraw.js
Normal file
|
|
@ -0,0 +1,63 @@
|
||||||
|
"use strict"
|
||||||
|
|
||||||
|
var o = require("../../ospec/ospec")
|
||||||
|
var domMock = require("../../test-utils/domMock")
|
||||||
|
|
||||||
|
var coreRenderer = require("../../render/render")
|
||||||
|
var apiPubSub = require("../../api/pubsub")
|
||||||
|
var autoredraw = require("../../api/autoredraw")
|
||||||
|
|
||||||
|
o.spec("autoredraw", function() {
|
||||||
|
var FRAME_BUDGET = Math.floor(1000 / 60)
|
||||||
|
var $window, root, renderer, pubsub, spy
|
||||||
|
o.beforeEach(function() {
|
||||||
|
$window = domMock()
|
||||||
|
root = $window.document.body
|
||||||
|
renderer = coreRenderer($window)
|
||||||
|
pubsub = apiPubSub()
|
||||||
|
spy = o.spy()
|
||||||
|
})
|
||||||
|
|
||||||
|
o("returns self-trigger", function() {
|
||||||
|
var run = autoredraw(root, renderer, pubsub, spy)
|
||||||
|
|
||||||
|
run()
|
||||||
|
|
||||||
|
o(spy.callCount).equals(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
o("registers onevent", function() {
|
||||||
|
autoredraw(root, renderer, pubsub, spy)
|
||||||
|
|
||||||
|
renderer.render(root, {tag: "div", attrs: {onclick: function() {}}})
|
||||||
|
|
||||||
|
var e = $window.document.createEvent("MouseEvents")
|
||||||
|
e.initEvent("click", true, true)
|
||||||
|
root.firstChild.dispatchEvent(e)
|
||||||
|
|
||||||
|
o(spy.callCount).equals(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
o("registers pubsub", function() {
|
||||||
|
autoredraw(root, renderer, pubsub, spy)
|
||||||
|
|
||||||
|
pubsub.publish()
|
||||||
|
|
||||||
|
o(spy.callCount).equals(1)
|
||||||
|
})
|
||||||
|
o("throttles", function(done) {
|
||||||
|
var run = autoredraw(root, renderer, pubsub, spy)
|
||||||
|
|
||||||
|
run()
|
||||||
|
run()
|
||||||
|
|
||||||
|
o(spy.callCount).equals(1)
|
||||||
|
|
||||||
|
setTimeout(function() {
|
||||||
|
o(spy.callCount).equals(2)
|
||||||
|
|
||||||
|
done()
|
||||||
|
}, FRAME_BUDGET)
|
||||||
|
})
|
||||||
|
|
||||||
|
})
|
||||||
|
|
@ -4,35 +4,23 @@ var o = require("../../ospec/ospec")
|
||||||
var domMock = require("../../test-utils/domMock")
|
var domMock = require("../../test-utils/domMock")
|
||||||
|
|
||||||
var m = require("../../render/hyperscript")
|
var m = require("../../render/hyperscript")
|
||||||
|
var apiPubSub = require("../../api/pubsub")
|
||||||
var apiMounter = require("../../api/mount")
|
var apiMounter = require("../../api/mount")
|
||||||
|
|
||||||
o.spec("m.mount", function() {
|
o.spec("mount", function() {
|
||||||
var FRAME_BUDGET = Math.floor(1000 / 60)
|
var FRAME_BUDGET = Math.floor(1000 / 60)
|
||||||
var $window, root, mount, renderers
|
var $window, root, redraw, mount
|
||||||
|
|
||||||
o.beforeEach(function() {
|
o.beforeEach(function() {
|
||||||
$window = domMock()
|
$window = domMock()
|
||||||
|
|
||||||
root = $window.document.body
|
root = $window.document.body
|
||||||
|
|
||||||
renderers = []
|
redraw = apiPubSub()
|
||||||
mount = apiMounter($window, renderers)
|
mount = apiMounter($window, redraw)
|
||||||
})
|
|
||||||
|
|
||||||
o("pushes a render function", function() {
|
|
||||||
mount(root, {
|
|
||||||
view : function() {
|
|
||||||
return m("div")
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
o(renderers.length).equals(1)
|
|
||||||
o(typeof renderers[0]).equals("function")
|
|
||||||
})
|
})
|
||||||
|
|
||||||
o("renders into `root`", function() {
|
o("renders into `root`", function() {
|
||||||
var mount = apiMounter($window, [])
|
|
||||||
|
|
||||||
mount(root, {
|
mount(root, {
|
||||||
view : function() {
|
view : function() {
|
||||||
return m("div")
|
return m("div")
|
||||||
|
|
@ -106,7 +94,7 @@ o.spec("m.mount", function() {
|
||||||
o(onupdate.callCount).equals(0)
|
o(onupdate.callCount).equals(0)
|
||||||
|
|
||||||
done()
|
done()
|
||||||
}, 20)
|
}, FRAME_BUDGET)
|
||||||
})
|
})
|
||||||
|
|
||||||
o("redraws when the render function is run", function(done) {
|
o("redraws when the render function is run", function(done) {
|
||||||
|
|
@ -125,7 +113,7 @@ o.spec("m.mount", function() {
|
||||||
o(oninit.callCount).equals(1)
|
o(oninit.callCount).equals(1)
|
||||||
o(onupdate.callCount).equals(0)
|
o(onupdate.callCount).equals(0)
|
||||||
|
|
||||||
renderers[0]()
|
redraw.publish()
|
||||||
|
|
||||||
// Wrapped to give time for the rate-limited redraw to fire
|
// Wrapped to give time for the rate-limited redraw to fire
|
||||||
setTimeout(function() {
|
setTimeout(function() {
|
||||||
|
|
|
||||||
|
|
@ -1,32 +1,30 @@
|
||||||
"use strict"
|
"use strict"
|
||||||
|
|
||||||
var o = require("../../ospec/ospec")
|
var o = require("../../ospec/ospec")
|
||||||
var createRedraw = require("../../api/redraw")
|
var apiPubSub = require("../../api/pubsub")
|
||||||
|
|
||||||
o.spec("m.redraw", function() {
|
|
||||||
var redraw, renderers
|
|
||||||
|
|
||||||
|
o.spec("pubsub", function() {
|
||||||
|
var pubsub
|
||||||
o.beforeEach(function() {
|
o.beforeEach(function() {
|
||||||
renderers = []
|
pubsub = apiPubSub()
|
||||||
redraw = createRedraw(renderers)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
o("it shouldn't error if there are no renderers", function() {
|
o("it shouldn't error if there are no renderers", function() {
|
||||||
redraw()
|
pubsub.publish()
|
||||||
})
|
})
|
||||||
|
|
||||||
o("it should run a single renderer entry", function() {
|
o("it should run a single renderer entry", function() {
|
||||||
var spy = o.spy()
|
var spy = o.spy()
|
||||||
|
|
||||||
renderers.push(spy)
|
pubsub.subscribe(spy)
|
||||||
|
|
||||||
redraw()
|
pubsub.publish()
|
||||||
|
|
||||||
o(spy.callCount).equals(1)
|
o(spy.callCount).equals(1)
|
||||||
|
|
||||||
redraw()
|
pubsub.publish()
|
||||||
redraw()
|
pubsub.publish()
|
||||||
redraw()
|
pubsub.publish()
|
||||||
|
|
||||||
o(spy.callCount).equals(4)
|
o(spy.callCount).equals(4)
|
||||||
})
|
})
|
||||||
|
|
@ -36,20 +34,20 @@ o.spec("m.redraw", function() {
|
||||||
var spy2 = o.spy()
|
var spy2 = o.spy()
|
||||||
var spy3 = o.spy()
|
var spy3 = o.spy()
|
||||||
|
|
||||||
renderers.push(spy1, spy2, spy3)
|
pubsub.subscribe(spy1)
|
||||||
|
pubsub.subscribe(spy2)
|
||||||
|
pubsub.subscribe(spy3)
|
||||||
|
|
||||||
redraw()
|
pubsub.publish()
|
||||||
|
|
||||||
o(spy1.callCount).equals(1)
|
o(spy1.callCount).equals(1)
|
||||||
o(spy2.callCount).equals(1)
|
o(spy2.callCount).equals(1)
|
||||||
o(spy3.callCount).equals(1)
|
o(spy3.callCount).equals(1)
|
||||||
|
|
||||||
redraw()
|
pubsub.publish()
|
||||||
redraw()
|
|
||||||
redraw()
|
|
||||||
|
|
||||||
o(spy1.callCount).equals(4)
|
o(spy1.callCount).equals(2)
|
||||||
o(spy2.callCount).equals(4)
|
o(spy2.callCount).equals(2)
|
||||||
o(spy3.callCount).equals(4)
|
o(spy3.callCount).equals(2)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
@ -5,11 +5,12 @@ var pushStateMock = require("../../test-utils/pushStateMock")
|
||||||
var domMock = require("../../test-utils/domMock")
|
var domMock = require("../../test-utils/domMock")
|
||||||
|
|
||||||
var m = require("../../render/hyperscript")
|
var m = require("../../render/hyperscript")
|
||||||
|
var apiPubSub = require("../../api/pubsub")
|
||||||
var apiRouter = require("../../api/router")
|
var apiRouter = require("../../api/router")
|
||||||
|
|
||||||
o.spec("m.route", function() {
|
o.spec("route", function() {
|
||||||
var FRAME_BUDGET = Math.floor(1000 / 60)
|
var FRAME_BUDGET = Math.floor(1000 / 60)
|
||||||
var $window, root, route, renderers
|
var $window, root, redraw, route
|
||||||
|
|
||||||
o.beforeEach(function() {
|
o.beforeEach(function() {
|
||||||
$window = {}
|
$window = {}
|
||||||
|
|
@ -22,21 +23,8 @@ o.spec("m.route", function() {
|
||||||
|
|
||||||
root = $window.document.body
|
root = $window.document.body
|
||||||
|
|
||||||
renderers = []
|
redraw = apiPubSub()
|
||||||
route = apiRouter($window, renderers)
|
route = apiRouter($window, redraw)
|
||||||
})
|
|
||||||
|
|
||||||
o("pushes a render function", function() {
|
|
||||||
route(root, "/", {
|
|
||||||
"/" : {
|
|
||||||
view: function() {
|
|
||||||
return m("div")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
o(renderers.length).equals(1)
|
|
||||||
o(typeof renderers[0]).equals("function")
|
|
||||||
})
|
})
|
||||||
|
|
||||||
o("renders into `root`", function() {
|
o("renders into `root`", function() {
|
||||||
|
|
@ -68,7 +56,7 @@ o.spec("m.route", function() {
|
||||||
|
|
||||||
o(oninit.callCount).equals(1)
|
o(oninit.callCount).equals(1)
|
||||||
|
|
||||||
renderers[0]()
|
redraw.publish()
|
||||||
|
|
||||||
// Wrapped to give time for the rate-limited redraw to fire
|
// Wrapped to give time for the rate-limited redraw to fire
|
||||||
setTimeout(function() {
|
setTimeout(function() {
|
||||||
|
|
@ -146,7 +134,7 @@ o.spec("m.route", function() {
|
||||||
o(onupdate.callCount).equals(0)
|
o(onupdate.callCount).equals(0)
|
||||||
|
|
||||||
done()
|
done()
|
||||||
}, 20)
|
}, FRAME_BUDGET)
|
||||||
})
|
})
|
||||||
|
|
||||||
o("changes location on route.link", function() {
|
o("changes location on route.link", function() {
|
||||||
|
|
|
||||||
|
|
@ -81,12 +81,4 @@ o.spec("throttle", function() {
|
||||||
|
|
||||||
o(spy.callCount).equals(2)
|
o(spy.callCount).equals(2)
|
||||||
})
|
})
|
||||||
|
|
||||||
o("it supports aborting when redraw is falsey", function() {
|
|
||||||
throttled({ redraw : false })
|
|
||||||
throttled({ redraw : 0 })
|
|
||||||
throttled({ redraw : "" })
|
|
||||||
|
|
||||||
o(spy.callCount).equals(0)
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
|
|
|
||||||
|
|
@ -7,7 +7,6 @@ module.exports = function(callback) {
|
||||||
var timeout = typeof requestAnimationFrame === "function" ? requestAnimationFrame : setTimeout
|
var timeout = typeof requestAnimationFrame === "function" ? requestAnimationFrame : setTimeout
|
||||||
return function(synchronous) {
|
return function(synchronous) {
|
||||||
var now = new Date().getTime()
|
var now = new Date().getTime()
|
||||||
if (typeof synchronous === "object" && "redraw" in synchronous && !synchronous.redraw) return
|
|
||||||
if (synchronous === true || last === 0 || now - last >= time) {
|
if (synchronous === true || last === 0 || now - last >= time) {
|
||||||
last = now
|
last = now
|
||||||
callback()
|
callback()
|
||||||
|
|
|
||||||
18
index.js
18
index.js
|
|
@ -2,18 +2,18 @@
|
||||||
|
|
||||||
var m = require("./render/hyperscript")
|
var m = require("./render/hyperscript")
|
||||||
var trust = require("./render/trust")
|
var trust = require("./render/trust")
|
||||||
var coreRenderer = require("./render/render")
|
|
||||||
var apiRedraw = require("./api/redraw")
|
|
||||||
var apiMounter = require("./api/mount")
|
|
||||||
var apiRouter = require("./api/router")
|
|
||||||
var coreRequester = require("./request/request")
|
var coreRequester = require("./request/request")
|
||||||
var renderers = []
|
var coreRenderer = require("./render/render")
|
||||||
|
var apiPubSub = require("./api/pubsub")
|
||||||
|
var apiMount = require("./api/mount")
|
||||||
|
var apiRouter = require("./api/router")
|
||||||
|
var redraw = apiPubSub()
|
||||||
|
|
||||||
m.redraw = apiRedraw(renderers)
|
|
||||||
m.trust = trust
|
m.trust = trust
|
||||||
m.render = coreRenderer(window).render
|
|
||||||
m.mount = apiMounter(window, renderers)
|
|
||||||
m.route = apiRouter(window, renderers)
|
|
||||||
m.request = coreRequester(window, Promise).ajax
|
m.request = coreRequester(window, Promise).ajax
|
||||||
|
m.render = coreRenderer(window).render
|
||||||
|
m.mount = apiMount(window, redraw)
|
||||||
|
m.route = apiRouter(window, redraw)
|
||||||
|
m.redraw = redraw.publish
|
||||||
|
|
||||||
module.exports = m
|
module.exports = m
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue