From 30ee5d0352eb990b9649cc2158bbe66cdd98658c Mon Sep 17 00:00:00 2001 From: Peter Damoc Date: Wed, 2 Mar 2016 22:35:40 +0200 Subject: [PATCH] Elm version. --- elm-api-extend/Button.elm | 28 +++++++ elm-api-extend/Counter.elm | 28 +++++++ elm-api-extend/Main.elm | 115 ++++++++++++++++++++++++++++ elm-api-extend/README.md | 9 +++ elm-api-extend/RandomGif.elm | 107 ++++++++++++++++++++++++++ elm-api-extend/RandomGifPair.elm | 71 +++++++++++++++++ elm-api-extend/RandomPairOfPair.elm | 71 +++++++++++++++++ elm-api-extend/elm-package.json | 18 +++++ 8 files changed, 447 insertions(+) create mode 100644 elm-api-extend/Button.elm create mode 100644 elm-api-extend/Counter.elm create mode 100644 elm-api-extend/Main.elm create mode 100644 elm-api-extend/README.md create mode 100644 elm-api-extend/RandomGif.elm create mode 100644 elm-api-extend/RandomGifPair.elm create mode 100644 elm-api-extend/RandomPairOfPair.elm create mode 100644 elm-api-extend/elm-package.json diff --git a/elm-api-extend/Button.elm b/elm-api-extend/Button.elm new file mode 100644 index 0000000..97f31bf --- /dev/null +++ b/elm-api-extend/Button.elm @@ -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 \ No newline at end of file diff --git a/elm-api-extend/Counter.elm b/elm-api-extend/Counter.elm new file mode 100644 index 0000000..afe748e --- /dev/null +++ b/elm-api-extend/Counter.elm @@ -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] diff --git a/elm-api-extend/Main.elm b/elm-api-extend/Main.elm new file mode 100644 index 0000000..927a75f --- /dev/null +++ b/elm-api-extend/Main.elm @@ -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 \ No newline at end of file diff --git a/elm-api-extend/README.md b/elm-api-extend/README.md new file mode 100644 index 0000000..4547e35 --- /dev/null +++ b/elm-api-extend/README.md @@ -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. diff --git a/elm-api-extend/RandomGif.elm b/elm-api-extend/RandomGif.elm new file mode 100644 index 0000000..531828c --- /dev/null +++ b/elm-api-extend/RandomGif.elm @@ -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 diff --git a/elm-api-extend/RandomGifPair.elm b/elm-api-extend/RandomGifPair.elm new file mode 100644 index 0000000..e6c658b --- /dev/null +++ b/elm-api-extend/RandomGifPair.elm @@ -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 + ] diff --git a/elm-api-extend/RandomPairOfPair.elm b/elm-api-extend/RandomPairOfPair.elm new file mode 100644 index 0000000..ff646c4 --- /dev/null +++ b/elm-api-extend/RandomPairOfPair.elm @@ -0,0 +1,71 @@ +module RandomPairOfPair where + +import Effects exposing (Effects) +import Html exposing (..) +import Html.Attributes exposing (..) + +import RandomGifPair + + +-- MODEL + +type alias Model = + { left : RandomGifPair.Model + , right : RandomGifPair.Model + } + + +init : String -> String -> String -> String -> (Model, Effects Action) +init leftTopic1 leftTopic2 rightTopic1 rightTopic2 = + let + (left, leftFx) = RandomGifPair.init leftTopic1 leftTopic2 + (right, rightFx) = RandomGifPair.init rightTopic1 rightTopic2 + in + ( Model left right + , Effects.batch + [ Effects.map Left leftFx + , Effects.map Right rightFx + ] + ) + + +-- UPDATE + +type Action + = Left RandomGifPair.Action + | Right RandomGifPair.Action + +isNewGif : Action -> Bool +isNewGif action = + case action of + Left act -> RandomGifPair.isNewGif act + Right act -> RandomGifPair.isNewGif act + +update : Action -> Model -> (Model, Effects Action) +update action model = + case action of + Left act -> + let + (left, fx) = RandomGifPair.update act model.left + in + ( Model left model.right + , Effects.map Left fx + ) + + Right act -> + let + (right, fx) = RandomGifPair.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") ] ] + [ RandomGifPair.view (Signal.forwardTo address Left) model.left + , RandomGifPair.view (Signal.forwardTo address Right) model.right + ] diff --git a/elm-api-extend/elm-package.json b/elm-api-extend/elm-package.json new file mode 100644 index 0000000..e7f3ef1 --- /dev/null +++ b/elm-api-extend/elm-package.json @@ -0,0 +1,18 @@ +{ + "version": "1.0.0", + "summary": "helpful summary of your project, less than 80 characters", + "repository": "https://github.com/user/project.git", + "license": "BSD3", + "source-directories": [ + "." + ], + "exposed-modules": [], + "dependencies": { + "elm-lang/core": "3.0.0 <= v < 4.0.0", + "evancz/elm-effects": "2.0.1 <= v < 3.0.0", + "evancz/elm-html": "4.0.2 <= v < 5.0.0", + "evancz/elm-http": "3.0.0 <= v < 4.0.0", + "evancz/start-app": "2.0.2 <= v < 3.0.0" + }, + "elm-version": "0.16.0 <= v < 0.17.0" +} \ No newline at end of file