Basic working example
This commit is contained in:
commit
5b1009f4eb
5 changed files with 249 additions and 0 deletions
7
.gitignore
vendored
Normal file
7
.gitignore
vendored
Normal file
|
@ -0,0 +1,7 @@
|
|||
# elm-package generated files
|
||||
elm-stuff
|
||||
# elm-repl generated files
|
||||
repl-temp-*
|
||||
|
||||
leaderboard.js
|
||||
leaderboard.min.js
|
33
elm.json
Normal file
33
elm.json
Normal file
|
@ -0,0 +1,33 @@
|
|||
{
|
||||
"type": "application",
|
||||
"source-directories": [
|
||||
"src"
|
||||
],
|
||||
"elm-version": "0.19.0",
|
||||
"dependencies": {
|
||||
"direct": {
|
||||
"NoRedInk/elm-json-decode-pipeline": "1.0.0",
|
||||
"elm/browser": "1.0.1",
|
||||
"elm/core": "1.0.2",
|
||||
"elm/html": "1.0.0",
|
||||
"elm/http": "2.0.0",
|
||||
"elm/json": "1.1.3",
|
||||
"elm/time": "1.0.0",
|
||||
"justinmimbs/date": "3.2.0",
|
||||
"rundis/elm-bootstrap": "5.2.0"
|
||||
},
|
||||
"indirect": {
|
||||
"avh4/elm-color": "1.0.0",
|
||||
"elm/bytes": "1.0.8",
|
||||
"elm/file": "1.0.5",
|
||||
"elm/parser": "1.1.0",
|
||||
"elm/regex": "1.0.0",
|
||||
"elm/url": "1.0.0",
|
||||
"elm/virtual-dom": "1.0.2"
|
||||
}
|
||||
},
|
||||
"test-dependencies": {
|
||||
"direct": {},
|
||||
"indirect": {}
|
||||
}
|
||||
}
|
18
index.html
Normal file
18
index.html
Normal file
|
@ -0,0 +1,18 @@
|
|||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<script src="leaderboard.js"></script>
|
||||
|
||||
<!-- Bootstrap CSS -->
|
||||
<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/css/bootstrap.min.css" integrity="sha384-ggOyR0iXCbMQv3Xipma34MD+dH/1fQ784/j6cY/iJTQUOhcWr7x9JvoRxT2MZw1T" crossorigin="anonymous">
|
||||
</head>
|
||||
<body>
|
||||
<div id="my-leaderboard-app"></div>
|
||||
<script>
|
||||
var app = Elm.Leaderboard.init({
|
||||
node: document.getElementById('my-leaderboard-app')
|
||||
})
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
19
justfile
Normal file
19
justfile
Normal file
|
@ -0,0 +1,19 @@
|
|||
watch: build
|
||||
elm-live src/Leaderboard.elm --open -- --output=leaderboard.js
|
||||
|
||||
build:
|
||||
elm make src/Leaderboard.elm --output=leaderboard.js
|
||||
|
||||
install:
|
||||
yarn global add elm-format elm-test elm-oracle elm-live
|
||||
nodenv rehash
|
||||
|
||||
debug:
|
||||
elm-live src/Leaderboard.elm --open -- --debug --output=leaderboard.js
|
||||
|
||||
build_release:
|
||||
elm make src/Leaderboard.elm --optimize --output=leaderboard.js
|
||||
uglifyjs leaderboard.js --compress "pure_funcs=[F2,F3,F4,F5,F6,F7,F8,F9,A2,A3,A4,A5,A6,A7,A8,A9],pure_getters,keep_fargs=false,unsafe_comps,unsafe" | uglifyjs --mangle --output=leaderboard.min.js
|
||||
|
||||
deploy: build_release
|
||||
rsync index.html leaderboard.min.js anthonycicchetti.com:/var/www/crossword.anthonycicchetti.com/public
|
172
src/Leaderboard.elm
Normal file
172
src/Leaderboard.elm
Normal file
|
@ -0,0 +1,172 @@
|
|||
module Leaderboard exposing (main)
|
||||
|
||||
import Browser
|
||||
import Date
|
||||
import Debug exposing (toString)
|
||||
import Html exposing (..)
|
||||
import Html.Attributes exposing (..)
|
||||
import Html.Events exposing (onClick, onInput)
|
||||
import Http
|
||||
import Json.Decode as Decode exposing (Decoder)
|
||||
import Json.Decode.Pipeline exposing (hardcoded, optional, required)
|
||||
import Task exposing (Task)
|
||||
|
||||
|
||||
|
||||
--- CONSTANTS
|
||||
|
||||
|
||||
apiUrlPrefix : String
|
||||
apiUrlPrefix =
|
||||
"https://acicchetti.dev/crossword/v1"
|
||||
|
||||
|
||||
|
||||
--- DECODERS
|
||||
|
||||
|
||||
entryDecoder : Decoder Entry
|
||||
entryDecoder =
|
||||
Decode.succeed Entry
|
||||
|> required "name" Decode.string
|
||||
|> required "time" Decode.string
|
||||
|
||||
|
||||
scoreDecoder : String -> Decoder Score
|
||||
scoreDecoder score =
|
||||
Decode.string
|
||||
|
||||
|
||||
|
||||
--- MODEL
|
||||
|
||||
|
||||
type alias Model =
|
||||
{ date : String
|
||||
, entries : List Entry
|
||||
, alertMessage : Maybe String
|
||||
, newDate : String
|
||||
}
|
||||
|
||||
|
||||
type alias Entry =
|
||||
{ name : String
|
||||
, score : Score
|
||||
}
|
||||
|
||||
|
||||
type alias Score =
|
||||
String
|
||||
|
||||
|
||||
|
||||
--- UPDATE
|
||||
|
||||
|
||||
type Msg
|
||||
= NewDate String
|
||||
| NewEntries (Result Http.Error (List Entry))
|
||||
|
||||
|
||||
update : Msg -> Model -> ( Model, Cmd Msg )
|
||||
update msg model =
|
||||
case msg of
|
||||
NewDate newDate ->
|
||||
( { model
|
||||
| date = newDate
|
||||
, entries = []
|
||||
, newDate = ""
|
||||
}
|
||||
, getScores newDate
|
||||
)
|
||||
|
||||
NewEntries (Ok newEntries) ->
|
||||
( { model | entries = List.sortBy .score newEntries }, Cmd.none )
|
||||
|
||||
NewEntries (Err err) ->
|
||||
-- ( { model | alertMessage = Just (toString err) }, Cmd.none)
|
||||
( model, Cmd.none )
|
||||
|
||||
|
||||
|
||||
--- COMMANDS
|
||||
|
||||
|
||||
getScores : String -> Cmd Msg
|
||||
getScores newDate =
|
||||
Http.request
|
||||
{ method = "GET"
|
||||
, headers = [ Http.header "Access-Control-Allow-Origin" "https://acicchetti.dev" ]
|
||||
, url = apiUrlPrefix ++ "/" ++ newDate
|
||||
, body = Http.emptyBody
|
||||
, expect = Http.expectJson NewEntries (Decode.list entryDecoder)
|
||||
, timeout = Nothing
|
||||
, tracker = Nothing
|
||||
}
|
||||
|
||||
|
||||
|
||||
--- VIEW
|
||||
|
||||
|
||||
viewDate : String -> Html Msg
|
||||
viewDate date =
|
||||
h2 []
|
||||
[ text date ]
|
||||
|
||||
|
||||
viewEntryList : List Entry -> Html Msg
|
||||
viewEntryList entries =
|
||||
ul []
|
||||
(List.map
|
||||
(\entry ->
|
||||
li []
|
||||
[ div [] [ text entry.name ]
|
||||
, div [] [ text entry.score ]
|
||||
]
|
||||
)
|
||||
entries
|
||||
)
|
||||
|
||||
|
||||
view : Model -> Html Msg
|
||||
view model =
|
||||
div []
|
||||
[ div []
|
||||
[ input
|
||||
[ type_ "date"
|
||||
, placeholder model.date
|
||||
, onInput NewDate
|
||||
, value model.date
|
||||
]
|
||||
[]
|
||||
]
|
||||
, viewEntryList model.entries
|
||||
, text (Maybe.withDefault "" model.alertMessage)
|
||||
]
|
||||
|
||||
|
||||
|
||||
--- SUBSCRIPTIONS
|
||||
|
||||
|
||||
subscriptions : Model -> Sub Msg
|
||||
subscriptions model =
|
||||
Sub.none
|
||||
|
||||
|
||||
init : () -> ( Model, Cmd Msg )
|
||||
init _ =
|
||||
( Model "2019-09-29" [] Nothing ""
|
||||
, getScores "2019-09-29"
|
||||
)
|
||||
|
||||
|
||||
main : Program () Model Msg
|
||||
main =
|
||||
Browser.element
|
||||
{ init = init
|
||||
, view = view
|
||||
, update = update
|
||||
, subscriptions = subscriptions
|
||||
}
|
Loading…
Add table
Reference in a new issue