This commit is contained in:
Sebastian Sandqvist 2016-08-15 12:20:02 -07:00
parent 4854b4f959
commit 4e1ab37a0e
2 changed files with 103 additions and 100 deletions

View file

@ -1,7 +1,7 @@
"use strict"
var log = console.error.bind(console)
var stream = require("./stream");
var stream = require("./stream/index");
var m = require("./render/hyperscript")
var renderService = require("./render/render")(window)
var requestService = require("./request/request")(window, log)

View file

@ -1,84 +1,85 @@
new function() {
var log = console.error.bind(console)
var log = console.error.bind(console)
var StreamFactory = function(log) {
var guid = 0, noop = function() {}, HALT = {}
function createStream() {
function stream() {
if (arguments.length > 0 && arguments[0] !== HALT) updateStream(stream, arguments[0], undefined)
return stream._state.value
function stream1() {
if (arguments.length > 0 && arguments[0] !== HALT) updateStream(stream1, arguments[0], undefined)
return stream1._state.value
}
initStream(stream, arguments)
if (arguments.length > 0 && arguments[0] !== HALT) updateStream(stream, arguments[0], undefined)
return stream
initStream(stream1, arguments)
if (arguments.length > 0 && arguments[0] !== HALT) updateStream(stream1, arguments[0], undefined)
return stream1
}
function initStream(stream, args) {
stream.constructor = createStream
stream._state = {id: guid++, value: undefined, error: undefined, state: 0, derive: undefined, recover: undefined, deps: {}, parents: [], errorStream: undefined, endStream: undefined}
stream.map = map, stream.ap = ap, stream.of = createStream
stream.valueOf = valueOf, stream.toJSON = toJSON, stream.toString = valueOf
stream.run = run, stream.catch = doCatch
Object.defineProperties(stream, {
function initStream(stream1, args) {
stream1.constructor = createStream
stream1._state = {id: guid++, value: undefined, error: undefined, state: 0, derive: undefined, recover: undefined, deps: {}, parents: [], errorStream: undefined, endStream: undefined}
stream1.map = map, stream1.ap = ap, stream1.of = createStream
stream1.valueOf = valueOf, stream1.toJSON = toJSON, stream1.toString = valueOf
stream1.run = run, stream1.catch = doCatch
Object.defineProperties(stream1, {
error: {get: function() {
if (!stream._state.errorStream) {
if (!stream1._state.errorStream) {
var errorStream = function() {
if (arguments.length > 0 && arguments[0] !== HALT) updateStream(stream, undefined, arguments[0])
return stream._state.error
if (arguments.length > 0 && arguments[0] !== HALT) updateStream(stream1, undefined, arguments[0])
return stream1._state.error
}
initStream(errorStream, [])
initDependency(errorStream, [stream], noop, noop)
stream._state.errorStream = errorStream
initDependency(errorStream, [stream1], noop, noop)
stream1._state.errorStream = errorStream
}
return stream._state.errorStream
return stream1._state.errorStream
}},
end: {get: function() {
if (!stream._state.endStream) {
if (!stream1._state.endStream) {
var endStream = createStream()
endStream.map(function(value) {
if (value === true) unregisterStream(stream), unregisterStream(endStream)
if (value === true) unregisterStream(stream1), unregisterStream(endStream)
return value
})
stream._state.endStream = endStream
stream1._state.endStream = endStream
}
return stream._state.endStream
return stream1._state.endStream
}}
})
}
function updateStream(stream, value, error) {
updateState(stream, value, error)
for (var id in stream._state.deps) updateDependency(stream._state.deps[id], false)
finalize(stream)
function updateStream(stream1, value, error) {
updateState(stream1, value, error)
for (var id in stream1._state.deps) updateDependency(stream1._state.deps[id], false)
finalize(stream1)
}
function updateState(stream, value, error) {
function updateState(stream1, value, error) {
error = unwrapError(value, error)
if (error !== undefined && typeof stream._state.recover === "function") {
if (!resolve(stream, updateValues, true)) return
if (error !== undefined && typeof stream1._state.recover === "function") {
if (!resolve(stream1, updateValues, true)) return
}
else updateValues(stream, value, error)
stream._state.changed = true
if (stream._state.state !== 2) stream._state.state = 1
else updateValues(stream1, value, error)
stream1._state.changed = true
if (stream1._state.state !== 2) stream1._state.state = 1
}
function updateValues(stream, value, error) {
stream._state.value = value
stream._state.error = error
function updateValues(stream1, value, error) {
stream1._state.value = value
stream1._state.error = error
}
function updateDependency(stream, mustSync) {
var state = stream._state, parents = state.parents
function updateDependency(stream1, mustSync) {
var state = stream1._state, parents = state.parents
if (parents.length > 0 && parents.filter(active).length === parents.length && (mustSync || parents.filter(changed).length > 0)) {
var failed = parents.filter(errored)
if (failed.length > 0) updateState(stream, undefined, failed[0]._state.error)
else resolve(stream, updateState, false)
if (failed.length > 0) updateState(stream1, undefined, failed[0]._state.error)
else resolve(stream1, updateState, false)
}
}
function resolve(stream, update, shouldRecover) {
function resolve(stream1, update, shouldRecover) {
try {
var value = shouldRecover ? stream._state.recover() : stream._state.derive()
var value = shouldRecover ? stream1._state.recover() : stream1._state.derive()
if (value === HALT) return false
update(stream, value, undefined)
update(stream1, value, undefined)
}
catch (e) {
update(stream, undefined, e.__error != null ? e.__error : e)
if (e.__error == null) reportUncaughtError(stream, e)
update(stream1, undefined, e.__error != null ? e.__error : e)
if (e.__error == null) reportUncaughtError(stream1, e)
}
return true
}
@ -89,28 +90,28 @@ var StreamFactory = function(log) {
}
return error
}
function finalize(stream) {
stream._state.changed = false
for (var id in stream._state.deps) stream._state.deps[id]._state.changed = false
function finalize(stream1) {
stream1._state.changed = false
for (var id in stream1._state.deps) stream1._state.deps[id]._state.changed = false
}
function reportUncaughtError(stream, e) {
if (Object.keys(stream._state.deps).length === 0) {
function reportUncaughtError(stream1, e) {
if (Object.keys(stream1._state.deps).length === 0) {
setTimeout(function() {
if (Object.keys(stream._state.deps).length === 0) log(e)
if (Object.keys(stream1._state.deps).length === 0) log(e)
}, 0)
}
}
function run(fn) {
var self = createStream(), stream = this
return initDependency(self, [stream], function() {
return absorb(self, fn(stream()))
var self = createStream(), stream1 = this
return initDependency(self, [stream1], function() {
return absorb(self, fn(stream1()))
}, undefined)
}
function doCatch(fn) {
var self = createStream(), stream = this
var derive = function() {return stream._state.value}
var recover = function() {return absorb(self, fn(stream._state.error))}
return initDependency(self, [stream], derive, recover)
var self = createStream(), stream1 = this
var derive = function() {return stream1._state.value}
var recover = function() {return absorb(self, fn(stream1._state.error))}
return initDependency(self, [stream1], derive, recover)
}
function combine(fn, streams) {
return initDependency(createStream(), streams, function() {
@ -119,12 +120,12 @@ var StreamFactory = function(log) {
return fn.apply(this, streams.concat([streams.filter(changed)]))
}, undefined)
}
function absorb(stream, value) {
function absorb(stream1, value) {
if (value != null && value.constructor === createStream) {
var absorbable = value
var update = function() {
updateState(stream, absorbable._state.value, absorbable._state.error)
for (var id in stream._state.deps) updateDependency(stream._state.deps[id], false)
updateState(stream1, absorbable._state.value, absorbable._state.error)
for (var id in stream1._state.deps) updateDependency(stream1._state.deps[id], false)
}
absorbable.map(update).catch(function(e) {
update()
@ -146,46 +147,52 @@ var StreamFactory = function(log) {
updateDependency(dep, true)
return dep
}
function registerDependency(stream, parents) {
function registerDependency(stream1, parents) {
for (var i = 0; i < parents.length; i++) {
parents[i]._state.deps[stream._state.id] = stream
registerDependency(stream, parents[i]._state.parents)
parents[i]._state.deps[stream1._state.id] = stream1
registerDependency(stream1, parents[i]._state.parents)
}
}
function unregisterStream(stream) {
for (var i = 0; i < stream._state.parents.length; i++) {
var parent = stream._state.parents[i]
delete parent._state.deps[stream._state.id]
function unregisterStream(stream1) {
for (var i = 0; i < stream1._state.parents.length; i++) {
var parent = stream1._state.parents[i]
delete parent._state.deps[stream1._state.id]
}
for (var id in stream._state.deps) {
var dependent = stream._state.deps[id]
var index = dependent._state.parents.indexOf(stream)
for (var id in stream1._state.deps) {
var dependent = stream1._state.deps[id]
var index = dependent._state.parents.indexOf(stream1)
if (index > -1) dependent._state.parents.splice(index, 1)
}
stream._state.state = 2 //ended
stream._state.deps = {}
stream1._state.state = 2 //ended
stream1._state.deps = {}
}
function map(fn) {return combine(function(stream) {return fn(stream())}, [this])}
function ap(stream) {return combine(function(s1, s2) {return s1()(s2())}, [this, stream])}
function map(fn) {return combine(function(stream1) {return fn(stream1())}, [this])}
function ap(stream1) {return combine(function(s1, s2) {return s1()(s2())}, [this, stream1])}
function valueOf() {return this._state.value}
function toJSON() {return this._state.value != null && typeof this._state.value.toJSON === "function" ? this._state.value.toJSON() : this._state.value}
function active(stream) {return stream._state.state === 1}
function changed(stream) {return stream._state.changed}
function notEnded(stream) {return stream._state.state !== 2}
function errored(stream) {return stream._state.error}
function active(stream1) {return stream1._state.state === 1}
function changed(stream1) {return stream1._state.changed}
function notEnded(stream1) {return stream1._state.state !== 2}
function errored(stream1) {return stream1._state.error}
function reject(e) {
var stream = createStream()
stream.error(e)
return stream
var stream1 = createStream()
stream1.error(e)
return stream1
}
function merge(streams) {
return combine(function () {
return streams.map(function(s) {return s()})
}, streams)
}
return {stream: createStream, merge: merge, combine: combine, reject: reject, HALT: HALT}
return {stream1: createStream, merge: merge, combine: combine, reject: reject, HALT: HALT}
}
var Stream = StreamFactory(log)
var stream = Stream.stream
stream.combine = Stream.combine
stream.reject = Stream.reject
stream.merge = Stream.merge
stream.HALT = Stream.HALT
;
function Vnode(tag, key, attrs, children, text, dom) {
return {tag: tag, key: key, attrs: attrs, children: children, text: text, dom: dom, domSize: undefined, state: {}, events: undefined, instance: undefined}
}
@ -786,8 +793,8 @@ var requestService = function($window, log) {
function setCompletionCallback(callback) {oncompletion = callback}
function xhr(args) {
var stream = Stream.stream()
if (args.initialValue !== undefined) stream(args.initialValue)
var stream2 = Stream.stream()
if (args.initialValue !== undefined) stream2(args.initialValue)
var useBody = typeof args.useBody === "boolean" ? args.useBody : args.method !== "GET" && args.method !== "TRACE"
@ -816,16 +823,16 @@ var requestService = function($window, log) {
try {
var response = (args.extract !== extract) ? args.extract(xhr, args) : args.deserialize(args.extract(xhr, args))
if (xhr.status >= 200 && xhr.status < 300) {
stream(cast(args.type, response))
stream2(cast(args.type, response))
}
else {
var error = new Error(xhr.responseText)
for (var key in response) error[key] = response[key]
stream.error(error)
stream2.error(error)
}
}
catch (e) {
stream.error(e)
stream2.error(e)
}
if (typeof oncompletion === "function") oncompletion()
}
@ -834,23 +841,23 @@ var requestService = function($window, log) {
if (useBody) xhr.send(args.data)
else xhr.send()
return stream
return stream2
}
function jsonp(args) {
var stream = Stream.stream()
if (args.initialValue !== undefined) stream(args.initialValue)
var stream2 = Stream.stream()
if (args.initialValue !== undefined) stream2(args.initialValue)
var callbackName = args.callbackName || "_mithril_" + Math.round(Math.random() * 1e16) + "_" + callbackCount++
var script = $window.document.createElement("script")
$window[callbackName] = function(data) {
script.parentNode.removeChild(script)
stream(cast(args.type, data))
stream2(cast(args.type, data))
if (typeof oncompletion === "function") oncompletion()
delete $window[callbackName]
}
script.onerror = function() {
script.parentNode.removeChild(script)
stream.error(new Error("JSONP request failed"))
stream2.error(new Error("JSONP request failed"))
if (typeof oncompletion === "function") oncompletion()
delete $window[callbackName]
}
@ -859,7 +866,7 @@ var requestService = function($window, log) {
args.data[args.callbackKey || "callback"] = callbackName
script.src = assemble(args.url, args.data)
$window.document.documentElement.appendChild(script)
return stream
return stream2
}
function interpolate(url, data) {
if (data == null) return url
@ -1134,11 +1141,7 @@ m.withAttr = function(attrName, callback, context) {
return callback.call(context || this, attrName in e.currentTarget ? e.currentTarget[attrName] : e.currentTarget.getAttribute(attrName))
}
}
m.prop = Stream.stream
m.prop.combine = Stream.combine
m.prop.reject = Stream.reject
m.prop.merge = Stream.merge
m.prop.HALT = Stream.HALT
m.prop = stream
m.render = renderService.render
m.redraw = redrawService.publish
m.request = requestService.xhr