A simple library for working with immutable data in mutable+observable MV* applications.
bower install --save https://github.com/iamnoah/mu.git
Functional programming makes programs easier to reason about. Add in persistent data structures and whole classes of errors can be ruled out, and performance can be optimized easily. But at some point, you need to mutate some state (even in Haskell.) Being able to observe and bind state changes reduces boilerplate and simplifies application logic.
mu tries to give you tools for finding a happy place between pure functional programming and imperative state mutation. With mu, you create a few persistent data structures, but have observable updates, and can effectively mutate your data without having to manually copy a bunch of objects. You can make your bindings as coarse or as fine grained as you need to, while still getting the benefits of functional, persistent data structures.
Atoms are an interface for manipulating persistent data structures and creating new Atoms focused on specific parts of the data. e.g.,
var atom = new Atom(myData);
var baz = atom.focus("foo", "array", 2, "bar", "baz");
baz.get() === myData.foo.array[2].bar.baz
baz.set(newValue)
atom.get() === { foo: [..., ..., { bar: { baz: newValue } }] }
mu does not directly depend on a compute implementation, so you must wrap the state yourself if you want to observe it.
var dataCompute = compute(myData);
var atom = new Atom(dataCompute);
dataCompute.bind("change", function() {...})
var baz = atom.focus("foo", "array", 2, "bar", "baz");
baz() === dataCompute().foo.array[2].bar.baz
baz.set(newValue) // dataCompute is updated and change handler is called
// you can also create a compute from baz
var bazCompute = compute(baz);
bazCompute.bind("change", function() {...});
atom.set({...}) // triggers an update of bazCompute
mu should hapilly run in any ES5 environement. i.e., IE9+, or node.
focus
with get
and set
should be able to handle 90% of what you need to do with atoms in a very DRY way. There are additional helper methods for the other 10%:
// Delete a Key
var dataCompute = compute({
foo: 123,
bar: "delete me"
});
var atom = new Atom(dataCompute);
atom.focus("bar").del();
atom() === { foo: 123 }
// extend/assign
atom.extend({
baz: 123,
});
atom() === { foo: 123, baz: 123 }
// Append to an array:
var dataCompute = compute({
qux: ["hi"]
});
var atom = new Atom(dataCompute);
atom.focus("qux").push("mom");
atom() === { qux: ["hi", "mom!"] }
// Remove from an array:
var dataCompute = compute({
qux: ["hi", "there", "mom"]
});
var atom = new Atom(dataCompute);
atom.focus("qux", 1).del();
atom() === { qux: ["hi", "mom"] }
// in addition to push, there is pop, shift, and unshift for arrays
// if you are using an array as a set, merge and remove are helpful:
atom.focus("array").set([1,2,3]);
atom.focus("array").merge([3,4,5]);
array === [1,2,3,4,5]
atom.focus("array").remove([3,5]);
array === [1,2,4]
// merge and remove both will take a property name or iterator as the
// 2nd argument to use as the identity.
atom.focus("array").set([{
id: "foo",
}, {
id: "bar",
}]);
atom.focus("array").merge([{
id: "foo",
}, {
id: "baz",
}], "id");
// now contains foo, bar and baz
Sometimes you may need to change several values at once in order for the atom to remain consistent. e.g., suppose you have an atom that stores the user's cursor position in a text editor and the current text. When the user types a character, if you update the text first, the cursor position will be wrong, but if you update the cursor posiition first, that is also wrong because the cursor would be after a character that doesn't yet exist in the text. You need to set both at the same time.
If you need to atomically change more than 1 value within the atom, use update:
atom.update(function(newAtom) { // must use the passed in atom
newAtom.focus("foo").set(123);
newAtom.focus("bar", "baz").set("abc");
});
Note that the update function receives a new atom. Only the new atom should be used to get/set values. The original atom will not show any changes until the update function returns. e.g.,
atom.update(function(newAtom) { // only use newAtom, not the out atom!
newAtom.focus("foo").set(123);
atom.focus("foo").get() !== 123 // BUG
atom.focus("foo").set(456) // BUG! this change will fire immediately, but be lost after this function returns!
newAtom.focus("bar", "baz").set("abc");
});
Atomic updates are recommended whenever you need to change more than one property at the same time.
Sometimes you may need to intercept changes to the atom before they are committed. e.g., a change to one part of the data structure requires that another part be updated to remain consistent. If you make the change after the atom is updated, any observers will see an inconsistent state, potentially causing bugs.
To fix this, you can use the atom's beforeChange
method:
atom.beforeChange(function(newData, oldData) {
if (newData.foo !== oldData.foo) {
// remember that newData is immutable, so you have to make a copy!
return Object.assign({}, newData, {
fooChangeCount: newData.fooChangeCount + 1,
});
}
// whatever value you return will be the new value of the atom
return newData;
});
Atom uses functional lenses internally to create focused computes. You can pass your own Lenses to focus and they will be inserted into the path.
TODO example
Need undo/redo functionality? Use the built in Undo helper:
var history = mu.Undo(atom);
atom.set(...);
history.reset(); // optional - if you get in a state you can't undo, reset to clear the history
history.undo();
history.redo();
By default, all changes will be kept. Since persistent data structures reuse most of their references, each additional change is generally very small, as you typically just need O(log n)
new pointers. But if your state changes constantly and/or you have memory problems, you can limit how many states are kept, and how frequently new states are recorded:
mu.Undo(atom, {
maxStates: 20, // keep only the last 20 states
timeBetweenStates: 2000, // save state at most every 2 seconds
})
timeBetweenStates
will make the history "lossy" in that an undo may revert multiple distinct changes, if they all happened within that time period. This can be good if, for example, you were to update your atom on every keystroke, so your user does not have to undo every single keystroke.
You can also filter out undo states that are not important, like transient state changes and internal book keeping.
mu.Undo(atom, {
isImportantChange: function(newState, oldState) {
return true/false;
},
})
If your function returns false, no entry will be made for that change, but the next truely important change will include whatever unimportant state changes happened.
So which is better, functional programming or imperative programming? mu