Elm - 命令
在前面的章節中,我們討論了 Elm 架構的各個元件及其功能。使用者和應用程式使用訊息相互通訊。
考慮一個示例,其中應用程式需要與其他元件(例如外部伺服器、API、微服務等)通訊以服務於使用者請求。這可以透過在 Elm 中使用命令來實現。訊息和命令不是同義詞。訊息表示終端使用者和應用程式之間的通訊,而命令表示 Elm 應用程式如何與其他實體通訊。命令是響應訊息觸發的。
下圖顯示了一個複雜的 Elm 應用程式的工作流程:
使用者與檢視互動。檢視根據使用者的操作生成相應的訊息。更新元件接收此訊息並觸發一個命令。
語法
定義命令的語法如下所示:
type Cmd msg
檢視生成的訊息傳遞給命令。
示例
以下示例向 API 發出請求並顯示 API 返回的結果。
應用程式接受使用者輸入的數字,將其傳遞給 Numbers API。此 API 返回與該數字相關的知識。
應用程式的各個元件如下:
Http 模組
Elm 的 Http 模組用於建立和傳送 HTTP 請求。此模組不是核心模組的一部分。我們將使用 Elm 包管理器來安裝此包。
API
在此示例中,應用程式將與 Numbers API 通訊 – "http://numbersapi.com/#42"。
檢視
應用程式的檢視包含一個文字框和一個按鈕。
view : Model -> Html Msg
view model =
div []
[ h2 [] [text model.heading]
,input [onInput Input, value model.input] []
, button [ onClick ShowFacts ] [ text "show facts" ]
, br [] []
, h3 [][text model.factText]
]
模型
模型表示使用者輸入的值以及 API 將返回的結果。
type alias Model =
{ heading : String
, factText : String
, input :String
}
訊息
應用程式具有以下三個訊息:
- ShowFacts
- Input
- NewFactArrived
單擊“顯示知識”按鈕時,ShowFacts 訊息將傳遞給更新方法。當用戶在文字框中鍵入某些值時,Input 訊息將傳遞給更新方法。最後,當接收到 Http 伺服器響應時,NewFactArrived 訊息將傳遞給更新。
type Msg = ShowFacts |Input String | NewFactArrived (Result Http.Error String)
更新
更新方法返回一個元組,其中包含模型和命令物件。當用戶單擊“顯示知識”按鈕時,訊息將傳遞給更新,然後更新呼叫 NumbersAPI。
update : Msg -> Model -> (Model, Cmd Msg)
update msg model =
case msg of
Input newInput ->
(Model "NumbersApi typing.." "" newInput ,Cmd.none)
ShowFacts ->
(model, getRadmonNumberFromAPI model.input)
NewFactArrived (Ok newFact) ->
(Model "DataArrived" newFact "", Cmd.none)
NewFactArrived (Err _) ->
(model, Cmd.none)
輔助函式
輔助函式 getRandomNumberFromAPI 呼叫 NumbersAPI 並將使用者輸入的數字傳遞給它。API 返回的結果用於更新模型。
getRadmonNumberFromAPI : String->Cmd Msg
getRadmonNumberFromAPI newNo =
let
url =
"http://numbersapi.com/"++newNo
in
Http.send NewFactArrived (Http.getString url)
| 序號 | 方法 | 簽名 | 描述 |
|---|---|---|---|
| 1 | Http.getString | getString : String -> Request String | 建立一個 GET 請求並將響應主體解釋為字串。 |
| 2 | Http.send | send:(Result Error a -> msg) -> Request a -> Cmd msg | 傳送一個 Http 請求。 |
main
這是 Elm 專案的入口點。
main =
Html.program
{ init = init
, view = view
, update = update
, subscriptions = subscriptions
}
綜合起來
步驟 1 - 建立資料夾 CommandApp 和檔案 CommandDemo.elm。
步驟 2 - 使用命令 elm package install elm-lang/http 安裝 http 模組。
步驟 2 - 鍵入 CommandDemo.elm 的內容,如下所示:
import Html exposing (..)
import Html.Attributes exposing (..)
import Html.Events exposing (..)
import Http
main =
Html.program
{ init = init
, view = view
, update = update
, subscriptions = subscriptions
}
-- MODEL
type alias Model =
{ heading : String
, factText : String
, input :String
}
init : (Model, Cmd Msg)
init =
( Model "NumbersAPI" "NoFacts" "42"-- set model two fields
, Cmd.none -- not to invoke api initially
)
-- UPDATE
type Msg
= ShowFacts
|Input String
| NewFactArrived (Result Http.Error String)
update : Msg -> Model -> (Model, Cmd Msg)
update msg model =
case msg of
Input newInput ->
(Model "NumbersApi typing.." "" newInput ,Cmd.none)
ShowFacts ->
(model, getRadmonNumberFromAPI model.input)
NewFactArrived (Ok newFact) ->
(Model "DataArrived" newFact "", Cmd.none)
NewFactArrived (Err _) ->
(model, Cmd.none)
- VIEW
view : Model -> Html Msg
view model =
div []
[ h2 [] [text model.heading]
,input [onInput Input, value model.input] []
, button [ onClick ShowFacts ] [ text "show facts" ]
, br [] []
, h3 [][text model.factText]
]
-- SUBSCRIPTIONS
subscriptions : Model -> Sub Msg
subscriptions model =
Sub.none
-- HTTP
getRadmonNumberFromAPI : String->Cmd Msg
getRadmonNumberFromAPI newNo =
let
url =
"http://numbersapi.com/"++newNo
in
Http.send NewFactArrived (Http.getString url)
步驟 4 - 執行命令。
C:\Users\dell\elm\CommandApp> elm make .\CommandDemo.elm
這將生成如下所示的 html 檔案。