Skip to content

Commit

Permalink
Merge pull request #4 from fredcy/mailbox
Browse files Browse the repository at this point in the history
Variation where components signal up via a context.
  • Loading branch information
slorber committed Mar 3, 2016
2 parents d0946d6 + ddb98dd commit a946435
Show file tree
Hide file tree
Showing 8 changed files with 477 additions and 0 deletions.
28 changes: 28 additions & 0 deletions upward-message/Button.elm
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
module Button where

import Html exposing (..)
import Html.Attributes exposing (style)
import Html.Events exposing (onClick)

type Action = Click

type alias Model = Bool

init : Model
init = True

update : Action -> Model -> Model
update action model = not model

view : Signal.Address Action -> Model -> Html
view address model =
button
[ style [("backgroundColor", if model then "green" else "red")]
, onClick address Click
]
[ text "Click"]



isActive : Model -> Bool
isActive model = model
28 changes: 28 additions & 0 deletions upward-message/Counter.elm
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
module Counter where

import Html exposing (..)

type Action = Increment Bool

type alias Model = Int

init : Model
init = 0

update : Action -> Model -> Model
update action model =
case action of
Increment True ->
if model >= 10 then model + 2 else model + 1
Increment False ->
model + 1

increment : Bool -> Model -> Model
increment button model =
update (Increment button) model


view : Signal.Address Action -> Model -> Html
view address model =
div []
[ text <| toString model]
146 changes: 146 additions & 0 deletions upward-message/Main.elm
Original file line number Diff line number Diff line change
@@ -0,0 +1,146 @@
module Main (..) where

import Html exposing (..)
import StartApp
import Task exposing (Task)
import Effects exposing (Effects, Never)
import RandomGif
import RandomGifPair
import RandomPairOfPair
import Button
import Counter


type Action
= Top RandomGif.Action
| Pair RandomGifPair.Action
| PairOfPair RandomPairOfPair.Action
| Button Button.Action
| Counter Counter.Action
| NewGif


type alias Model =
{ topGif : RandomGif.Model
, gifPair : RandomGifPair.Model
, gifPairOfPair : RandomPairOfPair.Model
, button : Button.Model
, counter : Counter.Model
}


init : ( Model, Effects Action )
init =
let
( topGif, topFx ) =
RandomGif.init "dogs"

( gifPair, pairFx ) =
RandomGifPair.init "cats" "lemurs"

