uneventful
root
job to replace detached
(which is now deprecated). Creating root-based rather than detached jobs means there is a single point from which all resources can be cleaned up.uneventful/shared
(NEW)
service()
: wrap a factory function to create a singleton service accessorfork()
: wrap a generator, generator function, or generator method to run in parallel, and have a result that can be waited on in parallel as wellexpiring()
: proxy an object so it cannot be accessed after the calling job endsuneventful/signals
rule.root
to replace rule.detached
(which is now deprecated).edit()
method to writable signals (to patch the existing value using a function)peek()
or action()
not being able to access the job of the enclosing rule, if it hadn't already been useduneventful/utils
decorateMethod()
: a helper for creating hybrid (TC39/legacy) decorator/function wrappersisGeneratorFunction()
to check for native generator functiontask
decorator was passing the job as an extra argument to the wrapped functionuneventful/signals
Any computed signal (i.e. a cached()
function or a value()
with a .setf()
) can now start jobs or register cleanups for their side-effects. (Previously, only rules could do this.)
The jobs are ended (or cleanups run) when the signal ceases to have subscribers, or when the values the signal depends on change. (Unobserved signals with jobs are also recalculated if they gain subscribers later, even if none of their dependencies have changed. This is so their side-effects will be restored without needing to wait for a change in their dependencies.)
You can also use the new isObserved()
function (from the uneventful
main package) to test whether the current code is running inside of an observed signal or rule, with the side-effect that if the signal is not currently observed, then it will be recalculated when it becomes observed. (This lets you avoid setting up jobs or cleanup-needing effects that will be immediately discarded due to a lack of subscribers, but still set them up as soon as there is demand for them.)
Added unchangedIf()
: allows reactive expressions (in cached()
or value().setf()
) to return their previous value if the new value is equivalent according to a custom comparison function (arrayEq()
by default)
Backward incompatibility: Removed the stop
parameter from rule functions, so that signals and zero-argument states can be used as rule actions. (Use rule.stop()
instead.)
uneventful/utils
batch()
factory for creating generic batch processorsGeneratorBase
for identifying generators with instanceof
arrayEq()
for comparing array contentsuneventful/signals
) and exposed the utils module as an export (uneventful/utils
).RuleFactory
interface:
rule.stop
can now be saved and then called from outside a rulerule.detached(...)
is a new shorthand for detached.run(rule, ...)
rule.setScheduler()
lets you change how a rule will be scheduled (from inside it)recalcWhen()
handle subscribes and unsubscribes so that they don't thrash on and off when there's a single subscriber that's synchronously removed and re-added.next()
and until()
should not resume job with the rule still activenoDeps()
has been renamed peek()
Signal
and Writable
are now interfaces instead of classesrule.if()
API@rule.method
decorator (with TC39/legacy decorator autodetection)rule.factory()
replaces RuleScheduler.for()
rule.stop()
stops the active rulerunRules()
can be given a scheduling function to flush a specific queueuntil()
API has now been split: next()
is used to get a signal or stream's next value, while until()
yields the next truthy value, and auto-converts zero-argument functions to signals. until()
also does not work on promises any more, since start()
, to()
and fromPromise()
all accept promises already.task(fn)
/@task
, and action(fn)
/@action
. Both support both the TC39 and legacy decorator protocols.Source
is now Stream
, and the type previously called Producer
is now Source
. This helps make the documentation clearer on some points.value()
objects now have a .setf()
method that can be used to set them to a "formula" (callback expression), not unlike a spreadsheet cell. Regular .set()
overwrites the formula with a value, and .setf()
replaces the value with a formula. This makes it easier to implement components with configurable signal bindings.lazy()
stream not forwarding its inletstart()
callbackforEach()
APIuntil()
rules should be ended when the job resumes