
TL;DR
In the previous post, we created the Picshare Photo Decoder. Today we are making our first Elm HTTP Request. This post is part of the functional language series, and it is based on a remarkable book, Programming Elm Build Safe and Maintainable Front-End Applications by Jeremy Fairbank.
Elm Commands
We have already learned that Elm is quite different from other programming languages. The HTTP module is also part of that difference.
HTTP library is not a core part of Elm. We have to install it first. Follow the instructions from the screenshot above.
module Picshare exposing (main) | |
import Browser | |
import Html exposing (..) | |
import Html.Events exposing (onClick, onInput, onSubmit) | |
import Html.Attributes exposing (class, src, placeholder, type_, disabled, value) | |
import Json.Decode exposing (Decoder, bool, int, list, string, succeed) | |
import Json.Decode.Pipeline exposing (hardcoded, required) | |
-- 1. add http lib | |
import Http | |
type alias Id = | |
Int | |
type alias Photo = | |
{ id : Id | |
, url : String | |
, caption : String | |
, liked : Bool | |
, comments : List String | |
, newComment : String | |
} | |
type alias Model = | |
Photo | |
photoDecoder : Decoder Photo | |
photoDecoder = | |
succeed Photo | |
|> required "id" int | |
|> required "url" string | |
|> required "caption" string | |
|> required "liked" bool | |
|> required "comments" (list string) | |
|> hardcoded "" | |
main : Program () Model Msg | |
main = | |
Browser.sandbox | |
{ init = initialModel | |
, view = view | |
, update = update | |
} | |
view : Model -> Html Msg | |
view model = | |
div [] | |
[ | |
div [ class "header" ] | |
[ h1 [] [text "Picshare"] ], | |
div [ class "content-flow"] | |
[ viewDetailedPhoto model] | |
] | |
type Msg | |
= ToggleLike | |
| UpdateComment String | |
| SaveComment | |
viewLoveButton : Model -> Html Msg | |
viewLoveButton model = | |
let | |
buttonClass = | |
if model.liked then | |
"fa-heart" | |
else | |
"fa-heart-o" | |
in | |
div [ class "like-button" ] | |
[ i | |
[ class "fa fa-2x" | |
, class buttonClass | |
, onClick ToggleLike | |
] | |
[] | |
] | |
viewComment : String -> Html Msg | |
viewComment comment = | |
li [] | |
[ strong [] [ text "Comment:" ] | |
, text (" " ++ comment) | |
] | |
viewCommentList : List String -> Html Msg | |
viewCommentList comments = | |
case comments of | |
[] -> | |
text "" | |
_ -> | |
div [ class "comments" ] | |
[ ul [] | |
(List.map viewComment comments) | |
] | |
viewComments : Model -> Html Msg | |
viewComments model = | |
div [] | |
[ viewCommentList model.comments | |
, form [ class "new-comment", onSubmit SaveComment] | |
[ input | |
[ type_ "text" | |
, placeholder "Add a comment..." | |
, value model.newComment | |
, onInput UpdateComment | |
] | |
[] | |
, button | |
[ disabled (String.isEmpty model.newComment) ] | |
[ text "Save" ] | |
] | |
] | |
viewDetailedPhoto : Model -> Html Msg | |
viewDetailedPhoto model = | |
div [ class "detailed-photo" ] | |
[ | |
img [ src model.url ] [] | |
, div [ class "photo-info"] | |
[ viewLoveButton model | |
, h2 [ class "caption" ] [ text model.caption] | |
, viewComments model | |
] | |
] | |
update : | |
Msg | |
-> Model | |
-> Model | |
update msg model = | |
case msg of | |
ToggleLike -> | |
{ model | liked = not model.liked } | |
UpdateComment comment -> | |
{model | newComment = comment } | |
SaveComment -> | |
saveNewComment model | |
saveNewComment : Model -> Model | |
saveNewComment model = | |
let | |
comment = | |
String.trim model.newComment | |
in | |
case comment of | |
"" -> model | |
_ -> | |
{ model | |
| comments = model.comments ++ [ comment ] | |
, newComment = "" | |
} | |
baseUrl : String | |
baseUrl = "https://www.hps.hr/files/data/" | |
initialModel : Model | |
initialModel = | |
{ id = 1 | |
, url = baseUrl ++ "27/kuce.folder/planinarska-kuca-picelj.jpg" | |
, caption = "Picelj Park Near Zabok" | |
, liked = False | |
, comments = [ "Really nice place!" ] | |
, newComment = "" | |
} | |
-- 2. fetchFeed constant implementation | |
fetchFeed : Cmd Msg | |
fetchFeed = | |
Http.get | |
{ url = baseUrl ++ "27/kuce.folder/planinarska-kuca-picelj.jpg" | |
, expect = Http.expectJson LoadFeed photoDecoder | |
} |
We imported the HTTP module but did not expose any package. This is due to the possible HTTP name domain could easily clash with other packages.
Then we created fetchFeed constant based on HTTP.get
Elm Cmd. The reason is that each function in Elm is pure. Pure means that function does not have side effects that could cause different results from run to run. HTTP.get is a good example because it could fail if the URL is not available at the moment (Google was not available for some time two days ago ).
fetchedFeed
is Elm Cmd that needs to be passed to Elm runtime to be executed.
Remember
- what is Elm pure function
- investigate Elm HTTP module documentation
Comments are closed.