Skip to content

Commit

Permalink
Elm version.
Browse files Browse the repository at this point in the history
  • Loading branch information
pdamoc committed Mar 2, 2016
1 parent 82f94b4 commit 30ee5d0
Show file tree
Hide file tree
Showing 8 changed files with 447 additions and 0 deletions.
28 changes: 28 additions & 0 deletions elm-api-extend/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 elm-api-extend/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]
115 changes: 115 additions & 0 deletions elm-api-extend/Main.elm
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
9 changes: 9 additions & 0 deletions elm-api-extend/README.md
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.
107 changes: 107 additions & 0 deletions elm-api-extend/RandomGif.elm
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
71 changes: 71 additions & 0 deletions elm-api-extend/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 : 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
]
Loading

0 comments on commit 30ee5d0

Please sign in to comment.