diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml
index 4be68eb..08e2113 100644
--- a/.github/workflows/build.yml
+++ b/.github/workflows/build.yml
@@ -52,6 +52,12 @@ jobs:
- uses: dw-labs-org/dart-sass-gha@v1
+ - name: install-happy
+ run: |
+ cabal path --installdir >> "$GITHUB_PATH"
+ cabal update -z
+ cabal install -z happy
+
- name: build-frontend
run: |
cd frontend
@@ -85,6 +91,12 @@ jobs:
- uses: dw-labs-org/dart-sass-gha@v1
+ - name: install-happy
+ run: |
+ cabal path --installdir >> "$GITHUB_PATH"
+ cabal update -z
+ cabal install -z happy
+
- name: build-frontend
run: |
cd frontend
diff --git a/app/App.hs b/app/App.hs
index 0bac028..2a96c39 100644
--- a/app/App.hs
+++ b/app/App.hs
@@ -2,6 +2,7 @@ module App (start) where
import GHC.Wasm.Prim
import Language.Javascript.JSaddle (JSM)
+import Reflex.TodoMVC qualified
import SimpleCounter qualified
import Snake qualified
import TodoMVC qualified
@@ -16,4 +17,5 @@ start e =
"todomvc" -> TodoMVC.start
"xhr" -> XHR.start
"2048" -> TwoZeroFourEight.start
+ "reflex-todomvc" -> Reflex.TodoMVC.main
_ -> fail "unknown example"
diff --git a/cabal.project b/cabal.project
index 8e408db..8c1924a 100644
--- a/cabal.project
+++ b/cabal.project
@@ -1,4 +1,4 @@
-packages: . hs2048
+packages: . hs2048 reflex-todomvc
index-state: 2024-10-20T13:44:33Z
@@ -19,3 +19,14 @@ if arch(wasm32)
package aeson
flags: -ordered-keymap
+
+-- for reflex-frp
+
+-- GHC 9.10 compat
+source-repository-package
+ type: git
+ location: https://github.com/amesgen/reflex-dom
+ tag: e43e0525d643f656a0a5b0f10e13e2a04712cd4e
+ subdir: reflex-dom-core
+
+allow-newer: dependent-sum-template:template-haskell
diff --git a/frontend/reflex-todomvc.html b/frontend/reflex-todomvc.html
new file mode 100644
index 0000000..fcfeb2b
--- /dev/null
+++ b/frontend/reflex-todomvc.html
@@ -0,0 +1,12 @@
+
+
+
+
+
+ TodoMVC | Reflex FRP example via GHC WASM
+
+
+
+
+
+
diff --git a/ghc-wasm-miso-examples.cabal b/ghc-wasm-miso-examples.cabal
index d8e3fe1..3ba39d0 100644
--- a/ghc-wasm-miso-examples.cabal
+++ b/ghc-wasm-miso-examples.cabal
@@ -15,6 +15,7 @@ executable ghc-wasm-miso-examples
, miso
, mtl
, random
+ , reflex-todomvc
, text
hs-source-dirs: app
default-language: GHC2021
diff --git a/reflex-todomvc/reflex-todomvc.cabal b/reflex-todomvc/reflex-todomvc.cabal
new file mode 100644
index 0000000..139501f
--- /dev/null
+++ b/reflex-todomvc/reflex-todomvc.cabal
@@ -0,0 +1,25 @@
+Cabal-version: 3.0
+Name: reflex-todomvc
+Version: 0.1
+Synopsis: Functional Reactive TodoMVC
+Description: An implementation of the TodoMVC specification using the Reflex-DOM functional reactive DOM library
+License: BSD-3-Clause
+License-file: LICENSE
+Author: Ryan Trinkle
+Maintainer: ryan.trinkle@gmail.com
+Stability: Experimental
+Category: FRP
+
+library
+ hs-source-dirs: src
+ build-depends:
+ base,
+ reflex,
+ ghcjs-dom == 0.9.*,
+ reflex-dom-core,
+ containers,
+ text,
+ mtl
+ exposed-modules:
+ Reflex.TodoMVC
+ ghc-options: -Wall -fwarn-tabs -funbox-strict-fields -O2
diff --git a/reflex-todomvc/src/Reflex/TodoMVC.hs b/reflex-todomvc/src/Reflex/TodoMVC.hs
new file mode 100644
index 0000000..3556574
--- /dev/null
+++ b/reflex-todomvc/src/Reflex/TodoMVC.hs
@@ -0,0 +1,322 @@
+{-# LANGUAGE ConstraintKinds #-}
+{-# LANGUAGE FlexibleContexts #-}
+{-# LANGUAGE OverloadedStrings #-}
+{-# LANGUAGE RecursiveDo #-}
+{-# LANGUAGE ScopedTypeVariables #-}
+{-# LANGUAGE TypeFamilies #-}
+module Reflex.TodoMVC where
+
+import Prelude hiding (mapM, mapM_, sequence)
+
+import Control.Monad hiding (mapM, mapM_, forM, forM_, sequence)
+import Control.Monad.Fix
+import Data.Map (Map)
+import qualified Data.Map as Map
+import Data.Foldable
+import Data.Monoid ((<>))
+import Data.Text (Text)
+import qualified Data.Text as T
+import qualified Data.Text.Encoding as T
+
+import GHCJS.DOM.Types (JSM)
+
+import Reflex
+import Reflex.Dom.Core
+
+--------------------------------------------------------------------------------
+-- Model
+--------------------------------------------------------------------------------
+
+data Task
+ = Task { taskDescription :: Text
+ , taskCompleted :: Bool
+ }
+ deriving (Show, Eq)
+
+-- | Add a new value to a map; automatically choose an unused key
+insertNew_ :: (Enum k, Ord k) => v -> Map k v -> Map k v
+insertNew_ v m = case Map.maxViewWithKey m of
+ Nothing -> Map.singleton (toEnum 0) v
+ Just ((k, _), _) -> Map.insert (succ k) v m
+
+initialTasks :: Map Int Task
+initialTasks = Map.empty
+
+--------------------------------------------------------------------------------
+-- Filters
+--------------------------------------------------------------------------------
+
+-- | Subsets of the task list that can be selected by the user
+data Filter
+ = All -- ^ All tasks
+ | Active -- ^ Uncompleted tasks
+ | Completed -- ^ Completed tasks
+ deriving (Show, Eq)
+
+-- | Determine whether this Task should be shown when this Filter is in effect
+satisfiesFilter :: Filter -> Task -> Bool
+satisfiesFilter f = case f of
+ All -> const True
+ Active -> not . taskCompleted
+ Completed -> taskCompleted
+
+--------------------------------------------------------------------------------
+-- View
+--------------------------------------------------------------------------------
+
+main :: JSM ()
+main = mainWidgetWithCss styleCss todoMVC
+ where
+ -- WASM backend is currently lacking TH
+ styleCss = T.encodeUtf8 "html,\nbody {\n\tmargin: 0;\n\tpadding: 0;\n}\n\nbutton {\n\tmargin: 0;\n\tpadding: 0;\n\tborder: 0;\n\tbackground: none;\n\tfont-size: 100%;\n\tvertical-align: baseline;\n\tfont-family: inherit;\n\tfont-weight: inherit;\n\tcolor: inherit;\n\t-webkit-appearance: none;\n\tappearance: none;\n\t-webkit-font-smoothing: antialiased;\n\t-moz-osx-font-smoothing: grayscale;\n}\n\nbody {\n\tfont: 14px 'Helvetica Neue', Helvetica, Arial, sans-serif;\n\tline-height: 1.4em;\n\tbackground: #f5f5f5;\n\tcolor: #111111;\n\tmin-width: 230px;\n\tmax-width: 550px;\n\tmargin: 0 auto;\n\t-webkit-font-smoothing: antialiased;\n\t-moz-osx-font-smoothing: grayscale;\n\tfont-weight: 300;\n}\n\n:focus {\n\toutline: 0;\n}\n\n.hidden {\n\tdisplay: none;\n}\n\n.todoapp {\n\tbackground: #fff;\n\tmargin: 130px 0 40px 0;\n\tposition: relative;\n\tbox-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.2),\n\t 0 25px 50px 0 rgba(0, 0, 0, 0.1);\n}\n\n.todoapp input::-webkit-input-placeholder {\n\tfont-style: italic;\n\tfont-weight: 300;\n\tcolor: rgba(0, 0, 0, 0.4);\n}\n\n.todoapp input::-moz-placeholder {\n\tfont-style: italic;\n\tfont-weight: 300;\n\tcolor: rgba(0, 0, 0, 0.4);\n}\n\n.todoapp input::input-placeholder {\n\tfont-style: italic;\n\tfont-weight: 300;\n\tcolor: rgba(0, 0, 0, 0.4);\n}\n\n.todoapp h1 {\n\tposition: absolute;\n\ttop: -140px;\n\twidth: 100%;\n\tfont-size: 80px;\n\tfont-weight: 200;\n\ttext-align: center;\n\tcolor: #b83f45;\n\t-webkit-text-rendering: optimizeLegibility;\n\t-moz-text-rendering: optimizeLegibility;\n\ttext-rendering: optimizeLegibility;\n}\n\n.new-todo,\n.edit {\n\tposition: relative;\n\tmargin: 0;\n\twidth: 100%;\n\tfont-size: 24px;\n\tfont-family: inherit;\n\tfont-weight: inherit;\n\tline-height: 1.4em;\n\tcolor: inherit;\n\tpadding: 6px;\n\tborder: 1px solid #999;\n\tbox-shadow: inset 0 -1px 5px 0 rgba(0, 0, 0, 0.2);\n\tbox-sizing: border-box;\n\t-webkit-font-smoothing: antialiased;\n\t-moz-osx-font-smoothing: grayscale;\n}\n\n.new-todo {\n\tpadding: 16px 16px 16px 60px;\n\tborder: none;\n\tbackground: rgba(0, 0, 0, 0.003);\n\tbox-shadow: inset 0 -2px 1px rgba(0,0,0,0.03);\n}\n\n.main {\n\tposition: relative;\n\tz-index: 2;\n\tborder-top: 1px solid #e6e6e6;\n}\n\n.toggle-all {\n\twidth: 1px;\n\theight: 1px;\n\tborder: none; /* Mobile Safari */\n\topacity: 0;\n\tposition: absolute;\n\tright: 100%;\n\tbottom: 100%;\n}\n\n.toggle-all + label {\n\twidth: 60px;\n\theight: 34px;\n\tfont-size: 0;\n\tposition: absolute;\n\ttop: -52px;\n\tleft: -13px;\n\t-webkit-transform: rotate(90deg);\n\ttransform: rotate(90deg);\n}\n\n.toggle-all + label:before {\n\tcontent: '\10095';\n\tfont-size: 22px;\n\tcolor: #e6e6e6;\n\tpadding: 10px 27px 10px 27px;\n}\n\n.toggle-all:checked + label:before {\n\tcolor: #737373;\n}\n\n.todo-list {\n\tmargin: 0;\n\tpadding: 0;\n\tlist-style: none;\n}\n\n.todo-list li {\n\tposition: relative;\n\tfont-size: 24px;\n\tborder-bottom: 1px solid #ededed;\n}\n\n.todo-list li:last-child {\n\tborder-bottom: none;\n}\n\n.todo-list li.editing {\n\tborder-bottom: none;\n\tpadding: 0;\n}\n\n.todo-list li.editing .edit {\n\tdisplay: block;\n\twidth: calc(100% - 43px);\n\tpadding: 12px 16px;\n\tmargin: 0 0 0 43px;\n}\n\n.todo-list li.editing .view {\n\tdisplay: none;\n}\n\n.todo-list li .toggle {\n\ttext-align: center;\n\twidth: 40px;\n\t/* auto, since non-WebKit browsers doesn't support input styling */\n\theight: auto;\n\tposition: absolute;\n\ttop: 0;\n\tbottom: 0;\n\tmargin: auto 0;\n\tborder: none; /* Mobile Safari */\n\t-webkit-appearance: none;\n\tappearance: none;\n}\n\n.todo-list li .toggle {\n\topacity: 0;\n}\n\n.todo-list li .toggle + label {\n\t/*\n\t\tFirefox requires `#` to be escaped - https://bugzilla.mozilla.org/show_bug.cgi?id=922433\n\t\tIE and Edge requires *everything* to be escaped to render, so we do that instead of just the `#` - https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/7157459/\n\t*/\n\tbackground-image: url('data:image/svg+xml;utf8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%2240%22%20height%3D%2240%22%20viewBox%3D%22-10%20-18%20100%20135%22%3E%3Ccircle%20cx%3D%2250%22%20cy%3D%2250%22%20r%3D%2250%22%20fill%3D%22none%22%20stroke%3D%22%23ededed%22%20stroke-width%3D%223%22/%3E%3C/svg%3E');\n\tbackground-repeat: no-repeat;\n\tbackground-position: center left;\n}\n\n.todo-list li .toggle:checked + label {\n\tbackground-image: url('data:image/svg+xml;utf8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%2240%22%20height%3D%2240%22%20viewBox%3D%22-10%20-18%20100%20135%22%3E%3Ccircle%20cx%3D%2250%22%20cy%3D%2250%22%20r%3D%2250%22%20fill%3D%22none%22%20stroke%3D%22%23bddad5%22%20stroke-width%3D%223%22/%3E%3Cpath%20fill%3D%22%235dc2af%22%20d%3D%22M72%2025L42%2071%2027%2056l-4%204%2020%2020%2034-52z%22/%3E%3C/svg%3E');\n}\n\n.todo-list li label {\n\tword-break: break-all;\n\tpadding: 15px 15px 15px 60px;\n\tdisplay: block;\n\tline-height: 1.2;\n\ttransition: color 0.4s;\n\tfont-weight: 400;\n\tcolor: #4d4d4d;\n}\n\n.todo-list li.completed label {\n\tcolor: #cdcdcd;\n\ttext-decoration: line-through;\n}\n\n.todo-list li .destroy {\n\tdisplay: none;\n\tposition: absolute;\n\ttop: 0;\n\tright: 10px;\n\tbottom: 0;\n\twidth: 40px;\n\theight: 40px;\n\tmargin: auto 0;\n\tfont-size: 30px;\n\tcolor: #cc9a9a;\n\tmargin-bottom: 11px;\n\ttransition: color 0.2s ease-out;\n}\n\n.todo-list li .destroy:hover {\n\tcolor: #af5b5e;\n}\n\n.todo-list li .destroy:after {\n\tcontent: '\215';\n}\n\n.todo-list li:hover .destroy {\n\tdisplay: block;\n}\n\n.todo-list li .edit {\n\tdisplay: none;\n}\n\n.todo-list li.editing:last-child {\n\tmargin-bottom: -1px;\n}\n\n.footer {\n\tpadding: 10px 15px;\n\theight: 20px;\n\ttext-align: center;\n\tfont-size: 15px;\n\tborder-top: 1px solid #e6e6e6;\n}\n\n.footer:before {\n\tcontent: '';\n\tposition: absolute;\n\tright: 0;\n\tbottom: 0;\n\tleft: 0;\n\theight: 50px;\n\toverflow: hidden;\n\tbox-shadow: 0 1px 1px rgba(0, 0, 0, 0.2),\n\t 0 8px 0 -3px #f6f6f6,\n\t 0 9px 1px -3px rgba(0, 0, 0, 0.2),\n\t 0 16px 0 -6px #f6f6f6,\n\t 0 17px 2px -6px rgba(0, 0, 0, 0.2);\n}\n\n.todo-count {\n\tfloat: left;\n\ttext-align: left;\n}\n\n.todo-count strong {\n\tfont-weight: 300;\n}\n\n.filters {\n\tmargin: 0;\n\tpadding: 0;\n\tlist-style: none;\n\tposition: absolute;\n\tright: 0;\n\tleft: 0;\n}\n\n.filters li {\n\tdisplay: inline;\n}\n\n.filters li a {\n\tcolor: inherit;\n\tmargin: 3px;\n\tpadding: 3px 7px;\n\ttext-decoration: none;\n\tborder: 1px solid transparent;\n\tborder-radius: 3px;\n}\n\n.filters li a:hover {\n\tborder-color: rgba(175, 47, 47, 0.1);\n}\n\n.filters li a.selected {\n\tborder-color: rgba(175, 47, 47, 0.2);\n}\n\n.clear-completed,\nhtml .clear-completed:active {\n\tfloat: right;\n\tposition: relative;\n\tline-height: 20px;\n\ttext-decoration: none;\n\tcursor: pointer;\n}\n\n.clear-completed:hover {\n\ttext-decoration: underline;\n}\n\n.info {\n\tmargin: 65px auto 0;\n\tcolor: #4d4d4d;\n\tfont-size: 11px;\n\ttext-shadow: 0 1px 0 rgba(255, 255, 255, 0.5);\n\ttext-align: center;\n}\n\n.info p {\n\tline-height: 1;\n}\n\n.info a {\n\tcolor: inherit;\n\ttext-decoration: none;\n\tfont-weight: 400;\n}\n\n.info a:hover {\n\ttext-decoration: underline;\n}\n\n/*\n\tHack to remove background from Mobile Safari.\n\tCan't use it globally since it destroys checkboxes in Firefox\n*/\n@media screen and (-webkit-min-device-pixel-ratio:0) {\n\t.toggle-all,\n\t.todo-list li .toggle {\n\t\tbackground: none;\n\t}\n\n\t.todo-list li .toggle {\n\t\theight: 40px;\n\t}\n}\n\n@media (max-width: 430px) {\n\t.footer {\n\t\theight: 50px;\n\t}\n\n\t.filters {\n\t\tbottom: 10px;\n\t}\n}\n"
+
+todoMVC
+ :: ( DomBuilder t m
+ , DomBuilderSpace m ~ GhcjsDomSpace
+ , MonadFix m
+ , MonadHold t m
+ , PostBuild t m
+ )
+ => m ()
+todoMVC = el "div" $ do
+ elAttr "section" ("class" =: "todoapp") $ do
+ mainHeader
+ rec tasks <- foldDyn ($) initialTasks $ mergeWith (.)
+ [ fmap insertNew_ newTask
+ , listModifyTasks
+ , fmap (const $ Map.filter $ not . taskCompleted) clearCompleted -- Call out the type and purpose of these things
+ ]
+ newTask <- taskEntry
+ listModifyTasks <- taskList activeFilter tasks
+ (activeFilter, clearCompleted) <- controls tasks
+ return ()
+ infoFooter
+
+-- | Display the main header
+mainHeader :: DomBuilder t m => m ()
+mainHeader = el "h1" $ text "todos"
+
+-- | Strip leading and trailing whitespace from the user's entry, and discard it if nothing remains
+stripDescription :: Text -> Maybe Text
+stripDescription d =
+ let trimmed = T.strip d
+ in if T.null trimmed
+ then Nothing
+ else Just trimmed
+
+keyCodeIs :: Key -> KeyCode -> Bool
+keyCodeIs k c = keyCodeLookup c == k
+
+-- | Display an input field; produce new Tasks when the user creates them
+taskEntry
+ :: ( DomBuilder t m
+ , MonadFix m
+ , PostBuild t m
+ , DomBuilderSpace m ~ GhcjsDomSpace
+ )
+ => m (Event t Task)
+taskEntry = el "header" $ do
+ -- Create the textbox; it will be cleared whenever the user presses enter
+ rec let newValueEntered = keypress Enter descriptionBox
+ descriptionBox <- inputElement $ def
+ & inputElementConfig_setValue .~ fmap (const "") newValueEntered
+ & inputElementConfig_elementConfig . elementConfig_initialAttributes .~
+ mconcat [ "class" =: "new-todo"
+ , "placeholder" =: "What needs to be done?"
+ , "name" =: "newTodo"
+ , "type" =: "text"
+ ]
+ -- -- Request focus on this element when the widget is done being built
+ -- schedulePostBuild $ liftIO $ focus $ _textInput_element descriptionBox
+ let -- | Get the current value of the textbox whenever the user hits enter
+ newValue = tag (current $ value descriptionBox) newValueEntered
+ -- -- Set focus when the user enters a new Task
+ -- performEvent_ $ fmap (const $ liftIO $ focus $ _textInput_element descriptionBox) newValueEntered
+ return $ fmap (\d -> Task d False) $ fmapMaybe stripDescription newValue
+
+-- | Display the user's Tasks, subject to a Filter; return requested modifications to the Task list
+taskList
+ :: ( DomBuilder t m
+ , DomBuilderSpace m ~ GhcjsDomSpace
+ , PostBuild t m
+ , MonadHold t m
+ , MonadFix m
+ , Ord k
+ )
+ => Dynamic t Filter
+ -> Dynamic t (Map k Task)
+ -> m (Event t (Map k Task -> Map k Task))
+taskList activeFilter tasks = elAttr "section" ("class" =: "main") $ do
+ let toggleAllState = all taskCompleted . Map.elems <$> tasks
+ toggleAllAttrs = ffor tasks $ \t -> "class" =: "toggle-all" <> "name" =: "toggle" <> if Map.null t then "style" =: "visibility:hidden" else mempty
+ toggleAll <- toggleInput toggleAllAttrs toggleAllState
+ elAttr "label" ("for" =: "toggle-all") $ text "Mark all as complete"
+ -- Filter the item list
+ let visibleTasks = zipDynWith (Map.filter . satisfiesFilter) activeFilter tasks
+ -- Hide the item list itself if there are no items
+ let itemListAttrs = ffor visibleTasks $ \t -> mconcat
+ [ "class" =: "todo-list"
+ , if Map.null t then "style" =: "visibility:hidden" else mempty
+ ]
+ -- Display the items
+ items <- elDynAttr "ul" itemListAttrs $ list visibleTasks todoItem
+ -- Aggregate the changes produced by the elements
+ let combineItemChanges = fmap (foldl' (.) id) . mergeList . map (\(k, v) -> fmap (flip Map.update k) v) . Map.toList
+ itemChangeEvent = fmap combineItemChanges items
+ itemChanges = switch $ current itemChangeEvent
+ return itemChanges
+
+toggleInput
+ :: ( DomBuilder t m
+ , DomBuilderSpace m ~ GhcjsDomSpace
+ , MonadFix m
+ , MonadHold t m
+ , PostBuild t m
+ )
+ => Dynamic t (Map AttributeName Text)
+ -> Dynamic t Bool
+ -> m (Event t ())
+toggleInput dynAttrs dynChecked = do
+ let attrs = (<> "class" =: "toggle") . ("type" =: "checkbox" <>) <$> dynAttrs
+ updatedAttrs = fmap Just <$> updated dynAttrs
+ updatedChecked = updated dynChecked
+ initialAttrs <- sample $ current attrs
+ initialChecked <- sample $ current dynChecked
+ domEvent Click <$> inputElement (def
+ & inputElementConfig_initialChecked .~ initialChecked
+ & inputElementConfig_setChecked .~ updatedChecked
+ & inputElementConfig_elementConfig . elementConfig_modifyAttributes .~ updatedAttrs
+ & inputElementConfig_elementConfig . elementConfig_initialAttributes .~ initialAttrs)
+
+buildCompletedCheckbox
+ :: ( DomBuilder t m
+ , DomBuilderSpace m ~ GhcjsDomSpace
+ , MonadFix m
+ , MonadHold t m
+ , PostBuild t m
+ )
+ => Dynamic t Task
+ -> Dynamic t Text
+ -> m (Event t Bool, Event t (), Event t ())
+buildCompletedCheckbox todo description = elAttr "div" ("class" =: "view") $ do
+ -- Display the todo item's completed status, and allow it to be set
+ completed <- holdUniqDyn $ fmap taskCompleted todo
+ checkboxClicked <- toggleInput (constDyn mempty) completed
+ let setCompleted = fmap not $ tag (current completed) checkboxClicked
+ -- Display the todo item's name for viewing purposes
+ (descriptionLabel, _) <- el' "label" $ dynText description
+ -- Display the button for deleting the todo item
+ (destroyButton, _) <- elAttr' "button" ("class" =: "destroy") $ return ()
+ return ( setCompleted
+ , domEvent Click destroyButton
+ , void $ domEvent Dblclick descriptionLabel
+ )
+
+-- | Display an individual todo item
+todoItem
+ :: ( DomBuilder t m
+ , DomBuilderSpace m ~ GhcjsDomSpace
+ , MonadFix m
+ , MonadHold t m
+ , PostBuild t m
+ )
+ => Dynamic t Task
+ -> m (Event t (Task -> Maybe Task))
+todoItem todo = do
+ description <- holdUniqDyn $ fmap taskDescription todo
+ rec -- Construct the attributes for our element
+ let attrs = ffor2 todo editing' $ \t e -> Map.singleton "class" $ T.unwords $ concat
+ [ [ "completed" | taskCompleted t ]
+ , [ "editing" | e ]
+ ]
+ (editing', changeTodo) <- elDynAttr "li" attrs $ do
+ (setCompleted, destroy, startEditing) <- buildCompletedCheckbox todo description
+ -- Set the current value of the editBox whenever we start editing (it's not visible in non-editing mode)
+ let setEditValue = tag (current description) $ ffilter id $ updated editing'
+ editBox <- inputElement $ def
+ & inputElementConfig_setValue .~ setEditValue
+ & inputElementConfig_elementConfig . elementConfig_initialAttributes
+ .~ ("class" =: "edit" <> "name" =: "title")
+ let -- Set the todo item's description when the user leaves the textbox or presses enter in it
+ setDescription = tag (current $ value editBox) $ leftmost
+ [ keypress Enter editBox
+ , domEvent Blur editBox
+ ]
+ -- Cancel editing (without changing the item's description) when the user presses escape in the textbox
+ cancelEdit = keypress Escape editBox
+ -- Put together all the ways the todo item can change itself
+ changeSelf = mergeWith (>=>) [ fmap (\c t -> Just $ t { taskCompleted = c }) setCompleted
+ , fmap (const $ const Nothing) destroy
+ , fmap (\d t -> fmap (\trimmed -> t { taskDescription = trimmed }) $ stripDescription d) setDescription
+ ]
+ -- Set focus on the edit box when we enter edit mode
+-- postGui <- askPostGui
+-- performEvent_ $ fmap (const $ liftIO $ void $ forkIO $ threadDelay 1000 >> postGui (liftIO $ focus $ _textInput_element editBox)) startEditing -- Without the delay, the focus doesn't take effect because the element hasn't become unhidden yet; we need to use postGui to ensure that this is threadsafe when built with GTK
+ -- Determine the current editing state; initially false, but can be modified by various events
+ editing <- holdDyn False $ leftmost [ fmap (const True) startEditing
+ , fmap (const False) setDescription
+ , fmap (const False) cancelEdit
+ ]
+ return (editing, changeSelf)
+ -- Return an event that fires whenever we change ourselves
+ return changeTodo
+
+buildActiveFilter
+ :: ( DomBuilder t m
+ , PostBuild t m
+ , MonadHold t m
+ , MonadFix m
+ )
+ => m (Dynamic t Filter)
+buildActiveFilter = elAttr "ul" ("class" =: "filters") $ do
+ rec activeFilter <- holdDyn All setFilter
+ let filterButton f = el "li" $ do
+ let buttonAttrs = ffor activeFilter $ \af -> "class" =: if f == af then "selected" else ""
+ (e, _) <- elDynAttr' "a" buttonAttrs $ text $ T.pack $ show f
+ return $ fmap (const f) (domEvent Click e)
+ allButton <- filterButton All
+ text " "
+ activeButton <- filterButton Active
+ text " "
+ completedButton <- filterButton Completed
+ let setFilter = leftmost [allButton, activeButton, completedButton]
+ return activeFilter
+
+-- | Display the control footer; return the user's currently-selected filter and an event that fires when the user chooses to clear all completed events
+controls
+ :: ( DomBuilder t m
+ , PostBuild t m
+ , MonadHold t m
+ , MonadFix m
+ )
+ => Dynamic t (Map k Task)
+ -> m (Dynamic t Filter, Event t ())
+controls tasks = do
+ -- Determine the attributes for the footer; it is invisible when there are no todo items
+ let controlsAttrs = ffor tasks $ \t -> "class" =: "footer" <> if Map.null t then "style" =: "visibility:hidden" else mempty
+ elDynAttr "footer" controlsAttrs $ do
+ -- Compute the number of completed and uncompleted tasks
+ let (tasksCompleted, tasksLeft) = splitDynPure $ ffor tasks $ \m ->
+ let completed = Map.size $ Map.filter taskCompleted m
+ in (completed, Map.size m - completed)
+ elAttr "span" ("class" =: "todo-count") $ do
+ el "strong" $ dynText $ fmap (T.pack . show) tasksLeft
+ dynText $ fmap (\n -> (if n == 1 then " item" else " items") <> " left") tasksLeft
+ activeFilter <- buildActiveFilter
+ let clearCompletedAttrs = ffor tasksCompleted $ \n -> mconcat
+ [ "class" =: "clear-completed"
+ , if n > 0 then mempty else "hidden" =: ""
+ ]
+ (clearCompletedAttrsButton, _) <- elDynAttr' "button" clearCompletedAttrs $ dynText $ ffor tasksCompleted $ \n -> "Clear completed (" <> T.pack (show n) <> ")"
+ return (activeFilter, domEvent Click clearCompletedAttrsButton)
+
+-- | Display static information about the application
+infoFooter :: DomBuilder t m => m ()
+infoFooter = elAttr "footer" ("class" =: "info") $ do
+ el "p" $ text "Click to edit a todo"
+ el "p" $ do
+ text "Written by "
+ elAttr "a" ("href" =: "https://github.com/ryantrinkle") $ text "Ryan Trinkle"
+ el "p" $ do
+ text "Part of "
+ elAttr "a" ("href" =: "http://todomvc.com") $ text "TodoMVC"