Functional User Interfaces
(please change slides with <space>)
Composability
Abstraction
## In a nutshell
* Displaying stuff
* Reacting to user input
## What we need
* Reusable
* Composable
* Flexible
## Android XML
data:image/s3,"s3://crabby-images/1699a/1699a02669cb0c7fe0d0e69a7774ec56b8d54e41" alt="xml"
* Awkward modularity
## Javascript {{{}}}-s
data:image/s3,"s3://crabby-images/15f15/15f15a4e71ad6b0d270fe61634e4c00ae1935810" alt="{{}}"
* String-based? Duh...
## ScalaTags [*](https://github.com/lihaoyi/scalatags)
* Just Scala code
* Use values and modules
## Demo
[data:image/s3,"s3://crabby-images/cfc27/cfc271dba70ecd0e20f644918e2ea1f1d1982c4a" alt="demo"](http://www.scala-js-fiddle.com/gist/e30f7c897b4ca822ca37)
## Hiccup [*](https://github.com/weavejester/hiccup)
### same idea in Clojure...
```clojure
(html [:span {:class "foo"} "bar"])
```
## Dommy [*](https://github.com/Prismatic/dommy)
### and again...
```clojure
(node
[:div#id.class1
(for [r (range 2)]
[:span.text (str "word" r)])])
```
## Teacup [*](https://github.com/goodeggs/teacup)
### and in CoffeeScript...
```coffeescript
output = render ->
ul ->
li 'Bergamot'
li 'Chamomile'
```
The
Flying
Spaghetti
Code
Monster
## Wikipedia on Spaghetti
data:image/s3,"s3://crabby-images/bc1ee/bc1ee688b83a76d584ea1e4d27165547e1ff96c8" alt="spag"
data:image/s3,"s3://crabby-images/69871/69871684f78155bb476fb892eb4db48523479694" alt="rav"
## UI is the new black
data:image/s3,"s3://crabby-images/0d369/0d36995173ebad444577d403fba72fbc479b928e" alt="neri"
## Adobe on UI [*](http://stlab.adobe.com/wiki/images/0/0c/Possible_future.pdf)
* 1/3 code is event handling
* 1/2 bugs are in that code
## Establishing connections
data:image/s3,"s3://crabby-images/c0e25/c0e2590849ae6bdf3332340c92282d0ee2726d53" alt="moog"
## Remember MS Excel?
data:image/s3,"s3://crabby-images/be49d/be49db9ee8b94866934cf72eb6d30241fb582290" alt="excel"
## Signals
* Mouse is a signal
* Textbox is a slot
* We need the cable!
## Elm [*](http://elm-lang.org/)
* Signal-based language
* Compiles to HTML+CSS+JS
## Demo
[data:image/s3,"s3://crabby-images/88c45/88c453f9f540a786973fc33036addec25604a8ed" alt="elm"](http://elm-lang.org/edit/examples/Reactive/Position.elm)
## A complete example
[data:image/s3,"s3://crabby-images/49a21/49a2184cc6100b19909bc6bf36435f10b6661b0e" alt="elm"](http://elm-lang.org/edit/examples/Intermediate/Physics.elm)
## Event Streams
* Keypress is an event
* Transform it
* Connect to a textbox
## bacon.js
* JavaScript library
* Turns DOM events into transformable streams
## Demo
[data:image/s3,"s3://crabby-images/5b8e3/5b8e322f4555be2b11b23bc50159e086b7740788" alt="bacon"](http://baconjs.github.io/)
## Observables
### same idea
* Reactive Extensions [*](https://rx.codeplex.com/)
* RxJava [*](https://github.com/Netflix/RxJava)
# Meanwhile, in our phones
## Macroid [*](https://github.com/macroid/macroid)
> Experimental modular functional UI language for Android, written in Scala.
## Bricks
```scala
w[Button]
```
## Bricks
```scala
l[LinearLayout](
w[Button],
w[TextView]
)
```
## Bricks
```scala
val brick1 = w[Button]
val brick2 = w[TextView]
l[LinearLayout](
brick1, brick2
)
```
## Tweaks
```scala
w[TextView] <~ text("Hello")
```
## Tweaks
```scala
def largeText(str: String) =
text(str) +
TextSize.large +
padding(left = 8 dp)
myTextView <~ largeText("Hello")
```
## A complete layout
```scala
var greeting = slot[TextView]
l[LinearLayout](
w[Button] <~
text("Greet me!") <~
On.click {
greeting <~ show
},
w[TextView] <~
text("Hello there") <~
wire(greeting) <~ hide
)
```
## Snails
```scala
myTextView <~~ fadeIn(400)
```
## Snails
```scala
myTextView <~~
fadeIn(400) <~
text("Faded!")
```
## Snails
```scala
val blink =
fadeIn(400) ++
delay(2000) ++
fadeOut(400)
myTextView <~~ blink
```
## UI actions
```scala
val action =
myTextView <~ text("Foo") <~ show
```
## UI actions
```scala
val action1 =
myTextView <~ text("Foo") <~ show
val action2 =
myProgressBar <~ hide
runUi(action1 ~ action2)
```
## UI actions
```scala
runUi {
(myProgressBar <~~ fadeOut(400)) ~~
(myTextView <~~ blink) ~~
(myOtherTextView <~ text("’SUP?"))
}
```
## Tweaks & Signals
```scala
// create a reactive variable
val caption = rx.Var("Olá")
// set text to “Olá”
myTextView <~ caption.map(text)
// text automatically updates to “Adeus”
caption.update("Adeus")
```
## Drawing a figure
* Wait till mouse is down
* Draw while it moves
* Finish once mouse is up
## Event Stream Workflows [*](http://infoscience.epfl.ch/record/176887/files/DeprecatingObservers2012.pdf)
```scala
Reactor.loop { self ⇒
// step 1
val path = new Path((self await mouseDown).position)
// step 2
self.loopUntil(mouseUp) {
val m = self awaitNext mouseMove
path.lineTo(m.position)
draw(path)
}
// step 3
path.close()
draw(path)
}
```
## Demo
[data:image/s3,"s3://crabby-images/a4675/a467514a625823485785e36658b556b45b9e590d" alt="flow"](http://www.scala-js-fiddle.com/gist/9759723/ScalaAsyncPaintbrush.scala)
## Further Reading
* See [*](#)-s in the slides
* [The “Deprecating Observers” paper (must read!)](http://infoscience.epfl.ch/record/176887/files/DeprecatingObservers2012.pdf)
* [A comprehensive overview of Haskell GUI libraries](http://www.slideshare.net/GideonSireling/graphical-user-interfaces-in-haskell-57554808)
This presentation: http://stanch.github.io/funlx-meetup-2