( gifPairOfPair', pairOfPaiFx ) =
RandomPairOfPair.init "unicorns" "minions" "pokemon" "lizards"

model =
{ topGif = topGif
, gifPair = gifPair
, gifPairOfPair = gifPairOfPair'
, button = Button.init
, counter = Counter.init
}
in
( model, Effects.batch [ Effects.map Top topFx, Effects.map Pair pairFx, Effects.map PairOfPair pairOfPaiFx ] )


context =
Signal.forwardTo actionsMailbox.address (always NewGif)


update : Action -> Model -> ( Model, Effects Action )
update action model =
case action of
Top act ->
let
( topGif', fx ) =
RandomGif.update context act model.topGif
in
( { model | topGif = topGif' }, Effects.map Top fx )

Pair act ->
let
( gifPair', fx ) =
RandomGifPair.update context act model.gifPair
in
( { model | gifPair = gifPair' }, Effects.map Pair fx )

PairOfPair act ->
let
( gifPairOfPair', fx ) =
RandomPairOfPair.update context act model.gifPairOfPair
in
( { model | gifPairOfPair = gifPairOfPair' }, Effects.map PairOfPair fx )

Button act ->
let
button' =
Button.update act model.button
in
( { model | button = button' }, Effects.none )

NewGif ->
let
counter' =
Counter.increment (Button.isActive model.button) model.counter
in
( { model | counter = counter' }, Effects.none )

_ ->
( model, Effects.none )


view : Signal.Address Action -> Model -> Html
view address model =
let
fwd =
Signal.forwardTo address

sep =
br [] []
in
div
[]
[ RandomGif.view (fwd Top) model.topGif
, sep
, RandomGifPair.view (fwd Pair) model.gifPair
, sep
, RandomPairOfPair.view (fwd PairOfPair) model.gifPairOfPair
, sep
, Button.view (fwd Button) model.button
, sep
, Counter.view (fwd Counter) model.counter
]


app : StartApp.App Model
app =
StartApp.start
{ init = init
, update = update
, view = view
, inputs = [ actionsMailbox.signal ]
}


main : Signal Html
main =
app.html


port tasks : Signal (Task Never ())
port tasks =
app.tasks


actionsMailbox : Signal.Mailbox Action
actionsMailbox =
Signal.mailbox NewGif
5 changes: 5 additions & 0 deletions upward-message/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
# Approach

This version blatantly copies Peter's as a basis.

Instead of intercepting subcomponent events at the top, this approach passes an address down to the components for them to send a message when a new GIF is obtained. The main watches this signal and increments the counter each time it gets a message.
110 changes: 110 additions & 0 deletions upward-message/RandomGif.elm
Original file line number Diff line number Diff line change
@@ -0,0 +1,110 @@
module RandomGif where

import Effects exposing (Effects, Never)
import Html exposing (..)
import Html.Attributes exposing (style)
import Html.Events exposing (onClick)
import Http
import Json.Decode as Json
import Task


-- MODEL

type alias Model =
{ topic : String
, gifUrl : String
}


init : String -> (Model, Effects Action)
init topic =
( Model topic waiting
, getRandomGif topic
)

-- UPDATE

type Action
= RequestMore
| NewGif (Maybe String)
| NoOp

isNewGif : Action -> Bool
isNewGif action =
case action of
NewGif _ -> True
_ -> False

update : Signal.Address () -> Action -> Model -> (Model, Effects Action)
update context action model =
case action of
RequestMore ->
(model, getRandomGif model.topic)

NewGif maybeUrl ->
( Model model.topic (Maybe.withDefault model.gifUrl maybeUrl)
, Signal.send context () |> Task.map (always NoOp) |> Effects.task
)

NoOp ->
( model, Effects.none )

-- VIEW

(=>) : a -> b -> ( a, b )
(=>) = (,)


view : Signal.Address Action -> Model -> Html
view address model =
div [ style [ "width" => "200px" ] ]
[ h2 [headerStyle] [text model.topic]
, div [imgStyle model.gifUrl] []
, button [ onClick address RequestMore ] [ text "More Please!" ]
]


headerStyle : Attribute
headerStyle =
style
[ "width" => "200px"
, "text-align" => "center"
]


imgStyle : String -> Attribute
imgStyle url =
style
[ "display" => "inline-block"
, "width" => "200px"
, "height" => "200px"
, "background-position" => "center center"
, "background-size" => "cover"
, "background-image" => ("url('" ++ url ++ "')")
]

waiting : String
waiting = ""

-- EFFECTS

getRandomGif : String -> Effects Action
getRandomGif topic =
Http.get decodeUrl (randomUrl topic)
|> Task.toMaybe
|> Task.map NewGif
|> Effects.task


randomUrl : String -> String
randomUrl topic =
Http.url "http://api.giphy.com/v1/gifs/random"
[ "api_key" => "dc6zaTOxFJmzC"
, "tag" => topic
]


decodeUrl : Json.Decoder String
decodeUrl =
Json.at ["data", "image_url"] Json.string
71 changes: 71 additions & 0 deletions upward-message/RandomGifPair.elm
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
module RandomGifPair where

import Effects exposing (Effects)
import Html exposing (..)
import Html.Attributes exposing (..)

import RandomGif


-- MODEL

type alias Model =
{ left : RandomGif.Model
, right : RandomGif.Model
}


init : String -> String -> (Model, Effects Action)
init leftTopic rightTopic =
let
(left, leftFx) = RandomGif.init leftTopic
(right, rightFx) = RandomGif.init rightTopic
in
( Model left right
, Effects.batch
[ Effects.map Left leftFx
, Effects.map Right rightFx
]
)


-- UPDATE

type Action
= Left RandomGif.Action
| Right RandomGif.Action

isNewGif : Action -> Bool
isNewGif action =
case action of
Left act -> RandomGif.isNewGif act
Right act -> RandomGif.isNewGif act

update : Signal.Address () -> Action -> Model -> (Model, Effects Action)
update context action model =
case action of
Left act ->
let
(left, fx) = RandomGif.update context act model.left
in
( Model left model.right
, Effects.map Left fx
)

Right act ->
let
(right, fx) = RandomGif.update context act model.right
in
( Model model.left right
, Effects.map Right fx
)


-- VIEW

view : Signal.Address Action -> Model -> Html
view address model =
div [ style [ ("display", "flex") ] ]
[ RandomGif.view (Signal.forwardTo address Left) model.left
, RandomGif.view (Signal.forwardTo address Right) model.right
]
Loading

0 comments on commit a946435

Please sign in to comment.