-
Notifications
You must be signed in to change notification settings - Fork 29
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #4 from fredcy/mailbox
Variation where components signal up via a context.
- Loading branch information
Showing
8 changed files
with
477 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 | ||
] |
Oops, something went wrong.