I'm wondering how reactive-banana was designed to handle the following situation:
Let's say I have a central data structure. The user is able to freely open and close any number of different types of windows that both display the data, and allow the user to modify it.
So, given the nature of this, I don't think just trying to create one big network would work well. Is this something where each window would have a network and somehow they were connected?
In other situations like this, I had put the data structure behind a single channel that everyone sent updates to. And then the data structure would "publish" updates (fire events) that the windows all "listened" to.
I have a related problem in one of my projects, which uses something akin to an MVC architecture. The central data structure is referenced as
Here is a stripped-down version of my controller declaration:
When constructed, most controllers get a reference to
dZip :: Discrete MyDataZip
, but have no way to modify it directly. The only way for a controller to specify an update is via theeUpdateZip
stream within the Controller data structure.Multiple controllers are assembled into a graph, which is just a list of Controllers placed in an existential type wrapper
data EControl = forall st. EControl (Controller st)
. I justmconcat
all of the individualeUpdateZip
parameters to get a single stream ofEvent (MyDataZip -> MyDataZip)
which is used to create theDiscrete MyDataZip
. The whole thing works because creating new controllers is pure, so it can be done in the same let-binding, allowing the recursive reference.Opening new windows and other IO tasks are done in the
eResponse
stream, which is similarlymconcat
d and then passed toreactimate
.You can examine the darcs repo for more details, look in "src/Jaek/UI/Controllers/" and "src/Jaek/UI/ControlGraph.hs".
EDIT: The central problem is that one big network is fairly unwieldy from a developer's POV. You can reduce the complexity by segmenting the networks, which is a good solution. In my design, I introduced a lot of structure to the network by making reactive participants conform to a specific controller model, and creating well-defined means for those models to interact. Since my controllers are persistent that's all set up statically, but it could be done dynamically, in which case I'd probably have controller manager running in one thread, and actions which generated new controllers (such as opening a new window) would send a message to the thread to create a new controller.