-
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.
- Loading branch information
Showing
8 changed files
with
447 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,115 @@ | ||
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 | ||
|
||
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]) | ||
|
||
|
||
update : Action -> Model -> (Model, Effects Action) | ||
update action model = | ||
case action of | ||
Top act -> | ||
let | ||
(topGif', fx) = RandomGif.update act model.topGif | ||
counter' = | ||
if RandomGif.isNewGif act | ||
then Counter.increment (Button.isActive model.button) model.counter | ||
else model.counter | ||
in | ||
({ model | topGif = topGif', counter=counter'}, Effects.map Top fx) | ||
|
||
Pair act -> | ||
let | ||
(gifPair', fx) = RandomGifPair.update act model.gifPair | ||
counter' = | ||
if RandomGifPair.isNewGif act | ||
then Counter.increment (Button.isActive model.button) model.counter | ||
else model.counter | ||
in | ||
({ model | gifPair = gifPair', counter=counter'}, Effects.map Pair fx) | ||
|
||
PairOfPair act -> | ||
let | ||
(gifPairOfPair', fx) = RandomPairOfPair.update act model.gifPairOfPair | ||
counter' = | ||
if RandomPairOfPair.isNewGif act | ||
then Counter.increment (Button.isActive model.button) model.counter | ||
else model.counter | ||
in | ||
({ model | gifPairOfPair = gifPairOfPair', counter=counter'}, Effects.map PairOfPair fx) | ||
|
||
Button act -> | ||
let | ||
button' = Button.update act model.button | ||
in | ||
({ model | button = button'}, 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 = []} | ||
|
||
main : Signal Html | ||
main = app.html | ||
|
||
|
||
port tasks : Signal (Task Never ()) | ||
port tasks = app.tasks |
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,9 @@ | ||
# Approach | ||
|
||
The main strategy for this approach is to extend the API of each component to include helpers. | ||
|
||
I have put the business logic into `Counter.elm` and exposed an `increment` function that increments the model based on the button state that it receives. | ||
|
||
The RandomGif family of components API is augmented with a function that answeres the question "is this action a NewGif?" | ||
|
||
RandomGif implements the direct `NewGif` detection, each subsequent enclosure implements its own version using it's child API. |
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,107 @@ | ||
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) | ||
|
||
isNewGif : Action -> Bool | ||
isNewGif action = | ||
case action of | ||
NewGif _ -> True | ||
_ -> False | ||
|
||
update : Action -> Model -> (Model, Effects Action) | ||
update action model = | ||
case action of | ||
RequestMore -> | ||
(model, getRandomGif model.topic) | ||
|
||
NewGif maybeUrl -> | ||
( Model model.topic (Maybe.withDefault model.gifUrl maybeUrl) | ||
, 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 : Action -> Model -> (Model, Effects Action) | ||
update action model = | ||
case action of | ||
Left act -> | ||
let | ||
(left, fx) = RandomGif.update act model.left | ||
in | ||
( Model left model.right | ||
, Effects.map Left fx | ||
) | ||
|
||
Right act -> | ||
let | ||
(right, fx) = RandomGif.update 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.