From fc641683865d5672ee3501db16c7cb968eac8228 Mon Sep 17 00:00:00 2001 From: Owen Graves Date: Tue, 22 Oct 2024 16:33:11 -0500 Subject: [PATCH 1/2] Add format repo script --- .tidyrc.json | 10 +++ compose.yml | 4 +- scripts/build | 6 +- scripts/format-repo | 15 ++++ spago.lock | 163 ++++++++++++++++++++++---------------------- 5 files changed, 110 insertions(+), 88 deletions(-) create mode 100644 .tidyrc.json create mode 100755 scripts/format-repo diff --git a/.tidyrc.json b/.tidyrc.json new file mode 100644 index 0000000..edcadc3 --- /dev/null +++ b/.tidyrc.json @@ -0,0 +1,10 @@ +{ + "importSort": "source", + "importWrap": "source", + "indent": 2, + "operatorsFile": null, + "ribbon": 1, + "typeArrowPlacement": "last", + "unicode": "never", + "width": null +} diff --git a/compose.yml b/compose.yml index f6672ae..b07df0f 100644 --- a/compose.yml +++ b/compose.yml @@ -1,12 +1,12 @@ services: dev: - image: ghcr.io/flipstone/purescript-tools:debian-stable-purescript-0.15.15-2024-09-05-15466b3 + image: ghcr.io/flipstone/purescript-tools:debian-stable-purescript-0.15.15-2024-10-21-d530b53 environment: IN_CONTAINER: 'true' stdin_open: true tty: true volumes: - .:/purescript-erumu - command: spago build + command: ./scripts/build working_dir: /purescript-erumu diff --git a/scripts/build b/scripts/build index 377a3ec..851fef9 100755 --- a/scripts/build +++ b/scripts/build @@ -4,8 +4,8 @@ set -e source "${BASH_SOURCE%/*}/lib/run-in-container.sh" -echo "=====" -echo "===== $0 $* =====" -echo "=====" +echo "Formatting with purs-tidy" +./scripts/format-repo +echo "Building" spago build diff --git a/scripts/format-repo b/scripts/format-repo new file mode 100755 index 0000000..d028c7e --- /dev/null +++ b/scripts/format-repo @@ -0,0 +1,15 @@ +#!/usr/bin/env bash + +set -e + +source "${BASH_SOURCE%/*}/../scripts/lib/run-in-container.sh" + +cd $(git rev-parse --show-toplevel) + +if [ "$CI" ]; then + MODE=check +else + MODE=format-in-place +fi + +purs-tidy $MODE --config-require "./src/**/*.purs" diff --git a/spago.lock b/spago.lock index 2146a0c..1bef26f 100644 --- a/spago.lock +++ b/spago.lock @@ -2,89 +2,86 @@ workspace: packages: erumu: path: ./ - core: - dependencies: - - arrays - - bifunctors - - console - - effect - - either - - enums - - foldable-traversable - - foreign - - foreign-object - - identity - - lists - - maybe - - newtype - - parallel - - partial - - prelude - - refs - - strings - - unsafe-coerce - - web-dom - - web-events - - web-file - - web-html - build_plan: - - arrays - - bifunctors - - console - - const - - contravariant - - control - - datetime - - distributive - - effect - - either - - enums - - exceptions - - exists - - foldable-traversable - - foreign - - foreign-object - - functions - - functors - - gen - - identity - - integers - - invariant - - js-date - - lazy - - lists - - maybe - - media-types - - newtype - - nonempty - - now - - nullable - - numbers - - ordered-collections - - orders - - parallel - - partial - - prelude - - profunctor - - refs - - safe-coerce - - st - - strings - - tailrec - - transformers - - tuples - - type-equality - - typelevel-prelude - - unfoldable - - unsafe-coerce - - web-dom - - web-events - - web-file - - web-html - - web-storage - test: - dependencies: [] - build_plan: [] + dependencies: + - arrays + - bifunctors + - console + - effect + - either + - enums + - foldable-traversable + - foreign + - foreign-object + - identity + - lists + - maybe + - newtype + - parallel + - partial + - prelude + - refs + - strings + - unsafe-coerce + - web-dom + - web-events + - web-file + - web-html + test_dependencies: [] + build_plan: + - arrays + - bifunctors + - console + - const + - contravariant + - control + - datetime + - distributive + - effect + - either + - enums + - exceptions + - exists + - foldable-traversable + - foreign + - foreign-object + - functions + - functors + - gen + - identity + - integers + - invariant + - js-date + - lazy + - lists + - maybe + - media-types + - newtype + - nonempty + - now + - nullable + - numbers + - ordered-collections + - orders + - parallel + - partial + - prelude + - profunctor + - refs + - safe-coerce + - st + - strings + - tailrec + - transformers + - tuples + - type-equality + - typelevel-prelude + - unfoldable + - unsafe-coerce + - web-dom + - web-events + - web-file + - web-html + - web-storage package_set: address: registry: 58.0.1 From 4a074ff58de8446c9023546ae666400e389d3f38 Mon Sep 17 00:00:00 2001 From: Owen Graves Date: Tue, 22 Oct 2024 16:33:53 -0500 Subject: [PATCH 2/2] Initial purs-tidy format --- src/DOM/Erumu/App.purs | 119 ++++---- src/DOM/Erumu/Array.purs | 100 ++++--- src/DOM/Erumu/Array/Dynamic.purs | 40 +-- src/DOM/Erumu/Decode.purs | 10 +- src/DOM/Erumu/Either.purs | 22 +- src/DOM/Erumu/Form.purs | 89 +++--- src/DOM/Erumu/HTML.purs | 2 +- src/DOM/Erumu/HTML/Attributes.purs | 3 +- src/DOM/Erumu/HTML/Decoder.purs | 5 +- src/DOM/Erumu/HTML/Elements.purs | 71 ++++- src/DOM/Erumu/HTML/Events.purs | 1 - src/DOM/Erumu/Maybe.purs | 39 +-- src/DOM/Erumu/Nodes.purs | 19 +- src/DOM/Erumu/RunWithCommands.purs | 135 +++++---- src/DOM/Erumu/Types.purs | 325 ++++++++++++---------- src/DOM/Erumu/Widget/CheckboxInput.purs | 40 +-- src/DOM/Erumu/Widget/FileInput.purs | 32 ++- src/DOM/Erumu/Widget/RadioInputGroup.purs | 42 +-- src/DOM/Erumu/Widget/Select.purs | 31 ++- src/DOM/Erumu/Widget/TextArea.purs | 32 ++- src/DOM/Erumu/Widget/TextInput.purs | 42 +-- src/DOM/Virtual.purs | 38 ++- src/DOM/Virtual/App.purs | 43 +-- 23 files changed, 739 insertions(+), 541 deletions(-) diff --git a/src/DOM/Erumu/App.purs b/src/DOM/Erumu/App.purs index 28ae401..13e550e 100644 --- a/src/DOM/Erumu/App.purs +++ b/src/DOM/Erumu/App.purs @@ -46,100 +46,111 @@ type RenderFn model msg = model -> HTML msg -- * Dispatching a message is always "safe" -- - e.g. you can save a dispatch function for later call it at any time. -- -newApp :: forall m model msg. - (model -> HTML msg) - -> UpdateFn m model msg - -> model - -> (m Unit -> Effect Unit) - -> (Effect Unit -> m Unit) - -> Effect (App m model msg) +newApp :: + forall m model msg. + (model -> HTML msg) -> + UpdateFn m model msg -> + model -> + (m Unit -> Effect Unit) -> + (Effect Unit -> m Unit) -> + Effect (App m model msg) newApp render update init execM liftM = do virtualApp <- Virtual.newApp (node "div" [] []) - stRef <- Ref.new { model: init - , messages: Nil - , processing: false - } + stRef <- Ref.new + { model: init + , messages: Nil + , processing: false + } let app = App { update, render, stRef, virtualApp, execM, liftM } renderApp app pure app -mountApp :: forall m model msg. - String - -> App m model msg - -> Effect (Maybe String) +mountApp :: + forall m model msg. + String -> + App m model msg -> + Effect (Maybe String) mountApp containerId (App { virtualApp }) = Virtual.mountApp containerId virtualApp - -dispatch :: forall m model msg. - App m model msg - -> msg - -> Effect Unit +dispatch :: + forall m model msg. + App m model msg -> + msg -> + Effect Unit dispatch app@(App { stRef }) msg = do pushMessage stRef msg processMessages app -renderApp :: forall m model msg. - App m model msg - -> Effect Unit +renderApp :: + forall m model msg. + App m model msg -> + Effect Unit renderApp app@(App { virtualApp, render, stRef }) = do st <- Ref.read stRef Virtual.rerenderApp (toVTree (render st.model) (dispatch app)) - virtualApp + virtualApp -tryTakeMessages :: forall model msg. - Ref (DispatchState model msg) - -> Effect (Maybe (List msg)) +tryTakeMessages :: + forall model msg. + Ref (DispatchState model msg) -> + Effect (Maybe (List msg)) tryTakeMessages stRef = (flip Ref.modify') stRef $ \st -> - if st.processing - then { state: st, value: Nothing } - else { state: st { messages = Nil } - , value: Just (reverse st.messages) - } - -setProcessing :: forall model msg. - Ref (DispatchState model msg) - -> Boolean - -> Effect Unit + if st.processing then { state: st, value: Nothing } + else + { state: st { messages = Nil } + , value: Just (reverse st.messages) + } + +setProcessing :: + forall model msg. + Ref (DispatchState model msg) -> + Boolean -> + Effect Unit setProcessing stRef processing = void $ Ref.modify (_ { processing = processing }) stRef -pushMessage :: forall model msg. - Ref (DispatchState model msg) - -> msg - -> Effect Unit +pushMessage :: + forall model msg. + Ref (DispatchState model msg) -> + msg -> + Effect Unit pushMessage stRef msg = void $ Ref.modify (\st -> st { messages = Cons msg st.messages }) stRef -processMessages :: forall m msg model. - App m model msg - -> Effect Unit +processMessages :: + forall m msg model. + App m model msg -> + Effect Unit processMessages app@(App { stRef }) = void $ do taken <- tryTakeMessages stRef case taken of - Nothing -> pure unit -- messages are already being processed - Just Nil -> pure unit -- there are no messages to process + Nothing -> pure unit -- messages are already being processed + Just Nil -> pure unit -- there are no messages to process Just msgs -> do setProcessing stRef true traverse_ (processMessage app) msgs setProcessing stRef false processMessages app -processMessage :: forall m msg model. - App m model msg - -> msg - -> Effect Unit +processMessage :: + forall m msg model. + App m model msg -> + msg -> + Effect Unit processMessage app@(App { update, stRef, execM, liftM }) msg = void $ do command <- (flip Ref.modify') stRef $ \st -> - let r = update msg st.model - in { state: st { model = r.model } - , value: r.command - } + let + r = update msg st.model + in + { state: st { model = r.model } + , value: r.command + } renderApp app runcmd command (liftM <<< dispatch app) execM diff --git a/src/DOM/Erumu/Array.purs b/src/DOM/Erumu/Array.purs index d2a8ff6..f32be86 100644 --- a/src/DOM/Erumu/Array.purs +++ b/src/DOM/Erumu/Array.purs @@ -34,14 +34,15 @@ import DOM.Erumu.Types ((!), HTML, Return(..), UpdateResult, liftReturn, liftUpd data Msg msg = Msg Int msg -render :: forall msg model. - (model -> HTML msg) - -> Array model - -> Array (HTML (Msg msg)) +render :: + forall msg model. + (model -> HTML msg) -> + Array model -> + Array (HTML (Msg msg)) render renderItem items = mapWithIndex renderIdx items - where - renderIdx idx item = (Msg idx) <$> renderItem item + where + renderIdx idx item = (Msg idx) <$> renderItem item -- renderLift allows you to render items that may contain a mixture of -- messages from parent a child (item) components. This can be useful in @@ -51,52 +52,59 @@ render renderItem items = -- into Array messages which will then need to be wrapped into parentMsgs using -- the appropriate parent-specific constructor -- The index of the item being rendered is provided for your convenience. -renderLift :: forall childMsg parentMsg model. - ((childMsg -> Msg childMsg) -> Int -> model -> HTML parentMsg) - -> Array model - -> Array (HTML parentMsg) +renderLift :: + forall childMsg parentMsg model. + ((childMsg -> Msg childMsg) -> Int -> model -> HTML parentMsg) -> + Array model -> + Array (HTML parentMsg) renderLift renderItem items = mapWithIndex renderIdx items - where - renderIdx idx item = renderItem (Msg idx) idx item - -update :: forall m msg model. - Applicative m - => (msg -> model -> UpdateResult m model msg) - -> Msg msg - -> Array model - -> Maybe (UpdateResult m (Array model) (Msg msg)) + where + renderIdx idx item = renderItem (Msg idx) idx item + +update :: + forall m msg model. + Applicative m => + (msg -> model -> UpdateResult m model msg) -> + Msg msg -> + Array model -> + Maybe (UpdateResult m (Array model) (Msg msg)) update f msg items = unwrap $ sequence $ updateF (Identity `compose2` f) msg items - where - compose2 :: forall a b c d. - (c -> d) - -> (a -> b -> c) - -> (a -> b -> d) - compose2 g h = \a b -> g (h a b) - -updateF :: forall f m msg model. - Functor f - => Applicative m - => (msg -> model -> f (UpdateResult m model msg)) - -> Msg msg - -> Array model - -> Maybe (f (UpdateResult m (Array model) (Msg msg))) + where + compose2 :: + forall a b c d. + (c -> d) -> + (a -> b -> c) -> + (a -> b -> d) + compose2 g h = \a b -> g (h a b) + +updateF :: + forall f m msg model. + Functor f => + Applicative m => + (msg -> model -> f (UpdateResult m model msg)) -> + Msg msg -> + Array model -> + Maybe (f (UpdateResult m (Array model) (Msg msg))) updateF updateItem (Msg idx msg) items = case items !! idx of Nothing -> Nothing Just oldItem -> Just $ - let liftItem newItem = fromMaybe items $ updateAt idx newItem items - in liftUpdate (Msg idx) (liftItem) <$> - updateItem msg oldItem + let + liftItem newItem = fromMaybe items $ updateAt idx newItem items + in + liftUpdate (Msg idx) (liftItem) <$> + updateItem msg oldItem -- Return -updateArrayWith :: forall m signal childmodel childmsg. - Applicative m - => (childmsg -> childmodel -> Return m childmodel childmsg signal) - -> Msg childmsg - -> Array childmodel - -> Return m (Array childmodel) (Msg childmsg) signal +updateArrayWith :: + forall m signal childmodel childmsg. + Applicative m => + (childmsg -> childmodel -> Return m childmodel childmsg signal) -> + Msg childmsg -> + Array childmodel -> + Return m (Array childmodel) (Msg childmsg) signal updateArrayWith updateFn (Msg idx msg) models = case models !! idx of @@ -104,7 +112,9 @@ updateArrayWith updateFn (Msg idx msg) models = Nothing -> Continue $ models ! [] Just oldItem -> - let liftModel newM = fromMaybe models $ updateAt idx newM models - in liftReturn (Msg idx) liftModel - $ updateFn msg oldItem + let + liftModel newM = fromMaybe models $ updateAt idx newM models + in + liftReturn (Msg idx) liftModel + $ updateFn msg oldItem diff --git a/src/DOM/Erumu/Array/Dynamic.purs b/src/DOM/Erumu/Array/Dynamic.purs index 11b36be..e66f9a0 100644 --- a/src/DOM/Erumu/Array/Dynamic.purs +++ b/src/DOM/Erumu/Array/Dynamic.purs @@ -1,6 +1,10 @@ module DOM.Erumu.Array.Dynamic ( Msg - , Edit, add, remove, editMsg, updateMsg + , Edit + , add + , remove + , editMsg + , updateMsg , render , update ) where @@ -13,8 +17,8 @@ import Data.Either (Either(..)) import DOM.Erumu.Array as Array import DOM.Erumu.Types (UpdateResult, HTML, liftUpdate, (!)) -data Edit model = - Add model +data Edit model + = Add model | Remove Int add :: forall model. model -> Edit model @@ -32,20 +36,24 @@ updateMsg = Msg <<< Right newtype Msg model msg = Msg (Either (Edit model) (Array.Msg msg)) -render :: forall model msg. - (Int -> model -> HTML (Either (Edit model) msg)) - -> Array model - -> Array (HTML (Msg model msg)) +render :: + forall model msg. + (Int -> model -> HTML (Either (Edit model) msg)) -> + Array model -> + Array (HTML (Msg model msg)) render renderItem items = - let renderItem' lifter idx model = Msg <<< map lifter <$> renderItem idx model - in Array.renderLift renderItem' items - -update :: forall m msg model. - Applicative m - => (msg -> model -> UpdateResult m model msg) - -> Msg model msg - -> Array model - -> Maybe (UpdateResult m (Array model) (Msg model msg)) + let + renderItem' lifter idx model = Msg <<< map lifter <$> renderItem idx model + in + Array.renderLift renderItem' items + +update :: + forall m msg model. + Applicative m => + (msg -> model -> UpdateResult m model msg) -> + Msg model msg -> + Array model -> + Maybe (UpdateResult m (Array model) (Msg model msg)) update _ (Msg (Left (Add model))) items = Just $ (items `snoc` model) ! [] diff --git a/src/DOM/Erumu/Decode.purs b/src/DOM/Erumu/Decode.purs index 0c60dca..b6b5b6b 100644 --- a/src/DOM/Erumu/Decode.purs +++ b/src/DOM/Erumu/Decode.purs @@ -34,7 +34,7 @@ event = Decode $ \e -> pure (pure e) domValue :: forall a b. (a -> Effect b) -> a -> Decode b domValue f a = Decode $ \_ -> pure <$> f a -domEffect :: Effect Unit -> Decode Unit +domEffect :: Effect Unit -> Decode Unit domEffect eff = Decode $ \_ -> pure <$> eff crash :: forall a. String -> Decode a @@ -51,10 +51,10 @@ instance applicativeDecode :: Applicative Decode where instance bindDecode :: Bind Decode where bind (Decode dA) f = Decode $ \e -> do - result <- dA e - case f <$> result of - Right (Decode dB) -> dB e - Left err -> pure (Left err) + result <- dA e + case f <$> result of + Right (Decode dB) -> dB e + Left err -> pure (Left err) instance monadDecode :: Monad Decode diff --git a/src/DOM/Erumu/Either.purs b/src/DOM/Erumu/Either.purs index d29ee10..17458a6 100644 --- a/src/DOM/Erumu/Either.purs +++ b/src/DOM/Erumu/Either.purs @@ -9,22 +9,24 @@ import Data.Either (Either(Left, Right)) import DOM.Erumu.Types (UpdateResult, liftUpdate, (!)) -updateLeft :: forall m msg model a. - Applicative m - => (model -> UpdateResult m model msg) - -> Either model a - -> UpdateResult m (Either model a) msg +updateLeft :: + forall m msg model a. + Applicative m => + (model -> UpdateResult m model msg) -> + Either model a -> + UpdateResult m (Either model a) msg updateLeft _ right@(Right _) = right ! [] updateLeft f (Left model) = liftUpdate identity Left (f model) -updateRight :: forall m msg model a. - Applicative m - => (model -> UpdateResult m model msg) - -> Either a model - -> UpdateResult m (Either a model) msg +updateRight :: + forall m msg model a. + Applicative m => + (model -> UpdateResult m model msg) -> + Either a model -> + UpdateResult m (Either a model) msg updateRight _ left@(Left _) = left ! [] diff --git a/src/DOM/Erumu/Form.purs b/src/DOM/Erumu/Form.purs index cf17027..2933e60 100644 --- a/src/DOM/Erumu/Form.purs +++ b/src/DOM/Erumu/Form.purs @@ -11,7 +11,8 @@ module DOM.Erumu.Form , getField , setField , nest - , disabledProp, placeholderProp + , disabledProp + , placeholderProp ) where import Prelude @@ -26,12 +27,13 @@ import DOM.Erumu.HTML (type_, input, select, value, textArea, disabled, placehol import DOM.Erumu.HTML.Decoder (textAreaValue, selectValue, inputValue) import DOM.Erumu.Types (HTML, Prop, onEventDecode) -data Model = - Field String +data Model + = Field String | Object (Foreign.Object Model) readModel :: Foreign -> F Model readModel f = Field <$> readString f + --case readString f of -- Right _ -> pure $ Field "foo" -- Left _ -> pure $ Field "bar" @@ -49,10 +51,10 @@ setField (Cons key rest) value (Field _) = Object (Foreign.singleton key (setField rest value (Field ""))) setField (Cons key rest) value (Object map) = - Object (Foreign.alter alter key map) + Object (Foreign.alter alter key map) where - alter Nothing = Just (setField rest value (Field "")) - alter (Just model) = Just (setField rest value model) + alter Nothing = Just (setField rest value (Field "")) + alter (Just model) = Just (setField rest value model) getChild :: String -> Model -> Model getChild key (Object map) = @@ -64,8 +66,8 @@ getField :: List String -> Model -> String getField Nil (Field s) = s getField (Cons key rest) (Object map) = case Foreign.lookup key map of - Just subForm -> getField rest subForm - Nothing -> "" + Just subForm -> getField rest subForm + Nothing -> "" getField _ _ = "" @@ -75,54 +77,67 @@ data Msg = empty :: Model empty = Object Foreign.empty -nest :: forall msg. - String - -> Model - -> ((Msg -> Msg) -> msg -> msg) - -> (Model -> HTML msg) - -> HTML msg +nest :: + forall msg. + String -> + Model -> + ((Msg -> Msg) -> msg -> msg) -> + (Model -> HTML msg) -> + HTML msg nest key model lift render = - lift pushKey <$> render subModel + lift pushKey <$> render subModel where - subModel = getChild key model - pushKey (SetField path value) = SetField (Cons key path) value + subModel = getChild key model + pushKey (SetField path value) = SetField (Cons key path) value disabledProp :: forall msg. Boolean -> Array (Prop msg) -disabledProp = if _ then [disabled "disabled"] else [] +disabledProp = if _ then [ disabled "disabled" ] else [] placeholderProp :: forall msg. Maybe String -> Array (Prop msg) -placeholderProp Nothing = [] -placeholderProp (Just s) = [placeholder s] +placeholderProp Nothing = [] +placeholderProp (Just s) = [ placeholder s ] textField :: Array (Prop Msg) -> Model -> String -> HTML Msg textField userProps model key = - let ourProps = [ type_ "text" - , onEventDecode "oninput" (decodeInputSetField (Cons key Nil)) - , value (getField (Cons key Nil) model) - ] - in input (ourProps <> userProps) [] + let + ourProps = + [ type_ "text" + , onEventDecode "oninput" (decodeInputSetField (Cons key Nil)) + , value (getField (Cons key Nil) model) + ] + in + input (ourProps <> userProps) [] passwordField :: Array (Prop Msg) -> Model -> String -> HTML Msg passwordField userProps model key = - let ourProps = [ type_ "password" - , onEventDecode "oninput" (decodeInputSetField (Cons key Nil)) - , value (getField (Cons key Nil) model) - ] - in input (ourProps <> userProps) [] + let + ourProps = + [ type_ "password" + , onEventDecode "oninput" (decodeInputSetField (Cons key Nil)) + , value (getField (Cons key Nil) model) + ] + in + input (ourProps <> userProps) [] textAreaField :: Array (Prop Msg) -> Model -> String -> HTML Msg textAreaField userProps model key = - let ourProps = [ onEventDecode "oninput" (decodeTextAreaSetField (Cons key Nil)) - , value (getField (Cons key Nil) model) - ] - in textArea (ourProps <> userProps) [] + let + ourProps = + [ onEventDecode "oninput" (decodeTextAreaSetField (Cons key Nil)) + , value (getField (Cons key Nil) model) + ] + in + textArea (ourProps <> userProps) [] selectField :: Array (Prop Msg) -> String -> Array (HTML Msg) -> HTML Msg selectField userProps key children = - let ourProps = [ onEventDecode "onchange" (decodeSelectSetField (Cons key Nil)) - ] + let + ourProps = + [ onEventDecode "onchange" (decodeSelectSetField (Cons key Nil)) + ] - in select (ourProps <> userProps) children + in + select (ourProps <> userProps) children update :: Msg -> Model -> Model update (SetField key value) m = setField key value m diff --git a/src/DOM/Erumu/HTML.purs b/src/DOM/Erumu/HTML.purs index 9ae4d21..188ef3b 100644 --- a/src/DOM/Erumu/HTML.purs +++ b/src/DOM/Erumu/HTML.purs @@ -5,5 +5,5 @@ module DOM.Erumu.HTML ) where import DOM.Erumu.HTML.Attributes (action, alt, ariaControls, ariaOrientation, ariaExpanded, ariaHaspopup, ariaHidden, ariaLabelledby, ariaModal, autocomplete, autofocus, cRadius, checked, classN_, class_, classes, clipRule, colSpan, cxCoord, cyCoord, data_, defaultValue, defer, dims, disabled, enctype, fill, fillRule, fontFamily, fontSize, for, gradientUnits, height, href, id_, method, name, noop, offset, placeholder, points, role, rows, rx, selected, spellcheck, src, stroke, strokeLinecap, strokeLinejoin, strokeWidth, style, tabindex, target, title, transform, type_, value, viewBox, width, x1Coord, x2Coord, xCoord, xmlns, y1Coord, y2Coord, yCoord) as Attributes -import DOM.Erumu.HTML.Elements (ElementFn, a, address, aside, br, button, circle, code, dd, div_, dl, dt, em, embed, fieldset, footer, form, graphic, h1, h2, h3,h4, h5, header, hr, i, iframe, img, input, label, legend, li, line, linearGradient, main, nav, noscript, object, ol, option, p, path, polygon, rect, script, section, select, span, stop, strong, svg, table, tbody, td, text, textArea, th, thead, time, tr, ul) as Elements +import DOM.Erumu.HTML.Elements (ElementFn, a, address, aside, br, button, circle, code, dd, div_, dl, dt, em, embed, fieldset, footer, form, graphic, h1, h2, h3, h4, h5, header, hr, i, iframe, img, input, label, legend, li, line, linearGradient, main, nav, noscript, object, ol, option, p, path, polygon, rect, script, section, select, span, stop, strong, svg, table, tbody, td, text, textArea, th, thead, time, tr, ul) as Elements import DOM.Erumu.HTML.Events (clickawayfn, onblur, onclick, onfocus, oninput, onmouseenter, onmouseleave, preventDefaultOnclick, terminalOnclick) as Events diff --git a/src/DOM/Erumu/HTML/Attributes.purs b/src/DOM/Erumu/HTML/Attributes.purs index d368407..82fee31 100644 --- a/src/DOM/Erumu/HTML/Attributes.purs +++ b/src/DOM/Erumu/HTML/Attributes.purs @@ -66,8 +66,7 @@ module DOM.Erumu.HTML.Attributes , y1Coord , y2Coord , yCoord - ) - where + ) where import Prelude import Data.String (joinWith) diff --git a/src/DOM/Erumu/HTML/Decoder.purs b/src/DOM/Erumu/HTML/Decoder.purs index 6a83a44..0f67a91 100644 --- a/src/DOM/Erumu/HTML/Decoder.purs +++ b/src/DOM/Erumu/HTML/Decoder.purs @@ -41,9 +41,8 @@ decodeTarget (Target description toElement) = do case toElement <$> mbNode of (Just (Right elem)) -> pure elem - (Just (Left err)) -> Decode.crash ("decodeTarget: " <> description <> ": " <> err) - Nothing -> Decode.crash ("decodeTarget: no node") - + (Just (Left err)) -> Decode.crash ("decodeTarget: " <> description <> ": " <> err) + Nothing -> Decode.crash ("decodeTarget: no node") -- -- Decoding input elements diff --git a/src/DOM/Erumu/HTML/Elements.purs b/src/DOM/Erumu/HTML/Elements.purs index d5f6ec3..da651be 100644 --- a/src/DOM/Erumu/HTML/Elements.purs +++ b/src/DOM/Erumu/HTML/Elements.purs @@ -1,28 +1,71 @@ module DOM.Erumu.HTML.Elements - ( div_, span, br, hr + ( div_ + , span + , br + , hr , a - , i, strong, code - , h1, h2, h3, h4, h5 + , i + , strong + , code + , h1 + , h2 + , h3 + , h4 + , h5 , p - , dt, dd, dl + , dt + , dd + , dl , footer - , ol, ul, li + , ol + , ul + , li , section - , table, thead, tbody, tr, td, th - - , iframe, object, embed + , table + , thead + , tbody + , tr + , td + , th + + , iframe + , object + , embed , img - , nav, main, header, aside, legend - - , label, form, input, textArea, select, option, button, time, em, fieldset + , nav + , main + , header + , aside + , legend + + , label + , form + , input + , textArea + , select + , option + , button + , time + , em + , fieldset , address - , script ,noscript - - , svg, path, rect, graphic, polygon, circle, line, linearGradient, stop, text + , script + , noscript + + , svg + , path + , rect + , graphic + , polygon + , circle + , line + , linearGradient + , stop + , text , ElementFn ) where diff --git a/src/DOM/Erumu/HTML/Events.purs b/src/DOM/Erumu/HTML/Events.purs index 7929af5..63fc57f 100644 --- a/src/DOM/Erumu/HTML/Events.purs +++ b/src/DOM/Erumu/HTML/Events.purs @@ -10,7 +10,6 @@ module DOM.Erumu.HTML.Events , onmouseleave ) where - import DOM.Erumu.Types (onEvent, onPreventDefaultEvent, onPropagatingEvent, Prop) onclick :: forall msg. msg -> Prop msg diff --git a/src/DOM/Erumu/Maybe.purs b/src/DOM/Erumu/Maybe.purs index 8baab82..87d320b 100644 --- a/src/DOM/Erumu/Maybe.purs +++ b/src/DOM/Erumu/Maybe.purs @@ -15,31 +15,34 @@ import DOM.Erumu.Types (UpdateResult, Return(..), liftUpdate, (!)) -- nested inside a maybe. It takes care of pushing the Maybe back down to the -- model rather than ending up with (Maybe (UpdateResult ...)) -- -update :: forall m msg model. - Applicative m - => (model -> UpdateResult m model msg) - -> Maybe model - -> UpdateResult m (Maybe model) msg +update :: + forall m msg model. + Applicative m => + (model -> UpdateResult m model msg) -> + Maybe model -> + UpdateResult m (Maybe model) msg update f = unwrap <<< updateF (Identity <<< f) -updateF :: forall f m msg model. - Applicative f - => Applicative m - => (model -> f (UpdateResult m model msg)) - -> Maybe model - -> f (UpdateResult m (Maybe model) msg) +updateF :: + forall f m msg model. + Applicative f => + Applicative m => + (model -> f (UpdateResult m model msg)) -> + Maybe model -> + f (UpdateResult m (Maybe model) msg) updateF _ Nothing = pure (Nothing ! []) updateF updateModel (Just model) = liftUpdate identity Just <$> updateModel model -- Conditionally update and lift a Maybe childmodel. -- Continue if the child is Nothing -maybeReturn :: forall m msg model parentModel parentMsg childSignal parentSignal. - Applicative m - => parentModel - -> (Return m model msg childSignal -> Return m parentModel parentMsg parentSignal) - -> (model -> (Return m model msg childSignal)) - -> Maybe model - -> Return m parentModel parentMsg parentSignal +maybeReturn :: + forall m msg model parentModel parentMsg childSignal parentSignal. + Applicative m => + parentModel -> + (Return m model msg childSignal -> Return m parentModel parentMsg parentSignal) -> + (model -> (Return m model msg childSignal)) -> + Maybe model -> + Return m parentModel parentMsg parentSignal maybeReturn parentModel liftFn updateFn model = maybe (Continue $ parentModel ! []) (liftFn <<< updateFn) model diff --git a/src/DOM/Erumu/Nodes.purs b/src/DOM/Erumu/Nodes.purs index 071a9e3..4f32da2 100644 --- a/src/DOM/Erumu/Nodes.purs +++ b/src/DOM/Erumu/Nodes.purs @@ -25,17 +25,17 @@ toSelectElement = unsafeToElement "SELECT" toTextAreaElement :: Node -> Either String HTMLTextAreaElement toTextAreaElement = unsafeToElement "TEXTAREA" -unsafeToElement :: forall elem. - String - -> Node - -> Either String elem +unsafeToElement :: + forall elem. + String -> + Node -> + Either String elem unsafeToElement expectedTagName node = lmap (annotateError expectedTagName) $ do genericElem <- nodeToElement node - if tagName genericElem == expectedTagName - then Right $ unsafeCoerce genericElem - else Left $ tagName genericElem + if tagName genericElem == expectedTagName then Right $ unsafeCoerce genericElem + else Left $ tagName genericElem annotateError :: String -> String -> String annotateError tagName err = @@ -43,7 +43,6 @@ annotateError tagName err = nodeToElement :: Node -> Either String Element nodeToElement node = - if fromEnum ElementNode == nodeTypeIndex node - then Right $ unsafeCoerce node - else Left $ "Node type: " <> show (nodeTypeIndex node) + if fromEnum ElementNode == nodeTypeIndex node then Right $ unsafeCoerce node + else Left $ "Node type: " <> show (nodeTypeIndex node) diff --git a/src/DOM/Erumu/RunWithCommands.purs b/src/DOM/Erumu/RunWithCommands.purs index e7236bf..2f96916 100644 --- a/src/DOM/Erumu/RunWithCommands.purs +++ b/src/DOM/Erumu/RunWithCommands.purs @@ -51,12 +51,13 @@ instance runWithCommandsParallel :: Par.Parallel f m => Par.Parallel (RunWithCom sequential = hoist Par.sequential Par.parallel parallel = hoist Par.parallel Par.sequential -hoist :: forall m f msg model. - Functor f - => (m ~> f) - -> (f ~> m) - -> RunWithCommands m msg model - -> RunWithCommands f msg model +hoist :: + forall m f msg model. + Functor f => + (m ~> f) -> + (f ~> m) -> + RunWithCommands m msg model -> + RunWithCommands f msg model hoist toF toM (RunWithCommands action) = let hoistUpdate update = @@ -67,43 +68,48 @@ hoist toF toM (RunWithCommands action) = RunWithCommands $ map hoistUpdate (toF action) -run :: forall m msg model. - RunWithCommands m msg model - -> m (UpdateResult m model msg) +run :: + forall m msg model. + RunWithCommands m msg model -> + m (UpdateResult m model msg) run (RunWithCommands action) = action -lift :: forall m msg model. - Applicative m - => m model - -> RunWithCommands m msg model +lift :: + forall m msg model. + Applicative m => + m model -> + RunWithCommands m msg model lift monad = RunWithCommands $ - map ({model: _, command: nocmd}) monad - -mapModel :: forall msg m modelA modelB. - Functor m - => (modelA -> modelB) - -> RunWithCommands m msg modelA - -> RunWithCommands m msg modelB + map ({ model: _, command: nocmd }) monad + +mapModel :: + forall msg m modelA modelB. + Functor m => + (modelA -> modelB) -> + RunWithCommands m msg modelA -> + RunWithCommands m msg modelB mapModel f (RunWithCommands action) = RunWithCommands $ map (Types.mapModel f) action -applyModel :: forall msg m modelA modelB. - Apply m - => RunWithCommands m msg (modelA -> modelB) - -> RunWithCommands m msg modelA - -> RunWithCommands m msg modelB +applyModel :: + forall msg m modelA modelB. + Apply m => + RunWithCommands m msg (modelA -> modelB) -> + RunWithCommands m msg modelA -> + RunWithCommands m msg modelB applyModel (RunWithCommands actionF) (RunWithCommands actionA) = RunWithCommands $ map Types.applyModel actionF <*> actionA -bindModel :: forall msg m modelA modelB. - Monad m - => RunWithCommands m msg modelA - -> (modelA -> RunWithCommands m msg modelB) - -> RunWithCommands m msg modelB +bindModel :: + forall msg m modelA modelB. + Monad m => + RunWithCommands m msg modelA -> + (modelA -> RunWithCommands m msg modelB) -> + RunWithCommands m msg modelB bindModel (RunWithCommands actionA) f = RunWithCommands $ do updateA <- actionA @@ -113,33 +119,37 @@ bindModel (RunWithCommands actionA) f = , command: updateA.command <> updateB.command } -pureModel :: forall msg m model. - Applicative m - => model - -> RunWithCommands m msg model +pureModel :: + forall msg m model. + Applicative m => + model -> + RunWithCommands m msg model pureModel model = RunWithCommands (pure $ model ! []) -mapMsg :: forall msgA msgB m model. - Functor m - => (msgA -> msgB) - -> RunWithCommands m msgA model - -> RunWithCommands m msgB model +mapMsg :: + forall msgA msgB m model. + Functor m => + (msgA -> msgB) -> + RunWithCommands m msgA model -> + RunWithCommands m msgB model mapMsg f (RunWithCommands action) = RunWithCommands $ map (Types.mapMsg f) action -fromUpdate :: forall m msg model. - Applicative m - => UpdateResult m model msg - -> RunWithCommands m msg model +fromUpdate :: + forall m msg model. + Applicative m => + UpdateResult m model msg -> + RunWithCommands m msg model fromUpdate update = RunWithCommands (pure update) -fromCommand :: forall m msg. - Applicative m - => Command m msg - -> RunWithCommands m msg Unit +fromCommand :: + forall m msg. + Applicative m => + Command m msg -> + RunWithCommands m msg Unit fromCommand command = RunWithCommands $ pure @@ -147,19 +157,21 @@ fromCommand command = , command: command } -toCommand :: forall m msg. - Bind m - => RunWithCommands m msg Unit - -> Command m msg +toCommand :: + forall m msg. + Bind m => + RunWithCommands m msg Unit -> + Command m msg toCommand (RunWithCommands action) = dispatchCmd $ \dispatch -> do update <- action runCommand dispatch update.command -sendMsg :: forall m msg. - Applicative m - => msg - -> RunWithCommands m msg Unit +sendMsg :: + forall m msg. + Applicative m => + msg -> + RunWithCommands m msg Unit sendMsg msg = fromCommand (dispatchCmd (\dispatch -> dispatch msg)) @@ -170,12 +182,13 @@ sendMsg msg = run so that any messages dispatched by the child will be delivery after the load message has been delivered to the parent. -} -loadChildViaMsg :: forall m parentMsg childMsg childModel. - Monad m - => (childModel -> parentMsg) - -> (childMsg -> parentMsg) - -> RunWithCommands m childMsg childModel - -> RunWithCommands m parentMsg Unit +loadChildViaMsg :: + forall m parentMsg childMsg childModel. + Monad m => + (childModel -> parentMsg) -> + (childMsg -> parentMsg) -> + RunWithCommands m childMsg childModel -> + RunWithCommands m parentMsg Unit loadChildViaMsg mkLoadMsg wrapChildMsg (RunWithCommands action) = RunWithCommands $ do childUpdate <- action diff --git a/src/DOM/Erumu/Types.purs b/src/DOM/Erumu/Types.purs index 63c7d6c..9483c2c 100644 --- a/src/DOM/Erumu/Types.purs +++ b/src/DOM/Erumu/Types.purs @@ -1,7 +1,10 @@ module DOM.Erumu.Types - ( UpdateFn, UpdateResult, Return(..) + ( UpdateFn + , UpdateResult + , Return(..) - , withCommands, (!) + , withCommands + , (!) , addCommands , mapModel , applyModel @@ -19,13 +22,24 @@ module DOM.Erumu.Types , SignalCommandEmitter , emitNone - , HTML, element, text, texts, noElement, toVTree - - , Prop, attribute , hookProp - , onEvent, onPreventDefaultEvent, onPropagatingEvent - , onEventDecode, onEventMaybeDecode - - , Command, DispatchFn + , HTML + , element + , text + , texts + , noElement + , toVTree + + , Prop + , attribute + , hookProp + , onEvent + , onPreventDefaultEvent + , onPropagatingEvent + , onEventDecode + , onEventMaybeDecode + + , Command + , DispatchFn , dispatchCmd , runCommand , nocmd @@ -71,11 +85,12 @@ instance functorReturn :: Functor (Return m model msg) where map f (Return signal update) = Return (f signal) update map _f (Continue update) = Continue update -withCommands :: forall m model msg. - Applicative m - => model - -> Array (Command m msg) - -> UpdateResult m model msg +withCommands :: + forall m model msg. + Applicative m => + model -> + Array (Command m msg) -> + UpdateResult m model msg withCommands model commands = { model: model, command: fold commands } -- Precedence set to 5 to allow combinations with $ like @@ -85,51 +100,57 @@ withCommands model commands = { model: model, command: fold commands } -- To work as desired infixl 5 withCommands as ! -addCommands :: forall m model msg. - Applicative m - => UpdateResult m model msg - -> Array (Command m msg) - -> UpdateResult m model msg +addCommands :: + forall m model msg. + Applicative m => + UpdateResult m model msg -> + Array (Command m msg) -> + UpdateResult m model msg addCommands update commands = { model: update.model , command: update.command <> fold commands } -mapModel :: forall m msg modelA modelB. - (modelA -> modelB) - -> UpdateResult m modelA msg - -> UpdateResult m modelB msg +mapModel :: + forall m msg modelA modelB. + (modelA -> modelB) -> + UpdateResult m modelA msg -> + UpdateResult m modelB msg mapModel f update = update { model = f update.model } -applyModel :: forall m msg modelA modelB. - Apply m - => UpdateResult m (modelA -> modelB) msg - -> UpdateResult m modelA msg - -> UpdateResult m modelB msg +applyModel :: + forall m msg modelA modelB. + Apply m => + UpdateResult m (modelA -> modelB) msg -> + UpdateResult m modelA msg -> + UpdateResult m modelB msg applyModel updateF updateA = { model: updateF.model updateA.model , command: updateF.command <> updateA.command } -mapMsg :: forall m model msgA msgB. - (msgA -> msgB) - -> UpdateResult m model msgA - -> UpdateResult m model msgB +mapMsg :: + forall m model msgA msgB. + (msgA -> msgB) -> + UpdateResult m model msgA -> + UpdateResult m model msgB mapMsg f update = update { command = f <$> update.command } -liftUpdate :: forall m model msg parentModel parentMsg. - (msg -> parentMsg) - -> (model -> parentModel) - -> (UpdateResult m model msg) - -> (UpdateResult m parentModel parentMsg) +liftUpdate :: + forall m model msg parentModel parentMsg. + (msg -> parentMsg) -> + (model -> parentModel) -> + (UpdateResult m model msg) -> + (UpdateResult m parentModel parentMsg) liftUpdate liftMsg liftModel update = liftUpdateAnd liftMsg liftModel identity update -liftUpdateAnd :: forall m model msg parentModel parentMsg. - (msg -> parentMsg) - -> (model -> parentModel) - -> (parentModel -> parentModel) - -> (UpdateResult m model msg) - -> (UpdateResult m parentModel parentMsg) +liftUpdateAnd :: + forall m model msg parentModel parentMsg. + (msg -> parentMsg) -> + (model -> parentModel) -> + (parentModel -> parentModel) -> + (UpdateResult m model msg) -> + (UpdateResult m parentModel parentMsg) liftUpdateAnd liftMsg liftModel modelFn update = { model: modelFn $ liftModel update.model , command: liftMsg <$> update.command @@ -141,80 +162,87 @@ emitNone :: forall m signal msg. SignalCommandEmitter m signal msg emitNone = const [] -- For parents just lifting the Return (of the same signal type) as their child -liftReturn :: forall m model msg signal parentMsg parentModel. - Applicative m - => (msg -> parentMsg) - -> (model -> parentModel) - -> Return m model msg signal - -> Return m parentModel parentMsg signal +liftReturn :: + forall m model msg signal parentMsg parentModel. + Applicative m => + (msg -> parentMsg) -> + (model -> parentModel) -> + Return m model msg signal -> + Return m parentModel parentMsg signal liftReturn liftMsg liftModel = liftReturnAnd liftMsg liftModel (const identity) emitNone -- For parents who want side effects of processing the Return (of the same signal type).. -- This applies the "and" `modifyModel` and `emitter` functions only on `Return` cases -liftReturnAnd :: forall m model msg signal parentMsg parentModel. - Applicative m - => (msg -> parentMsg) - -> (model -> parentModel) - -> (signal -> parentModel -> parentModel) - -> SignalCommandEmitter m signal parentMsg - -> Return m model msg signal - -> Return m parentModel parentMsg signal +liftReturnAnd :: + forall m model msg signal parentMsg parentModel. + Applicative m => + (msg -> parentMsg) -> + (model -> parentModel) -> + (signal -> parentModel -> parentModel) -> + SignalCommandEmitter m signal parentMsg -> + Return m model msg signal -> + Return m parentModel parentMsg signal liftReturnAnd liftMsg liftModel modifyModel emitter r = case r of - Continue upd -> Continue $ liftUpdate liftMsg liftModel upd - Return v upd -> Return v $ addCommands (liftUpdateAnd liftMsg liftModel (modifyModel v) upd) (emitter v) + Continue upd -> Continue $ liftUpdate liftMsg liftModel upd + Return v upd -> Return v $ addCommands (liftUpdateAnd liftMsg liftModel (modifyModel v) upd) (emitter v) -updateResultFrom :: forall m signal model msg. - Return m model msg signal - -> UpdateResult m model msg +updateResultFrom :: + forall m signal model msg. + Return m model msg signal -> + UpdateResult m model msg updateResultFrom r = case r of Return _val upd -> upd - Continue upd -> upd + Continue upd -> upd -- for parents that quietly return UpdateResult m -liftIntoUpdate :: forall m model msg signal parentMsg parentModel. - Applicative m - => (msg -> parentMsg) - -> (model -> parentModel) - -> Return m model msg signal - -> UpdateResult m parentModel parentMsg +liftIntoUpdate :: + forall m model msg signal parentMsg parentModel. + Applicative m => + (msg -> parentMsg) -> + (model -> parentModel) -> + Return m model msg signal -> + UpdateResult m parentModel parentMsg liftIntoUpdate liftMsg liftModel = liftIntoUpdateAnd liftMsg liftModel (const identity) emitNone -- for parents that return UpdateResult m with side effects -liftIntoUpdateAnd :: forall m model msg signal parentMsg parentModel. - Applicative m - => (msg -> parentMsg) - -> (model -> parentModel) - -> (signal -> parentModel -> parentModel) - -> SignalCommandEmitter m signal parentMsg - -> Return m model msg signal - -> UpdateResult m parentModel parentMsg +liftIntoUpdateAnd :: + forall m model msg signal parentMsg parentModel. + Applicative m => + (msg -> parentMsg) -> + (model -> parentModel) -> + (signal -> parentModel -> parentModel) -> + SignalCommandEmitter m signal parentMsg -> + Return m model msg signal -> + UpdateResult m parentModel parentMsg liftIntoUpdateAnd liftMsg liftModel modifyModel emitter r = updateResultFrom $ liftReturnAnd liftMsg liftModel modifyModel emitter r -- For parents that want to ignore their child's Return and quietly lift-then-Continue -liftContinue :: forall m model msg childSignal parentSignal parentMsg parentModel. - Applicative m - => (msg -> parentMsg) - -> (model -> parentModel) - -> Return m model msg childSignal - -> Return m parentModel parentMsg parentSignal +liftContinue :: + forall m model msg childSignal parentSignal parentMsg parentModel. + Applicative m => + (msg -> parentMsg) -> + (model -> parentModel) -> + Return m model msg childSignal -> + Return m parentModel parentMsg parentSignal liftContinue liftMsg liftModel r = Continue $ liftIntoUpdate liftMsg liftModel r -- For parents that want to quietly lift-then-Continue after processing the child Return -liftContinueAnd :: forall m model msg childSignal parentSignal parentMsg parentModel. - Applicative m - => (msg -> parentMsg) - -> (model -> parentModel) - -> (childSignal -> parentModel -> parentModel) - -> SignalCommandEmitter m childSignal parentMsg - -> Return m model msg childSignal - -> Return m parentModel parentMsg parentSignal +liftContinueAnd :: + forall m model msg childSignal parentSignal parentMsg parentModel. + Applicative m => + (msg -> parentMsg) -> + (model -> parentModel) -> + (childSignal -> parentModel -> parentModel) -> + SignalCommandEmitter m childSignal parentMsg -> + Return m model msg childSignal -> + Return m parentModel parentMsg parentSignal liftContinueAnd liftMsg liftModel modifyModel emitter r = Continue $ liftIntoUpdateAnd liftMsg liftModel modifyModel emitter r @@ -225,12 +253,13 @@ runCommand :: forall m msg. DispatchFn m msg -> Command m msg -> m Unit runCommand f (Command c) = c f -hoistCommand :: forall m f msg. - (m ~> f) - -> (f ~> m) - -> Command m msg - -> Command f msg -hoistCommand toF toM (Command mAction)= +hoistCommand :: + forall m f msg. + (m ~> f) -> + (f ~> m) -> + Command m msg -> + Command f msg +hoistCommand toF toM (Command mAction) = Command $ \dispatchF -> toF (mAction (toM <<< dispatchF)) @@ -254,21 +283,22 @@ runAndSend action = Command (action >>= _) sendMsg :: forall m msg. msg -> Command m msg sendMsg msg = Command (\dispatch -> dispatch msg) -seqCmd :: forall m msg. - Apply m - => Command m msg - -> Command m msg - -> Command m msg +seqCmd :: + forall m msg. + Apply m => + Command m msg -> + Command m msg -> + Command m msg seqCmd (Command c1) (Command c2) = - Command c1c2 + Command c1c2 where - c1c2 d = c1 d *> c2 d - + c1c2 d = c1 d *> c2 d -mapCmd :: forall m msg1 msg2. - (msg1 -> msg2) - -> Command m msg1 - -> Command m msg2 +mapCmd :: + forall m msg1 msg2. + (msg1 -> msg2) -> + Command m msg1 -> + Command m msg2 mapCmd f (Command action) = Command (action <<< liftMessages f) instance functorCommand :: Functor (Command m) where @@ -280,17 +310,19 @@ instance semigroupCommand :: Apply m => Semigroup (Command m msg) where instance monoidCommand :: Applicative m => Monoid (Command m msg) where mempty = nocmd -liftMessages :: forall m msg1 msg2. - (msg1 -> msg2) - -> DispatchFn m msg2 - -> DispatchFn m msg1 +liftMessages :: + forall m msg1 msg2. + (msg1 -> msg2) -> + DispatchFn m msg2 -> + DispatchFn m msg1 liftMessages f g = g <<< f -runcmd :: forall m msg. - Command m msg - -> DispatchFn m msg - -> (m Unit -> Effect Unit) - -> Effect Unit +runcmd :: + forall m msg. + Command m msg -> + DispatchFn m msg -> + (m Unit -> Effect Unit) -> + Effect Unit runcmd (Command f) dispatch execEff = execEff (f dispatch) instance functorProp :: Functor Prop where @@ -310,13 +342,15 @@ noElement = text "" element :: forall msg. String -> Array (Prop msg) -> Array (HTML msg) -> HTML msg element name props dsls = - HTML render + HTML render where - render :: DispatchFnEff msg -> VTree - render disp = - let attrs = map (toAttrR disp) props - trees = map (toVTreeR disp) dsls - in node name attrs trees + render :: DispatchFnEff msg -> VTree + render disp = + let + attrs = map (toAttrR disp) props + trees = map (toVTreeR disp) dsls + in + node name attrs trees hookProp :: forall msg. String -> Virtual.Hook -> Prop msg hookProp name hook = @@ -325,28 +359,28 @@ hookProp name hook = onEvent :: forall msg. String -> msg -> Prop msg onEvent name dat = handlerProp name - nonPropagatingEventHandler - (\_ -> pure [dat]) + nonPropagatingEventHandler + (\_ -> pure [ dat ]) onPropagatingEvent :: forall msg. String -> msg -> Prop msg onPropagatingEvent name dat = handlerProp name - propagatingEventHandler - (\_ -> pure [dat]) + propagatingEventHandler + (\_ -> pure [ dat ]) onPreventDefaultEvent :: forall msg. String -> msg -> Prop msg onPreventDefaultEvent name dat = handlerProp name - preventDefaultEventHandler - (\_ -> pure [dat]) + preventDefaultEventHandler + (\_ -> pure [ dat ]) -- This event decode property is used to declare a handler that will *always* process the event into -- a Schmods Msg to dispatch, and stops propagation to potential parent handlers. onEventDecode :: forall msg. String -> (Decode msg) -> Prop msg onEventDecode name decode = handlerProp name - nonPropagatingEventHandler - (\event -> pure <$> Decode.runOrCrash decode event) + nonPropagatingEventHandler + (\event -> pure <$> Decode.runOrCrash decode event) -- This event decode property is used to declare a handler that *may or may not* process the event -- and it may (will) propagate the event to potential parent handlers. @@ -355,19 +389,22 @@ onEventDecode name decode = onEventMaybeDecode :: forall msg. String -> (Decode (Maybe msg)) -> Prop msg onEventMaybeDecode name decode = handlerProp name - propagatingEventHandler - (\event -> catMaybes <$> pure <$> Decode.runOrCrash decode event ) - -handlerProp :: forall msg. - String - -> EventHandler - -> (Event -> Effect (Array msg)) - -> Prop msg + propagatingEventHandler + (\event -> catMaybes <$> pure <$> Decode.runOrCrash decode event) + +handlerProp :: + forall msg. + String -> + EventHandler -> + (Event -> Effect (Array msg)) -> + Prop msg handlerProp name handler mkMsg = - Prop (\dispatch -> - { key: name - , value: handler (traverse_ dispatch <=< mkMsg) - }) + Prop + ( \dispatch -> + { key: name + , value: handler (traverse_ dispatch <=< mkMsg) + } + ) attribute :: forall msg. String -> String -> Prop msg attribute name value = diff --git a/src/DOM/Erumu/Widget/CheckboxInput.purs b/src/DOM/Erumu/Widget/CheckboxInput.purs index d5db951..2216ef0 100644 --- a/src/DOM/Erumu/Widget/CheckboxInput.purs +++ b/src/DOM/Erumu/Widget/CheckboxInput.purs @@ -30,21 +30,29 @@ isChecked (Model b) = b render :: Model -> HTML Msg render = renderWith identity [] -renderWith :: forall msg. - (Msg -> msg) - -> Array (Prop msg) - -> Model - -> HTML msg +renderWith :: + forall msg. + (Msg -> msg) -> + Array (Prop msg) -> + Model -> + HTML msg renderWith liftMsg userProps (Model m) = - let checkedProp = if m - then [ checked "checked" ] - else [] - ourProps = [ type_ "checkbox" - , onEventDecode "onclick" (liftMsg <<< Msg <$> inputChecked) - ] - - in input (checkedProp <> ourProps <> userProps) [] - -update :: forall m. Applicative m - => Msg -> Model -> UpdateResult m Model Msg + let + checkedProp = + if m then [ checked "checked" ] + else [] + ourProps = + [ type_ "checkbox" + , onEventDecode "onclick" (liftMsg <<< Msg <$> inputChecked) + ] + + in + input (checkedProp <> ourProps <> userProps) [] + +update :: + forall m. + Applicative m => + Msg -> + Model -> + UpdateResult m Model Msg update (Msg checked) _ = Model checked ! [] diff --git a/src/DOM/Erumu/Widget/FileInput.purs b/src/DOM/Erumu/Widget/FileInput.purs index fd6ce9b..e207b76 100644 --- a/src/DOM/Erumu/Widget/FileInput.purs +++ b/src/DOM/Erumu/Widget/FileInput.purs @@ -36,17 +36,25 @@ setValue fl' = const $ withValue fl' render :: Array (Prop Msg) -> Model -> HTML Msg render = renderWith identity -renderWith :: forall msg. - (Msg -> msg) - -> Array (Prop msg) - -> Model - -> HTML msg +renderWith :: + forall msg. + (Msg -> msg) -> + Array (Prop msg) -> + Model -> + HTML msg renderWith liftMsg userProps _ = - let ourProps = [ type_ "file" - , onEventDecode "onchange" (liftMsg <<< NewInput <$> inputFiles) - ] - in input (ourProps <> userProps) [] - -update :: forall m. Applicative m - => Msg -> Model -> UpdateResult m Model Msg + let + ourProps = + [ type_ "file" + , onEventDecode "onchange" (liftMsg <<< NewInput <$> inputFiles) + ] + in + input (ourProps <> userProps) [] + +update :: + forall m. + Applicative m => + Msg -> + Model -> + UpdateResult m Model Msg update (NewInput newValue) _ = Model newValue ! [] diff --git a/src/DOM/Erumu/Widget/RadioInputGroup.purs b/src/DOM/Erumu/Widget/RadioInputGroup.purs index 336c17a..3ca20a5 100644 --- a/src/DOM/Erumu/Widget/RadioInputGroup.purs +++ b/src/DOM/Erumu/Widget/RadioInputGroup.purs @@ -3,7 +3,8 @@ module DOM.Erumu.Widget.RadioInputGroup , Msg , empty , selected - , fill, fillBy + , fill + , fillBy , itemEventProp , render , renderOne @@ -19,9 +20,9 @@ import DOM.Erumu.HTML.Decoder (inputChecked) import DOM.Erumu.Types (HTML, Prop, onEventDecode, noElement, text) newtype Model a = Model - { groupName :: String - , label :: a -> String - , items :: Array a + { groupName :: String + , label :: a -> String + , items :: Array a , selectedIndex :: Int } @@ -60,24 +61,27 @@ renderOneWith userProps model@(Model m) idx = case m.items !! idx of Nothing -> noElement Just item -> div_ userProps - [ renderItem model idx item ] + [ renderItem model idx item ] renderItem :: forall a. Model a -> Int -> a -> HTML Msg renderItem (Model m) idx item = - let checkedProp = if m.selectedIndex == idx - then [ checked "checked" ] - else [] - ourProps = [ type_ "radio" - , itemEventProp idx - , name m.groupName - ] - - in div_ [] - [ input (checkedProp <> ourProps) [] - , label [ class_ "margin-5-left margin-3-above"] - [ text $ m.label item - ] - ] + let + checkedProp = + if m.selectedIndex == idx then [ checked "checked" ] + else [] + ourProps = + [ type_ "radio" + , itemEventProp idx + , name m.groupName + ] + + in + div_ [] + [ input (checkedProp <> ourProps) [] + , label [ class_ "margin-5-left margin-3-above" ] + [ text $ m.label item + ] + ] itemEventProp :: Int -> Prop Msg itemEventProp idx = onEventDecode "onclick" (Selected idx <$> inputChecked) diff --git a/src/DOM/Erumu/Widget/Select.purs b/src/DOM/Erumu/Widget/Select.purs index f7fd914..f7cbf81 100644 --- a/src/DOM/Erumu/Widget/Select.purs +++ b/src/DOM/Erumu/Widget/Select.purs @@ -48,8 +48,7 @@ fillAddIfMissing :: forall a. Eq a => a -> Model a -> Model a fillAddIfMissing a originalModel@(Model m) = let modelWithItem = - if elem a m.items - then originalModel + if elem a m.items then originalModel else Model m { items = snoc m.items a } in fill a modelWithItem @@ -75,26 +74,32 @@ setDisabled disabled (Model m) = render :: forall a. Array (Prop Msg) -> Model a -> HTML Msg render userProps model@(Model m) = - let items = mapWithIndex (renderItem model) m.items - in select (staticProps <> disabledProp m.disabled <> userProps) items + let + items = mapWithIndex (renderItem model) m.items + in + select (staticProps <> disabledProp m.disabled <> userProps) items staticProps :: Array (Prop Msg) -staticProps = [onEventDecode "onchange" (Selected <$> selectedIndex)] +staticProps = [ onEventDecode "onchange" (Selected <$> selectedIndex) ] renderItem :: forall a. Model a -> Int -> a -> HTML Msg renderItem (Model m) idx item = - let attrs = if m.selectedIndex == idx - then [HTML.selected "selected"] - else [] + let + attrs = + if m.selectedIndex == idx then [ HTML.selected "selected" ] + else [] - in option attrs [text $ fromMaybe "Select..." $ m.showFn item] + in + option attrs [ text $ fromMaybe "Select..." $ m.showFn item ] update :: forall a m. Applicative m => Msg -> Model a -> Return m (Model a) Msg a update (Selected idx) (Model m) = - let newModel = Model m { selectedIndex = idx } - in case selected newModel of - Just v -> Return v $ newModel ! [] - Nothing -> Continue $ newModel ! [] + let + newModel = Model m { selectedIndex = idx } + in + case selected newModel of + Just v -> Return v $ newModel ! [] + Nothing -> Continue $ newModel ! [] itemList :: forall a. Model a -> Array a itemList (Model m) = m.items diff --git a/src/DOM/Erumu/Widget/TextArea.purs b/src/DOM/Erumu/Widget/TextArea.purs index 5485d77..3992dc6 100644 --- a/src/DOM/Erumu/Widget/TextArea.purs +++ b/src/DOM/Erumu/Widget/TextArea.purs @@ -35,17 +35,25 @@ setValue s = const $ withValue s render :: Array (Prop Msg) -> Model -> HTML Msg render = renderWith identity -renderWith :: forall msg. - (Msg -> msg) - -> Array (Prop msg) - -> Model - -> HTML msg +renderWith :: + forall msg. + (Msg -> msg) -> + Array (Prop msg) -> + Model -> + HTML msg renderWith liftMsg userProps (Model currentValue) = - let ourProps = [ onEventDecode "oninput" (liftMsg <<< NewInput <$> textAreaValue) - , HTML.defaultValue currentValue - ] - in textArea (ourProps <> userProps) [] - -update :: forall m. Applicative m - => Msg -> Model -> UpdateResult m Model Msg + let + ourProps = + [ onEventDecode "oninput" (liftMsg <<< NewInput <$> textAreaValue) + , HTML.defaultValue currentValue + ] + in + textArea (ourProps <> userProps) [] + +update :: + forall m. + Applicative m => + Msg -> + Model -> + UpdateResult m Model Msg update (NewInput newValue) _ = Model newValue ! [] diff --git a/src/DOM/Erumu/Widget/TextInput.purs b/src/DOM/Erumu/Widget/TextInput.purs index 62a7b94..9653d0b 100644 --- a/src/DOM/Erumu/Widget/TextInput.purs +++ b/src/DOM/Erumu/Widget/TextInput.purs @@ -25,7 +25,7 @@ newtype Model = Model Fields type Fields = { currentValue :: String - , disabled :: Boolean + , disabled :: Boolean } newtype Msg = NewInput String @@ -35,9 +35,10 @@ empty = withValue "" withValue :: String -> Model withValue s = - Model { currentValue: s - , disabled: false - } + Model + { currentValue: s + , disabled: false + } value :: Model -> String value (Model m) = m.currentValue @@ -59,19 +60,26 @@ disabled (Model m) = m.disabled render :: Array (Prop Msg) -> Model -> HTML Msg render = renderWith identity -renderWith :: forall msg. - (Msg -> msg) - -> Array (Prop msg) - -> Model - -> HTML msg +renderWith :: + forall msg. + (Msg -> msg) -> + Array (Prop msg) -> + Model -> + HTML msg renderWith liftMsg userProps (Model m) = - let ourProps = - [ type_ "text" - , onEventDecode "oninput" (liftMsg <<< NewInput <$> inputValue) - , HTML.value m.currentValue - ] - in input (ourProps <> disabledProp m.disabled <> userProps) [] + let + ourProps = + [ type_ "text" + , onEventDecode "oninput" (liftMsg <<< NewInput <$> inputValue) + , HTML.value m.currentValue + ] + in + input (ourProps <> disabledProp m.disabled <> userProps) [] -update :: forall m. Applicative m - => Msg -> Model -> UpdateResult m Model Msg +update :: + forall m. + Applicative m => + Msg -> + Model -> + UpdateResult m Model Msg update (NewInput newValue) model = setValue newValue model ! [] diff --git a/src/DOM/Virtual.purs b/src/DOM/Virtual.purs index 7958b21..4aef3a8 100644 --- a/src/DOM/Virtual.purs +++ b/src/DOM/Virtual.purs @@ -1,14 +1,27 @@ module DOM.Virtual - ( Value, VTree, Diff, Attribute, Attributes - , createElement, patchElement, diff - - , node, text - - , Hook, HookFunctions, newHook, hookValue + ( Value + , VTree + , Diff + , Attribute + , Attributes + , createElement + , patchElement + , diff + + , node + , text + + , Hook + , HookFunctions + , newHook + , hookValue , EventHandler - , unsafeValue, stringValue - , preventDefaultEventHandler, propagatingEventHandler, nonPropagatingEventHandler + , unsafeValue + , stringValue + , preventDefaultEventHandler + , propagatingEventHandler + , nonPropagatingEventHandler ) where import Prelude @@ -28,10 +41,11 @@ foreign import diff :: VTree -> VTree -> Diff foreign import createElement :: VTree -> Effect Element foreign import patchElement :: Diff -> Element -> Effect Element -foreign import node :: String - -> Attributes - -> Array VTree - -> VTree +foreign import node :: + String -> + Attributes -> + Array VTree -> + VTree foreign import text :: String -> VTree diff --git a/src/DOM/Virtual/App.purs b/src/DOM/Virtual/App.purs index e2bcab9..27378f1 100644 --- a/src/DOM/Virtual/App.purs +++ b/src/DOM/Virtual/App.purs @@ -30,15 +30,17 @@ type DOMState = newtype App = App (Ref DOMState) -newApp :: VTree - -> Effect App +newApp :: + VTree -> + Effect App newApp tree = do elem <- createAppElement tree App <$> createDomRef tree elem -mountApp :: String - -> App - -> Effect (Maybe String) +mountApp :: + String -> + App -> + Effect (Maybe String) mountApp containerId (App domRef) = do win <- window htmlDoc <- document win @@ -49,21 +51,22 @@ mountApp containerId (App domRef) = do case container of Nothing -> pure $ Just $ joinWith " " - [ "DOM.Virtual.App.mountApp:" - , "Couldn't find element with id" - , containerId - , "to mount the app." - ] + [ "DOM.Virtual.App.mountApp:" + , "Couldn't find element with id" + , containerId + , "to mount the app." + ] Just containerElem -> do domState <- Ref.read domRef _ <- appendChild (toNode domState.elem) - (toNode containerElem) + (toNode containerElem) pure Nothing -rerenderApp :: VTree - -> App - -> Effect Unit +rerenderApp :: + VTree -> + App -> + Effect Unit rerenderApp newTree (App domRef) = do domState <- Ref.read domRef @@ -75,12 +78,14 @@ rerenderApp newTree (App domRef) = do -- Function aliases to sort out the effect types -- of the actors involved -- -createAppElement :: VTree - -> Effect Element +createAppElement :: + VTree -> + Effect Element createAppElement tree = createElement tree -createDomRef :: VTree - -> Element - -> Effect (Ref DOMState) +createDomRef :: + VTree -> + Element -> + Effect (Ref DOMState) createDomRef tree elem = Ref.new { tree, elem }