この記事は https://github.com/evancz/elm-architectu... から翻訳されたものです...
このチュートリアルでは、「アーキテクチャの概要」について説明します。 Elm プログラム」では、TodoMVC や dreamwriter から実稼働環境で実行される NoRedInk や CircuitHub コードに至るまで、すべての Elm プログラムでこのプログラムが表示されます。この基本パターンは、Elm または JS フロントエンド コードを作成する場合に役立ちます。
Elm アーキテクチャは、モジュール性、コードの再利用、テストに最適な、無限にネストされたコンポーネントのシンプルなパターンです。さらに、このパターンにより、複雑な Web アプリケーションをモジュール形式で簡単に作成できます。 8 つの例を通して、その中心となる原則とパターンを段階的に学習します:
カウンター
ダブルカウンター
カウンタリスト
カウンタリスト (バリアント)
GIF 抽出
デュアルGIF Extractor
GIF Extractor Queue
2 つのアニメーション
Ben いくつかの側面では、チュートリアル前例のない前例のないことと言えます!例 7 と例 8 の開発を非常に簡単にするために必要な概念とアイデアを説明します。基本的な投資には間違いなく価値があります。
これらの例のアーキテクチャには非常に興味深い点があります。それは Elm から自然に現れています。このドキュメントを読んでその利点を知っているかどうかに関係なく、Elm 言語自体の設計によってこのアーキテクチャにたどり着きます。 Elm を使用しているときにこのパターンに偶然出会い、そのシンプルさと強力さに驚かされました。
注: このチュートリアルを使用するには、コードと一緒に学習する必要があります。 Elm と Fork プロジェクトをインストールします。プロジェクトのコードを実行する方法については、このチュートリアルの各例で説明されています。
各 Elm プログラムのロジックは、完全に分離された 3 つの部分に分割されます:
モデル
更新
表示
以下の足場を自信を持って使用でき、特定のニーズに合わせて追加し続けることができます。詳細。
Elm コードを初めて読む場合は、構文から「関数的思考」まですべてをカバーしている公式の elm 言語ドキュメントを確認してください。あるいは、この完全ガイドの最初の 2 章を読めば、すぐに始めることができます。
-- MODELtype alias Model = { ... }-- UPDATEtype Action = Reset | ...update : Action -> Model -> Modelupdate action model = case action of Reset -> ... ...-- VIEWview : Model -> Htmlview = ...
このチュートリアルはすべて、このパターンのバリエーションと拡張に関するものです。
最初の例は、増加または減少できる単純なカウンターです。
このコードは、非常に単純なモデルから始まります。追跡する必要があるのは 1 つの数値だけです:
type alias Model = Int
モデルを更新するときは、物事は再び簡単になります。実行できる一連のアクションと、これらのアクションを実際に実行するための更新関数を定義します。
type Action = Increment | Decrementupdate : Action -> Model -> Modelupdate action model = case action of Increment -> model + 1 Decrement -> model - 1
Action 共用体タイプは何も実行しないことに注意してください。可能なアクションを簡単に説明します。特定のボタンが押されたときにカウンターが 2 倍になるはずだと誰かが考えた場合は、新しいアクションを追加するだけです。これは、このコードがモデルをどのように変更すべきかについて非常に明確であることを意味します。このコードを読めば、何が許可され、何が禁止されているかがすぐにわかります。さらに、一貫した方法で新しい機能を追加する方法を知ることができます。
最後に、モデルを表示するビューを作成します。 [elm-html][] を使用して、ブラウザに表示される HTML を作成します。まず、最も外側の div を作成します。これには、デクリメント ボタン、現在のカウントを表示する div、およびインクリメント ボタンが含まれます。
view : Signal.Address Action -> Model -> Htmlview address model = div [] [ button [ onClick address Decrement ] [ text "-" ] , div [ countStyle ] [ text (toString model) ] , button [ onClick address Increment ] [ text "+" ] ]countStyle : AttributecountStyle = ...
さらに厄介なのは、ビューの Address 関数です。これについては次の章で詳しく説明します。ここで、このコードは完全に単なるステートメントであることに注意してください。 Model を使用して HTML を生成します。それだけです。DOM を手動で変更することは一切ありません。これにより、一部のライブラリがより自由に賢く最適化を行い、レンダリングを高速化できるようになります。これはクレイジーです!また、ビューは通常の関数なので、ビューを作成すると、Elm のモジュール システム、テスト フレームワーク、ライブラリを取得できます。
このパターンは、Elm プログラムの構造の本質です。これから説明するすべての例は、この基本パターン (モデル、更新、ビュー) をわずかに変更したものになります。
ほぼすべての Elm プログラムには、アプリケーション全体を駆動する短いコードがあります。このチュートリアルの各例では、このコードの名前は Main.elm です。これは反例ですが、それでも興味深いです:
import Counter exposing (update, view)import StartApp.Simple exposing (start)main = start { model = 0, update = update, view = view }
StartApp ライブラリを使用して、初期モデルの更新とビューを接続します。 Elm の信号を簡単にカプセル化するだけなので、その原理をまだ詳しく調べる必要はありません。
装配应用的关键概念是 Address。每个事件处理器在 view函数中得到一个特定的地址,并且和数据块一起传递过来。 StartApp监听所有传给这个地址的消息,然后把它们发送给 update函数。 model获得更新, 而 [elm-html][] 负责渲染和高效的修改。
这意味着,Elm 程序中的数据只会在一个方向上流动,类似这样:
蓝色部分是我们 Elm 程序的核心,这正是 model/update/view,我们一直在讨论的模式。使用 Elm 编程,你可以一直呆在这个舒服的盒子里面,并取得很大的进步。
注意,我们 不执行送回应用程序的 action。我们只是转发一些数据。这种分离是一个关键的细节,使我们的逻辑完全从视图代码中分离出来。
在上一个例子里我们搞了一个计数器,如果增加到两个计数器时这个模式会怎样变化呢?我们能继续保持模块化吗?
如果我们能完全重用 例子1 的代码就再好不过了。Elm 架构最疯狂的就是: 我们可以一句不变地重用代码。当我们实现 例子1 的 Counter模块时,它包括了所有细节,所以我们可以在任何地方使用它。
module Counter (Model, init, Action, update, view) wheretype Modelinit : Int -> Modeltype Actionupdate : Action -> Model -> Modelview : Signal.Address Action -> Model -> Html
编写模块代码其实完全是在创建一种很强的抽象。我们期待的是提供合适的函数接口,但是隐藏具体执行过程。从 Counter模块的外部我们只能看到一些基础的值: Model、 init、 Action、 update和 view。我们完全不用关心这些是如何实现的。事实上,也不可能知道这些是如何实现的。这意味着没人需要依赖这些不公开的实现细节。
我们本可以完全复制 Counter模块, 但我们还是使用它的一部分来实现 CounterPair。 像往常一样, 我们从一个 Model开始:
type alias Model = { topCounter : Counter.Model , bottomCounter : Counter.Model }init : Int -> Int -> Modelinit top bottom = { topCounter = Counter.init top , bottomCounter = Counter.init bottom }
我们的 Model纪录了两个计数器, 其中一个是需要在屏幕上显示的。这个 Model完全描述了应用所有的状态。我们还有一个 init函数可以在任何地方创建一个新的 Model。
下一步来描述下我们想要支持的 Actions。我们需要的功能是:重置所有的计数器,更新顶部的计数器,或者更新下面的计数器。
type Action = Reset | Top Counter.Action | Bottom Counter.Action
请注意,我们的 [union type][] 是参考 Counter.Action类型,但我们并不知道那些 action的细节。当我们创建 update函数时,主要工作是路由这些 Counter.Actions到正确的地方:
update : Action -> Model -> Modelupdate action model = case action of Reset -> init 0 0 Top act -> { model | topCounter = Counter.update act model.topCounter } Bottom act -> { model | bottomCounter = Counter.update act model.bottomCounter }
所以最后要做的事情就是创建一个 view函数显示两个计数器和两个重置按钮。
view : Signal.Address Action -> Model -> Htmlview address model = div [] [ Counter.view (Signal.forwardTo address Top) model.topCounter , Counter.view (Signal.forwardTo address Bottom) model.bottomCounter , button [ onClick address Reset ] [ text "RESET" ] ]
请注意,我们可以在两个计数器之中复用 Counter.view函数,给每个计数器创建一个转发地址。大体上,这里做的事情其实是:“让这俩计数器给所有向外传递的消息打上 Top或 Bottom标志,以便区分”
这就是所有的工作。最屌的是我们可以一层又一层地保持嵌套。我们可以创建 CounterPair模块,暴露关键值和方法,然后创建 CounterPairPair或者任何其他我们需要的。
两个计数器已经很屌了,一个可以随意添加和删除的计数器队列会怎么样呢?这种模式还有效吗?
甚至我们可以完全像 例子1 和 例子2 里那样复用 Counter!
module Counter (Model, init, Action, update, view)
这意味着我们可以开始创建 CounterList模块了。 像往常一样, 我们从 Model开始:
type alias Model = { counters : List ( ID, Counter.Model ) , nextID : ID }type alias ID = Int
现在,我们的 model有了一个计数器队列,每个计数器有一个唯一的 ID。这些 ID 使我们可以区别它们,所以如果我们要更新 4 号计数器,我们可以很轻松的找到它。(当我们考虑优化渲染时,这个 ID 也给了我们一些 key 的便利,然而它并不是这个教程的重点!)我们的 modal还包含一个 nextID帮助我们指定 ID 给每一个新增的计数器。
现在我们可以定义一组 Action来操作 model。我们希望可以添加计数器,删除计数器,以及更新特定的计数器。
type Action = Insert | Remove | Modify ID Counter.Action
我们的 Action[union type][] 令人震惊的接近高阶描述。下面我们可以定义 update函数了。
update : Action -> Model -> Modelupdate action model = case action of Insert -> let newCounter = ( model.nextID, Counter.init 0 ) newCounters = model.counters ++ [ newCounter ] in { model | counters = newCounters, nextID = model.nextID + 1 } Remove -> { model | counters = List.drop 1 model.counters } Modify id counterAction -> let updateCounter (counterID, counterModel) = if counterID == id then (counterID, Counter.update counterAction counterModel) else (counterID, counterModel) in { model | counters = List.map updateCounter model.counters }
这里有对每种情况的高阶描述:
Insert— 首先我们创造一个新的计数器,并把它当在计数器队列的最后。然后我们给 nextID加一,以便下一次添加时有一个新的ID。
Remove— 删除计数器列表的第一个成员。
Modify— 遍历所有计数器,当找到匹配的 ID 时,用所给的 Action操作这个计数器。
下面唯一要做的就是定义 view。
view : Signal.Address Action -> Model -> Htmlview address model = let counters = List.map (viewCounter address) model.counters remove = button [ onClick address Remove ] [ text "Remove" ] insert = button [ onClick address Insert ] [ text "Add" ] in div [] ([remove, insert] ++ counters)viewCounter : Signal.Address Action -> (ID, Counter.Model) -> HtmlviewCounter address (id, model) = Counter.view (Signal.forwardTo address (Modify id)) model
这里的 viewCounter函数比较有趣。它必须使用同一个 Counter.view函数,但在这里我们提供了一个转发地址来标记所有的消息和正在渲染的计数器的 ID。
实际上,当我们创建 view函数时,我们映射 viewCounter到所有的计数器,然后创建添加和删除按钮直接返回 address。
这个 ID 的玩法可以用在任何你需要数目可变的子模块时。计数器是简单的,但是这种模式可以完全不变的在用户信息,tweets,新闻列表或者产品列表上复用。
OK,在一个动态的计数器列表上保持简单和模块化是很屌的,但是如果不要一个通用的删除按钮,而是每个计数器有一个单独的删除按钮呢?它会把事情搞糟吗?
不, 它仍然有效.
在这里,我们的目标是找到一种新的方法给每个计数器添加一个删除按钮。有趣的是,我们可以继续使用原有的 view函数并添加一个新的 viewWithRemoveButton函数,这个函数为我们依赖的 Model提供一个微小的变化。屌屌屌,我们不用重复任何代码更不用做任何疯狂的继承和重载。我们只是给公开的 API 添加了一个函数暴露新的功能!
module Counter (Model, init, Action, update, view, viewWithRemoveButton, Context) where...type alias Context = { actions : Signal.Address Action , remove : Signal.Address () }viewWithRemoveButton : Context -> Model -> HtmlviewWithRemoveButton context model = div [] [ button [ onClick context.actions Decrement ] [ text "-" ] , div [ countStyle ] [ text (toString model) ] , button [ onClick context.actions Increment ] [ text "+" ] , div [ countStyle ] [] , button [ onClick context.remove () ] [ text "X" ] ]
viewWithRemoveButton函数添加了一个额外的按钮。请注意 增加/减少按钮发送消息给 actions地址,但是删除按钮发送消息给 remove这个地址。我们发给 remove的消息其实是在说:“嘿,无论谁拥有我,请删掉我!” 这个计数器的拥有者负责删除。
既然我们有了新的 viewWithRemoveButton, 我们可以创建一个新的 CounterList模块把所有独立的计数器放在一起。这个 Model和 栗子3 中的一样: 带各自 ID 的计数器列表。
type alias Model = { counters : List ( ID, Counter.Model ) , nextID : ID }type alias ID = Int
我们的 action稍有不同。不是删除一个旧的计数器,而是删除特定的一个,所以 Remove需要一个 ID。
type Action = Insert | Remove ID | Modify ID Counter.Action
update函数和 栗子3 中的非常像。
update : Action -> Model -> Modelupdate action model = case action of Insert -> { model | counters = ( model.nextID, Counter.init 0 ) :: model.counters, nextID = model.nextID + 1 } Remove id -> { model | counters = List.filter (\(counterID, _) -> counterID /= id) model.counters } Modify id counterAction -> let updateCounter (counterID, counterModel) = if counterID == id then (counterID, Counter.update counterAction counterModel) else (counterID, counterModel) in { model | counters = List.map updateCounter model.counters }
在 Remove时,我们取出拥有该 ID 的计数器。否则,退出直接退出并保持原来那样。
最后,我们我们把萌宝宝们都放进 view中:
view : Signal.Address Action -> Model -> Htmlview address model = let insert = button [ onClick address Insert ] [ text "Add" ] in div [] (insert :: List.map (viewCounter address) model.counters)viewCounter : Signal.Address Action -> (ID, Counter.Model) -> HtmlviewCounter address (id, model) = let context = Counter.Context (Signal.forwardTo address (Modify id)) (Signal.forwardTo address (always (Remove id))) in Counter.viewWithRemoveButton context model
在 viewCounter函数中, 我们构造了 Counter.Context来传递所有必需的转发地址。在两种情况下分别声明 Counter.Action以便我们知道哪个计数器需要修改或删除。
基础模式— 任何事都是围绕 Model创建出来的,包括更新 model的函数, 以及 model的 view。任何事都可以看作基础模式的变体。
嵌套 Modules— 转发地址使基础模式的嵌套变的简单,完全隐藏实现细节。我们可以无限深地嵌套这种模式,并且每一层只需要知道下一层在发生什么。
添加上下文— 有时对 modal进行 update或者 view操作时需要额外的信息。我们随时可以添加 Context给这些函数并传递所有的附加信息而不需要改变 Model。
update : Context -> Action -> Model -> Modelview : Context' -> Model -> Html
在嵌套的每一层,我们都可以为每个子模块衍生出所需的 Context。
测试变的简单— 我们创建的所有函数都是 纯洁函数。这样测试 update函数变的极其简单。不需要特别的初始化、模拟、配置步骤,你只要带着你想要测试的参数直接调用函数即可。
我们已经讲了如何创建可无限嵌套的组件,但当我们在某个组件里发出一个 HTTP 请求时会发生什么呢?与数据库通信呢?这个栗子使用 elm-effects包 来创建一个简单的组件,这个组件可以从 giphy.com 获取随机的可爱喵星人的 gif。
如果看了 这个栗子的实现, 你会注意到它和 栗子1 中的代码非常接近。它的 Model非常典型:
type alias Model = { topic : String , gifUrl : String }
我们需要知道要查找的 topic值和当前展示的 gifUrl。这里唯一新颖的东西是 init和 update的类型:
init : String -> (Model, Effects Action)update : Action -> Model -> (Model, Effects Action)
并非只是返回一个新的 Model我们还返回一些我们需要执行的效果。所以我们将会使用 EffectsAPI ,看起来像这样:
module Effects wheretype Effects anone : Effects a -- don't do anythingtask : Task Never a -> Effects a -- request a task, do HTTP and database stuff
Effects类型本质上是一个包含了一些会在之后执行的独立任务的数据类型。让我们通过分析这里的 update来更深入了解下这是怎么工作的:
type Action = RequestMore | NewGif (Maybe String)update : Action -> Model -> (Model, Effects Action)update msg model = case msg of RequestMore -> ( model , getRandomGif model.topic ) NewGif maybeUrl -> ( Model model.topic (Maybe.withDefault model.gifUrl maybeUrl) , Effects.none )-- getRandomGif : String -> Effects Action
所以用户可以通过点击 “More Please!” 按钮来触发 RequestMore,当服务器响应请求后它会给我们一个 NewGif的 action。我们在 update函数中处理这两种情况。
在这里 RequestMore第一次返回已经存在的 model。用户只是点击了一个按钮,这时并没有任何改变。我们还使用 getRandomGif函数创建了一个 Effects Action。我们马上将会知道 getRandomGif是如何定义的。到此为止,我们只需知道当一个 Effects Action运行时,会有一系列 Action值产生并被传递给整个应用。所以 getRandomGif model.topic最终会产生像这样的一个 action like:
NewGif (Just "http://s3.amazonaws.com/giphygifs/media/ka1aeBvFCSLD2/giphy.gif")
它返回一个 Maybe因为向服务器发出的请求可能失败。那个 Action将会原路返回给 update函数。所以当我们执行 NewGif时,我们只是更新当前的 gifUrl,如果他可以被更新。当请求失败后,我们只是停留在当前的 model.gifUrl。
我们看到同样的事情发生在 init函数中,它定义了初始时的 modal并且通过 giphy.com 的 API 请求一个特定话题的 GIF。
init : String -> (Model, Effects Action)init topic = ( Model topic "assets/waiting.gif" , getRandomGif topic )-- getRandomGif : String -> Effects Action
再一次,当随机的 GIF 下载完成,它会产生一个 Action发送给 update函数。
注意:之前我们使用的是来自 the start-app package的 StartApp.Simple模块,但是现在请升级到 StartApp模块。它可以处理更实际的 web 应用中的复杂情况。它有 更优雅的 API。更至关重要的改变是它可以处理我们新的 init和 update类型。
这个例子中一个至关重要的方面是 getRandomGif函数,它描述了如何得到一张随机的 GIF。它使用了 任务[http] 库, 我会尽力概述它是如何运做的。让我们看定义:
getRandomGif : String -> Effects ActiongetRandomGif topic = Http.get decodeImageUrl (randomUrl topic) |> Task.toMaybe |> Task.map NewGif |> Effects.task-- The first line there created an HTTP GET request. It tries to-- get some JSON at `randomUrl topic` and decodes the result-- with `decodeImageUrl`. Both are defined below!---- Next we use `Task.toMaybe` to capture any potential failures and-- apply the `NewGif` tag to turn the result into a `Action`.-- Finally we turn it into an `Effects` value that can be used in our-- `init` or `update` functions.-- Given a topic, construct a URL for the giphy API.randomUrl : String -> StringrandomUrl topic = Http.url "http://api.giphy.com/v1/gifs/random" [ "api_key" => "dc6zaTOxFJmzC" , "tag" => topic ]-- A JSON decoder that takes a big chunk of JSON spit out by-- giphy and extracts the string at `json.data.image_url` decodeImageUrl : Json.Decoder StringdecodeImageUrl = Json.at ["data", "image_url"] Json.string
一旦我们写了上面这些,我们就可以在 init和 update函数中复用 getRandomGif。
有趣的是, getRandomGif返回的任务是 永远不会失败的。原因是任何可能的失败 必须被明确的处理,我们不希望任何任务静静地失败。
我试图确切地解释下它是如何实现的,虽然这对于整个项目的正常运行并不特别重要。Okay,这样每个 Task有一个失败的类型和一个成功的类型。例如,一个 HTTP 任务可能有类型如: Task Http.Error String,我们可以在失败时返回一个 Http.Error或者成功时返回一个 String。这样可以优雅地把一组任务串在一起而不用过多的担心出错。现在,假设我们的组件请求了一个任务,但是任务失败了。会发生什么呢?谁会被通知?如何恢复?通过设置失败类型为 Never,我们强制任何可能的错误变成成功类型,这样它们就可以被组件明确的处理了。在这个例子里,我们用 Task.toMaybe : Task x a -> Task y (Maybe a)所以 update函数精确的处理了 HTTP 失败。这意味着任务不能静默的失败,你永远精确的处理着未知的错误。
好了,结果搞定了,但是 嵌套的结果呢?你是否思考过这个问题?!这个例子完全重用栗子5中的 GIF 查看器的代码创建了两个独立的 GIF 查看器。
你阅读 这个实现代码时,会注意到它和栗子2中的两个计数器的代码几乎一样。 Model被定义为两个 RandomGif.Model的值:
type alias Model = { left : RandomGif.Model , right : RandomGif.Model }
这让我们可以独立地分别跟踪它们。我们的 action只是路由消息到正确的自模块。
type Action = Left RandomGif.Action | Right RandomGif.Action
有趣的是,我们实际上使用了 Leftand Right标签在 update和 init函数中。
-- Effects.map : (a -> b) -> Effects a -> Effects bupdate : Action -> Model -> (Model, Effects Action)update action model = case action of Left msg -> let (left, fx) = RandomGif.update msg model.left in ( Model left model.right , Effects.map Left fx ) Right msg -> let (right, fx) = RandomGif.update msg model.right in ( Model model.left right , Effects.map Right fx )
所以不论在哪个分支中调用 RandomGif.update函数时都会返回一个新 model和一些被我们称作 fx的操作。我们像往常一样返回一个更新过的 model,但是需要在操作上做一些额外的工作。并非直接返回它们,我们使用 Effects.map 函数把他们转化为一种 Action。这工作很像 Signal.forwardTo,让我们标记这些值以便确定如何路由。
init函数也是一样。我们提供一个 topic给每个随机 GIF 查看器,然后得到一个初始的 model和一些 effects。
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 ] )-- Effects.batch : List (Effects a) -> Effects a
在这里我们并非只用 Effects.map来标记合适的结果,还要用 Effects.batch 函数来把他们归并到一起。所有请求的任务将会被生成并且独立运行,所以左边和右边两个 effects会同时被处理。
这个例子实现了一个随机 GIF 查看器的队列,你可以自己为他设置话题。而且,我们完全重用了 RandomGif模块的核心。
仔细看看 它的代码你会发现它和 例子3 几乎一致。我们把所有子模块放进一个关联了 ID 的列表,并依据这些 ID 来进行操作。唯一新鲜的是我们使用 Effects在 init和 update函数中,把他们和 Effects.map以及 Effects.batch放在一起。
如果你对它的实现细节还不够清楚,请创建一个 issue。
现在,我们已经看到了带任务的组件可以很轻松地嵌套在一起,但是用它如何实现动画呢?
很有趣,它们完全一样!(或许你已经不再感到惊奇了,相同的模式在这里也适用,真是一个可爱的模式!)
这个例子是两个可点击的方块。当你点击一个方块时,它旋转 90 度。总体上,这里的代码是对 例子2 和 例子6 的调整,我们保留了所有的动画逻辑在 SpinSquare.elm里面,并且在 SpinSquarePair.elm里多次复用它。
所有新的和有趣的东西都发生在里,所以我们来关注下这里的代码。首先我们需要一个 model:
type alias Model = { angle : Float , animationState : AnimationState }type alias AnimationState = Maybe { prevClockTime : Time, elapsedTime: Time }rotateStep = 90duration = second
所以 model 的核心是方块当前的 angle和一些用来记录每个动画要做什么的 animationState。如果没有动画就是 Nothing,但是如果有动作发生,它就变为:
prevClockTime— 用于计算时间差的最近时间。它帮我们精确地确定上一帧后过了多少毫秒。
elapsedTime— 0 到 duration之间的一个数字,告诉我们当前动画已经进行了多久。
常量 rotateStep只是声明每次点击转变多少度。你可以随意修改它,而不会影响正常运行。
现在, update里发生了一些有趣的事:
type Action = Spin | Tick Timeupdate : Action -> Model -> (Model, Effects Action)update msg model = case msg of Spin -> case model.animationState of Nothing -> ( model, Effects.tick Tick ) Just _ -> ( model, Effects.none ) Tick clockTime -> let newElapsedTime = case model.animationState of Nothing -> 0 Just {elapsedTime, prevClockTime} -> elapsedTime + (clockTime - prevClockTime) in if newElapsedTime > duration then ( { angle = model.angle + rotateStep , animationState = Nothing } , Effects.none ) else ( { angle = model.angle , animationState = Just { elapsedTime = newElapsedTime, prevClockTime = clockTime } } , Effects.tick Tick )
有两种 Action我们需要处理:
Spin标示一个用户点击了方块,请求一次旋转。所以在 update函数中,如果没有正在进行的动画,我们就请求一个时间戳,并把状态设置为一个动画正在进行。
Tick标示我们已经得到了一个时间戳,所以我们需要进行一次动画。在 update函数中,这意味着我们需要更新 animationState。所以,首先,我们检查当前是否有正在进行的动画。如果有,我们只是计算出 newElapsedTime的值,通过把当前的 elapsedTime加上一个时间差。如果当前经过的时间大于 duration,我们就停止动画并请求一个新的时间戳。否则,我们更新动画状态,也请求一个新的时间戳。
再一次,随着写了这么多类似的代码,审视一遍它们,我们会发现一个通用的模式。发现它时你一定很激动!
终于,无论如何我们有了一个有趣的 view函数!这个例子有了一个优雅又充满活力的动画,而我们只是在时间线上增加了 elapsedTime而已。这是怎么做到的呢?
view的代码本身就是一个标准的 elm-svg ,可以制作一些漂亮的可点击图形。 代码中 最牛X 的是 toOffset,它计算了当前 AnimationState的旋转的度数。
-- import Easing exposing (ease, easeOutBounce, float)toOffset : AnimationState -> FloattoOffset animationState = case animationState of Nothing -> 0 Just {elapsedTime} -> ease easeOutBounce float 0 rotateStep duration elapsedTime
我们使用 @Dandandan的 easing库,它使得对数字、颜色、点以及其他任何疯狂的东西的 补间排序变得很简单。
所以 ease函数从 0 到 duration之间取出一个数。然后它把它转变成一个 0 到 rotateStep(我们之前的代码里已经把它设置为 90 度了)之间的一个数。在这里你还提供了一个 补间给 easeOutBounce这意味着随着它从 0 到 duration变化,我们会得到一个从 0 到 90 变化的数字。太疯狂了!尝试替换 easeOutBounce为 另一个补间看看是什么效果!
从这儿开始,我们把所有东西都拼装到了一起成为 SpinSquarePair, 而它的代码几乎与 例子2 和 例子6 的一模一样。
好了,这就是用这些工具实现动画的基础!如果把所有东西都摆在这儿,可能不够清晰,所以当你有了更多的经验,请让我们知道你的收获。希望我们可以把她变得更简单!
注意:我期待我们可以在这些核心思想之上构建一些抽象概念。这个例子做了一些基础的事情,但是我打赌随着我们继续为它做出的工作,我们可以找到一些优雅的模式使它更简单。如果你觉得它现在还是很复杂,请试着让它变得更好,并把你的想法告诉我们吧!