FRP explanation using reactive-banana - Revision history
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&action=history
Revision history for this page on the wikienMediaWiki 1.19.14+dfsg-1Tue, 27 Sep 2016 15:40:16 GMTDavidJameson: Fixed incorrect apostrophe
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&diff=58629&oldid=prev
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&diff=58629&oldid=prev<p>Fixed incorrect apostrophe</p>
<table class='diff diff-contentalign-left'>
<tr valign='top'>
<td colspan='1' style="background-color: white; color:black;">← Older revision</td>
<td colspan='1' style="background-color: white; color:black;">Revision as of 19:13, 3 August 2014</td>
</tr></table>Sun, 03 Aug 2014 19:13:32 GMTDavidJamesonhttps://wiki.haskell.org/Talk:FRP_explanation_using_reactive-bananaMadjestic13: updated the "the worlds worst synthesizer" according to 0.5.0.0 -> 0.8.0.0 changes
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&diff=57883&oldid=prev
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&diff=57883&oldid=prev<p>updated the "the worlds worst synthesizer" according to 0.5.0.0 -> 0.8.0.0 changes</p>
<table class='diff diff-contentalign-left'>
<tr valign='top'>
<td colspan='1' style="background-color: white; color:black;">← Older revision</td>
<td colspan='1' style="background-color: white; color:black;">Revision as of 19:44, 16 April 2014</td>
</tr></table>Wed, 16 Apr 2014 19:44:17 GMTMadjestic13https://wiki.haskell.org/Talk:FRP_explanation_using_reactive-bananaJasonDagit: /* Making the example runnable */
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&diff=56160&oldid=prev
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&diff=56160&oldid=prev<p><span dir="auto"><span class="autocomment">Making the example runnable</span></span></p>
<table class='diff diff-contentalign-left'>
<tr valign='top'>
<td colspan='1' style="background-color: white; color:black;">← Older revision</td>
<td colspan='1' style="background-color: white; color:black;">Revision as of 21:08, 5 June 2013</td>
</tr></table>Wed, 05 Jun 2013 21:08:40 GMTJasonDagithttps://wiki.haskell.org/Talk:FRP_explanation_using_reactive-bananaWmeyer: type signature of eOctChange was wrong
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&diff=45228&oldid=prev
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&diff=45228&oldid=prev<p>type signature of eOctChange was wrong</p>
<table class='diff diff-contentalign-left'>
<tr valign='top'>
<td colspan='1' style="background-color: white; color:black;">← Older revision</td>
<td colspan='1' style="background-color: white; color:black;">Revision as of 18:23, 12 April 2012</td>
</tr></table>Thu, 12 Apr 2012 18:23:33 GMTWmeyerhttps://wiki.haskell.org/Talk:FRP_explanation_using_reactive-bananaPeterminten: Put an explanation/tutorial on FRP on the wiki
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&diff=45074&oldid=prev
https://wiki.haskell.org/index.php?title=FRP_explanation_using_reactive-banana&diff=45074&oldid=prev<p>Put an explanation/tutorial on FRP on the wiki</p>
<p><b>New page</b></p><div>This is an attempt to explain Functional Reactive Programming (FRP) enough to give a reader with no previous exposure to FRP an intuition what FRP is about. After reading this you should hopefully understand enough of FRP to understand the [http://www.haskell.org/haskellwiki/Reactive-banana reactive-banana] examples.<br />
<br />
FRP has certain terms such as behavior, event and time-varying that can be confusing for people unfamiliar with it. I'll avoid these terms at first and will focus on spreadsheets and a generalization of spreadsheet cells (which I will call boxes). Later, once the most important concepts are explained, reactive-banana syntax will be introduced along with an example that demonstrates how to work with behaviors and events in reactive-banana. Finally some theory about time-varying functions and how events and behaviors can be implemented using pure functions by making time explicit should provide the necessary background to understand reactive-banana's haddock comments.<br />
<br />
The version of reactive-banana used here is [http://hackage.haskell.org/package/reactive-banana-0.5.0.0 0.5.0.0].<br />
<br />
== Reactive programming for the masses: the spreadsheet ==<br />
<br />
Spreadsheets are something we all (for certain values of we) know about. Let's talk about a typical, simplified, spreadsheet. We have a list of products that we sell and want to compute their price with the Value Added Tax (VAT) added. We might have cells A1 to A10 contain the raw prices of our products and cell B1 contain the current VAT rate (say 19 for a 19% VAT). In cells C1 to C10 we'd like to see the prices including VAT.<br />
<br />
In cell C1 we'd have a formula: <tt>=A1*(1+B1/100)</tt>, in cell C2 <tt>=A2*(1+B1/100)</tt>, etc. So if A1 contains $100 C1 would contain $119.<br />
<br />
But what if the government, in it's eternal quest to reduce the budget deficit, raises the VAT rate? We'd adjust cell B1, just change it to 20. And like magic all the C cells are updated.<br />
<br />
Though this may seem mundane what we've just seen is actually a very good example of reactive programming. We didn't tell the C cells to update; they updated on their own because a value they depend on changed.<br />
<br />
== From cells to boxes: generalizing the spreadsheet ==<br />
<br />
Spreadsheets are nice, but if we want to truly get a feel for FRP we'll have to think beyond them. If we look at a spreadsheet at an abstract level it pretty much consists of cells of two types: value cells (<tt>19</tt>) and formula cells (<tt>=A1*(1+B1/100)</tt>). Let's lose the reference to spreadsheets and talk about boxes.<br />
<br />
Say, for now, that there are two kinds of boxes: formula boxes and value boxes. Both support a &ldquo;get&rdquo; operation that returns a value. Value boxes additionally support a &ldquo;set&rdquo; operation that sets the value.<br />
<br />
Formula boxes can contain any kind of pure function. They can also refer to the values of other boxes (both formula and value boxes). Value boxes don't have a function inside them, they have a value.<br />
<br />
The translation of our VAT spreadsheet would be something like a formula box ''fIncl1'' containing the expression <tt>get(vExcl1) * (1 + get(vVat) / 100)</tt>. This expression uses two value boxes: ''vExcl1'' and ''vVat''.<br />
<br />
We could also write ''fIncl1'' using a helper formula box ''fVat''. Let ''fVat'' have the formula <tt>1 + get(vVat) / 100</tt> and ''fIncl1'' have the formula <tt>get(vExcl1) * get(vVat)</tt>. I'll use <tt>:=</tt> for this kind of definition, the <tt>:=</tt> is there to remind you that this isn't Haskell.<br />
<br />
It's important to note that any kind of value may be put into value boxes, including IO actions and functions.<br />
<br />
Try doing this with a spreadsheet: <tt>fIncls := [get(ve) * get(vVat) | ve <- vExcls]</tt>. Or this: <tt>fIncl1 := apply(get(vVatFunc), get(vExcl1))</tt>.<br />
<br />
If you're wondering why I'm not using Haskell syntax, it's to focus on the meaning of boxes rather than what the functions and combinators mean. That said, this pseudo-imperative syntax is on its way out as it's getting too clunky (that <tt>apply</tt> function is really just ugly). For a quick peek ahead the last few examples would be something like this in reactive-banana:<br />
<br />
<haskell>fIncls = map (\ve -> (*) <$> ve <*> fVat) vExcls<br />
fIncl1 = fVatFunc <*> vExcl1</haskell><br />
<br />
== Events ==<br />
<br />
Let's say we want to build the worlds worst synthesizer. We have 7 buttons: &ldquo;a&rdquo;, &ldquo;b&rdquo;, &ldquo;c&rdquo;, &ldquo;d&rdquo;, &ldquo;e&rdquo;, &ldquo;f&rdquo; and &ldquo;g&rdquo;. Our output is generated by sampling a box twice per second and playing the frequency in the box until the next sample is taken.<br />
<br />
This can't be expressed with the crude formula and value boxes system we've had so far. There is no way to express key presses in that system, a key press isn't like changing a value, it's something that occurs on a specific point in time but after it occurs it's forgotten (your keyboard doesn't remember key strokes, at least mine doesn't).<br />
<br />
In this new system we'll forget about formula boxes and value boxes and introduce event boxes. Event boxes are like formula boxes in that they can refer to the value of other boxes. Event boxes can also react to events.<br />
<br />
Events can be thought of like signals in something like D-Bus. Multiple things (event boxes) can listen to them and do something once a specific event is fired (triggered).<br />
<br />
Every event has a value associated with it. Sometimes the value isn't important because the fact that the event has occurred is what's interesting but often we do want to know the value.<br />
<br />
Events come in streams. When we say event box ''b1'' changes to the value of event ''e1'' when it receives that event we're actually saying that whenever an event from the stream of events we colloquially call ''e1'' comes ''b1'' changes to the value of that event.<br />
<br />
Yes, that's confusing so I'll try to be precise. Just remember that when I refer to something like ''e1'' when defining an event box it's always to a stream of events, never a specific event.<br />
<br />
If you're puzzled by the stream just think of it as an acknowledgement that a certain kind of event can occur multiple times. It actually goes a lot deeper than that, involving those confusing <tt>[(t, e)]</tt> types, but for now just remembering that a kind of event can occur multiple times with possibly different values is good enough.<br />
<br />
Events have values and we can use that to chose to do something only for events with some value. So not only can we determine which streams of events we'll do something with when defining an event box, we can also determine for what values of events we'll do something.<br />
<br />
For example if we have an event that directly sets our synthesizer frequency we can apply a filter that only allows events with frequencies that are pleasant to the human ear.<br />
<br />
== Some reactive-banana syntax ==<br />
<br />
Expressing event handling with the pseudo-code I've used before is tricky and gets near impossible soon. So it's a good thing that once you've understood or at least got a basic idea of the concepts of event streams and event boxes the syntax of reactive-banana starts to make sense. In this section I'll explain the most fundamental functions and operators.<br />
<br />
If you're reading the [http://hackage.haskell.org/package/reactive-banana-0.5.0.0 reactive-banana 0.5 haddocks] there are a few things to keep in mind. The first is that what I've called an event box in the previous section is called a Behaviour in reactive-banana. To avoid confusion I'll stop using the term event box from here on.<br />
<br />
In the reactive-banana haddocks you'll find a lot of references to time-varying functions and lists involving time variables. Just ignore those, they're important but we'll get to them later. As a general rule just ignore what you don't understand.<br />
<br />
You'll also notice a <hask>t</hask> parameter on the <hask>Event</hask> and <hask>Behavior</hask> types. It's basically similar to the <hask>s</hask> parameter for <hask>STRef</hask>, it's a trick to use the type system to prevent constructs that would result in undefined or incorrect behavior. Just ignore it.<br />
<br />
For understanding the next sections you'll need to know about a basic subset of reactive-banana which I'll explain here. First up: event streams.<br />
<br />
=== Events ===<br />
<br />
Event streams are represented by the type Event in reactive-banana. The type <hask>Event t Int</hask> means a stream of events carrying Int values.<br />
<br />
There are three basic things you can do with just event streams. You can transform the events in them, you can filter in events in them and you can combine the event streams.<br />
<br />
Transforming an event stream means changing the values carried by the events in them. As this is ''Functional'' Reactive Programming the streams themselves are not changed. When you transform a stream you create a new stream. Whenever an event in the old stream is fired an event in the new stream is also fired, but with a different value.<br />
<br />
Transforming an event stream is done primarily by good old fmap. The expression <hask>show `fmap` eInt</hask> (or with the (<hask><$></hask>) operator <hask>show <$> eInts</hask>) with ''eInt'' having the type <hask>Event t Int</hask> creates a new event stream of the type <hask>Event String</hask> with the int in every event in the original stream being a string in the new stream.<br />
<br />
To replace the value of an event (if the event doesn't carry a useful value) just use (<hask><$</hask>): <hask>"MOO!" <$ eWhatever</hask> causes every value in the stream eWhatever to be replaced by MOO!. Like (<hask><$></hask>) this is an operator from Control.Applicative.<br />
<br />
Filtering is done using the <hask>filterE</hask> function. When you filter an event stream you create a new event stream with the same associated values but which doesn't contain all the events from the original stream. To only deal with events with positive integers you can create a filtered stream using <hask>filterE (>= 0) eInt</hask>.<br />
<br />
Combining event streams creates a new event stream with all the events from both streams. So if you combine ''eOne'' and ''eTwo'' into ''eThree'' there will be an event in ''eThree'' for every event in ''eOne'' and for every event in ''eTwo''.<br />
<br />
Combining is done with the union function: <hask>eThree = eOne `union` eTwo</hask>. Beware however that when events come in at the same time things can get a little tricky, you'd have to wonder in what order the events are processed. Reactive-banana contains several functions to handle simultaneous events. For the purpose of this page we'll do the easiest thing and ignore simultaneous events. In real world code you would need to think about it.<br />
<br />
=== Behaviors ===<br />
<br />
To create a behavior (event box) in reactive-banana you'll typically use one of two functions: <hask>stepper</hask> and <hask>accumB</hask>. Both work with an initial value and an event stream. The difference is that when an event occurs <hask>stepper</hask> changes the value of the behavior to the value in the event while <hask>accumB</hask> applies the function in the event to the value of the behavior.<br />
<br />
<haskell>eNewVal :: Event t Int<br />
bSet :: Behavior t Int<br />
bSet = stepper 0 eNewVal<br />
<br />
eUpdater :: Event t (Int -> Int)<br />
bUpdated :: Behavior t Int<br />
bUpdated = accumB 0 eUpdater</haskell><br />
The expression <hask>bSet = stepper 0 eNewVal</hask> creates a behavior named ''bSet'' with initially value 0. Once an event from the ''eNewVal'' stream comes in the value of ''bInt'' changes to the value in that event. So if an event comes in with value 2 the value of bSet becomes 2.<br />
<br />
On the other hand the expression <hask>bUpdated = accumB 0 eUpdater</hask> makes ''bUpdated'' a behavior with initially the value 0 but which gets updated (modified) whenever an event comes in. If an event comes in with value (+1) (a slice, so <hask>\x -> x + 1</hask>) and the current value of ''bUpdated'' is 1 the new value becomes 2.<br />
<br />
That's basically it for behaviors. Well, there's a third way to create behaviors: using <hask>pure</hask>. To create a behavior with the value 1 which doesn't change at all use <hask>pure 1</hask>. In case you didn't know, for applicative functors (which behaviors are) <hask>pure</hask> is what <hask>return</hask> is for monads.<br />
<br />
To create a behavior that's depends on old behaviors (<hask>f3 := get(f1) + get(f2)</hask> in our old formula box syntax) we have to use applicative functor functions in reactive-banana. There is unfortunately no option to use monad syntax. To express that the value of ''b3'' is the sum of the value of ''b1'' and the value of ''b2'' we write: <hask>b3 = (+) <$> b1 <*> b2</hask>.<br />
<br />
== Example: the worlds worst synthesizer ==<br />
<br />
Now for an example. We'd like to create a synthesizer. The synthesizer will use our keyboard for input, which we notice through a stream of events called ''eKey'' with as associated value a Char containing the key that was pressed. Something outside our program (and scope of discussion) samples the behavior ''bNote'' every 100ms and plays the tone currently in there until the next sample time.<br />
<br />
To avoid getting caught up in music theory (read: I'm lazy and can't be bothered to look up tone frequencies) the note to play is expressed as an algebraic data type.<br />
<br />
<haskell>type Octave = Int<br />
data Pitch = PA | PB | PC | PD | PE | PF | PG<br />
data Note = Note Octave Pitch<br />
<br />
-- Type signature for the key event, it comes from outside our<br />
-- system.<br />
eKey :: Event t Char</haskell><br />
You'll notice the octave. To change the octave we'll use the '-' and '+' keys. To set the pitch we'll use the 'a'..'g' keys on the keyboard. Never mind that it's really annoying to play with those keys as they're scattered all over the keyboard, this is about the FRP logic not practicality.<br />
<br />
Those chars in the ''eKey'' event stream need to be translated to pitches. Here's one way to do that.<br />
<br />
<haskell>ePitch :: Event t Pitch<br />
ePitch = (PA <$ filterE (=='a') eKey) `union`<br />
(PB <$ filterE (=='b') eKey) `union`<br />
... <br />
(PG <$ filterE (=='g') eKey)</haskell><br />
The &ldquo;trouble&rdquo; here is that we're filtering the stream multiple times, not very efficient. Here's a better way.<br />
<br />
<haskell>table = [('a', PA), ('b', PB), ..., ('g', PG)]<br />
ePitch = filterJust $ (\e -> lookup e table) <$> eKey</haskell><br />
The <hask>filterJust</hask> function is a simple helper in reactive-banana. It filters out <hask>Nothing</hask> events and returns the value inside the <hask>Just</hask> constructor for <hask>Just</hask> events. To get ''ePitch'' we first look up the characters in the translation table and then remove all events who's chars aren't in the table, removing the <hask>Just</hask> wrapper from events who's chars are in the table at the same time.<br />
<br />
The ''bNote'' behavior will not use these events directly, instead ''bOctave'' and ''bPitch'' will each store part of the note and ''bNote'' will combine the information.<br />
<br />
<haskell>eOctChange :: Char -> Just (Octave -> Octave)<br />
eOctChange c = case c of<br />
'+' -> Just (+1)<br />
'-' -> Just (subtract 1)<br />
_ -> Nothing<br />
<br />
bOctave :: Behavior t Octave<br />
bOctave = accumB 0 $ filterJust (eOctChange <$> eKey)<br />
<br />
bPitch :: Behavior t Pitch<br />
bPitch = stepper PC ePitch<br />
<br />
bNote :: Behavior t Note<br />
bNote = Note <$> bOctave <*> bPitch</haskell><br />
If you understand what's going on here you should have a basic idea of what FRP is in practice. There are of course considerations in the real world that we've skipped over here, such as how to get the keyboard event and how to play the sounds. To get a better idea of what FRP in the real world looks take a look at the [http://www.haskell.org/haskellwiki/Reactive-banana/Examples reactive-banana examples], they should be easy to follow.<br />
<br />
When following those examples you'll come across the (<hask><@</hask>) and (<hask><@></hask>) operators. I'll give a short introduction here to make it easier to understand the examples. The (<hask><@</hask>) operator is used like this: <hask>e2 = b1 <@ e1</hask>, if an event in stream ''e1'' comes in the value of that event is replaced in the ''e2'' stream by whatever value is in ''b1'' at the time. The (<hask><@></hask>) operator is used in much the same way, but it doesn't replace the value from ''e1'' outright but uses it to compute a new value.<br />
<br />
<haskell>bOne :: Behavior t Int<br />
bOne = pure 1<br />
<br />
bPlusOne :: Behavior t (Int -> Int)<br />
bPlusOne = pure (+1)<br />
<br />
eAlwaysOne, ePlusOne :: Event t Int<br />
eAlwaysOne = bOne <@ eWhatever<br />
ePlusOne = bPlusOne <@> eInt</haskell><br />
<br />
== Time-varying values and functions ==<br />
<br />
If you've read about FRP before you're likely to have come across the term &ldquo;time-varying function&rdquo;. This sounds difficult, but once you understand the basics of behaviors and events it's really no big deal.<br />
<br />
Here's the clue: a behavior contains a value, but the value can change. Therefore at different points in time a behavior can have different values. So we could say that a behavior has a value that varies in time.<br />
<br />
We could also throw away the concept of boxes and say a behavior ''is'' a value that varies in time. This is more correct, those boxes are helpful as teaching concepts but once we talk directly about time they are no longer needed.<br />
<br />
So, a time-varying value is simply a behavior as behaviors can have different values at different points in time. A time-varying function is also just a behavior, one where the value is a function (functional programming 101: the clue to every riddle is that functions are values).<br />
<br />
To go further down the rabbit hole a time-varying value can actually be thought of as a function by making time explicit. If a behavior has value 1 up to the 30th's second and from that point forward value 2 we could express the behavior as: <hask>\t -> if t < 30 then 1 else 2</hask>. This is important: by making time explicit we can reason about behaviors as if they were pure functions. While in practice we're dealing with applicative functors (or in other libraries monads or arrows) we can think of behaviors as pure functions.<br />
<br />
Real world behaviors aren't as simple as from 30 seconds onwards change to value 2. They interact with events. So to express such behaviors as pure functions events need to be expressed in a way that works for pure functions. This is where the <hask>[(t,e)]</hask> type comes in. We can see events as a list of values at certain points in time, for example <hask>[(10, 1), (20, 2), (30, 3)]</hask> for events that occur on second 10, 20 and 30 with values 1, 2 and 3 respectively.<br />
<br />
When viewing events in such a way it becomes easy to create a behavior that changes to whatever value was last:<br />
<br />
<haskell>type Time = Int<br />
stepped :: [(Time, Int)] -> Time -> Int<br />
stepped es t = case takeWhile (\(t', _) -> t' < t) es of <br />
[] -> 0 <br />
xs -> snd (last xs)</haskell><br />
For once this is actually runnable code. If we invoke it as <hask>stepped [(10,1),(20,1),(30,1)] 2</hask> the result is 0, if we invoke it as <hask>stepped [(10,1),(20,1),(30,1)] 12</hask> the result is 1, as expected.<br />
<br />
Stepped sounds a lot like stepper and we can create that function by making a few small adjustments.<br />
<br />
<haskell>type Time = Int<br />
stepper :: a -> [(Time, a)] -> (Time -> a)<br />
stepper d es = \t -> case takeWhile (\(t', _) -> t' < t) es of<br />
[] -> d<br />
xs -> snd (last xs)</haskell><br />
If you understand this bit, why behaviors and events can be expressed by making time explicit you have a good intuition of what FRP is.<br />
<br />
== Making the example runnable ==<br />
<br />
After I wrote this page I got questions about how to actually run the reactive-banana examples. Here's a big block of code that can be pasted into a file and run:<br />
<br />
<haskell>import Data.Char (toUpper)<br />
import Control.Monad (forever)<br />
import System.IO (BufferMode(..), hSetEcho, hSetBuffering, stdin)<br />
import Reactive.Banana<br />
<br />
type Octave = Int<br />
<br />
data Pitch = PA | PB | PC | PD | PE | PF | PG<br />
deriving (Eq, Enum)<br />
<br />
-- Mapping between pitch and the char responsible for it.<br />
pitchChars :: [(Pitch, Char)]<br />
pitchChars = [(p, toEnum $ fromEnum 'a' + fromEnum p) |<br />
p <- [PA .. PG]]<br />
<br />
-- Reverse of pitchChars<br />
charPitches :: [(Char, Pitch)]<br />
charPitches = [(b, a) | (a, b) <- pitchChars]<br />
<br />
data Note = Note Octave Pitch<br />
<br />
instance Show Pitch where<br />
show p = case lookup p pitchChars of<br />
Nothing -> error "cannot happen"<br />
Just c -> [toUpper c]<br />
<br />
instance Show Note where<br />
show (Note o p) = show p ++ show o<br />
<br />
-- Filter and transform events at the same time.<br />
filterMapJust :: (a -> Maybe b) -> Event t a -> Event t b<br />
filterMapJust f = filterJust . fmap f<br />
<br />
-- Change the original octave by adding a number of octaves, taking<br />
-- care to limit the resulting octave to the 0..10 range.<br />
changeOctave :: Int -> Octave -> Octave<br />
changeOctave d = max 0 . min 10 . (d+)<br />
<br />
-- Get the octave change for the '+' and '-' chars.<br />
getOctaveChange :: Char -> Maybe Int<br />
getOctaveChange c = case c of<br />
'+' -> Just 1<br />
'-' -> Just (-1)<br />
_ -> Nothing<br />
<br />
makeNetworkDescription :: AddHandler Char<br />
-> NetworkDescription t ()<br />
makeNetworkDescription addKeyEvent = do<br />
eKey <- fromAddHandler addKeyEvent<br />
let<br />
eOctaveChange = filterMapJust getOctaveChange eKey<br />
bOctave = accumB 3 (changeOctave <$> eOctaveChange)<br />
ePitch = filterMapJust (`lookup` charPitches) eKey<br />
bPitch = stepper PC ePitch<br />
bNote = Note <$> bOctave <*> bPitch<br />
eNoteChanged <- changes bNote<br />
reactimate $ (\n -> putStrLn ("Now playing " ++ show n))<br />
<$> eNoteChanged<br />
<br />
main = do<br />
(addKeyEvent, fireKey) <- newAddHandler<br />
let networkDescr = makeNetworkDescription addKeyEvent<br />
network <- compile networkDescr<br />
actuate network<br />
hSetEcho stdin False<br />
hSetBuffering stdin NoBuffering<br />
forever (getChar >>= fireKey)</haskell><br />
If you compare this to the previous example you'll notice some changes in the events and behaviors. This version is more real-world, there are bounds checks and some abstraction (<hask>filterMapJust</hask>).<br />
<br />
The parts answering the question &ldquo;how to run it?&rdquo; are all in <hask>makeNetworkDescription</hask> and <hask>main</hask>. In reactive-banana you first create a description of an event network, then compile that network, enable it and call your normal Haskell event loop. At least, that's the case when you don't use a GUI framework with reactive-banana integration, if you use the wx integration for reactive-banana the event loop is already written for you and some other details are different. See the [http://www.haskell.org/haskellwiki/Reactive-banana/Examples examples].<br />
<br />
All the events and behaviors live in what reactive-banana calls an event network. It's not possible to use behaviors outside this network, you can't for example see inside a behavior from IO code. You can use events though and that's how you couple the FRP stuff inside the event network to the outside world.<br />
<br />
To create an event that you can fire from outside the event network use <hask>newAddHandler</hask> in the IO monad. It gives you two things, an <hask>AddHandler</hask> value and an IO action that takes a value and fires an event with that value. By passing the <hask>AddHandler</hask> value to <hask>fromAddHandler</hask> (inside the <hask>NetworkDescription</hask> monad) you'll get an event stream which you can use in a normal fashion.<br />
<br />
To handle output events use <hask>reactimate</hask>, pass it an event stream containing IO actions and whenever an event from that stream occurs the action is executed. You'll pretty much always want to use <hask>reactimate</hask> with (<hask><$></hask>)/<hask>fmap</hask> to generate those IO actions.<br />
<br />
Once the network description is complete you can compile it. This gives you back an <hask>EventNetwork</hask> value. You can start the <hask>EventNetwork</hask> with <hask>actuate</hask>. This sets things up so that events can actually be received.<br />
<br />
In our roll-your-own program we'll use <hask>getChar</hask> (after turning off echo and buffering on the standard input stream) to receive characters. When a character is received the ''bNote'' is updated and a message is printed with the new value (actually playing a tone appears to involve more code than I'd like to include here).<br />
<br />
Wait? Didn't I say we can't actually observe the value of a behavior? So how do we know which note to play? Turns out reactive-banana has a nice little function called <hask>changes</hask>. When used on a behavior it returns an event stream which contains the new values whenever the behavior is updated. This event stream is passed to <hask>reactimate</hask> to trigger the prints.<br />
<br />
== A broader view of FRP ==<br />
<br />
FRP is a field which is currently in active development, many different approach are being tried and no real standard has been found yet. The [http://hackage.haskell.org/packages/archive/pkg-list.html#cat:frp FRP category on hackage] has a lot of different packages with varying approaches to implementing FRP. Some focus on applicative functors, some on monads and others on arrows.<br />
<br />
Semantics can be subtly or not-so-sublty different between FRP libraries and sometimes naming is different as well (for example signal instead of event).<br />
<br />
In this page I've tried to explain FRP using one specific library (reactive-banana). While this makes things simpler for getting an idea of FRP it can give some wrong impressions due to conflating the limitations of reactive-banana and that of FRP in general.<br />
<br />
[[Category:FRP]] [[Category:Tutorials]]</div>Sat, 31 Mar 2012 14:17:03 GMTPetermintenhttps://wiki.haskell.org/Talk:FRP_explanation_using_reactive-banana