From 2253c59e4b6919c76daa630f72bea7d9358ff371 Mon Sep 17 00:00:00 2001 From: anotherhale Date: Sun, 10 Jan 2021 14:46:53 -0800 Subject: [PATCH 1/6] Added support for Cancel event and update for Elm 19.1 --- elm.json | 2 +- src/Modal.elm | 2032 +++++++++++++++++++++++++------------------------ 2 files changed, 1044 insertions(+), 990 deletions(-) diff --git a/elm.json b/elm.json index 83033b8..6bfd09a 100644 --- a/elm.json +++ b/elm.json @@ -16,7 +16,7 @@ "elm/json": "1.0.0 <= v < 2.0.0", "elm/virtual-dom": "1.0.2 <= v < 2.0.0", "iodevs/elm-history": "1.0.0 <= v < 2.0.0", - "rtfeldman/elm-css": "15.1.0 <= v < 16.0.0" + "rtfeldman/elm-css": "16.0.0 <= v < 17.0.0" }, "test-dependencies": {} } diff --git a/src/Modal.elm b/src/Modal.elm index f1905f6..b864b39 100755 --- a/src/Modal.elm +++ b/src/Modal.elm @@ -1,989 +1,1043 @@ -module Modal exposing - ( ClosingAnimation(..) - , ClosingEffect(..) - , Config - , Model - , Msg - , OpenedAnimation(..) - , OpeningAnimation(..) - , animationEnd - , closeModal - , cmdGetWindowSize - , initModel - , newConfig - , openModal - , setBody - , setBodyBorderRadius - , setBodyCss - , setBodyFromTop - , setBodyHeight - , setBodyWidth - , setClosingAnimation - , setClosingEffect - , setFooter - , setFooterCss - , setHeader - , setHeaderCss - , setOpenedAnimation - , setOpeningAnimation - , subscriptions - , update - , view - ) - -import Browser.Dom exposing (getViewport) -import Browser.Events exposing (onResize) -import Css -import Css.Animations as Animations -import Css.Transitions as Transitions -import History exposing (History(..), create, current, forward, rewind, rewindAll) -import Html -import Html.Styled as Styled exposing (..) -import Html.Styled.Attributes exposing (..) -import Html.Styled.Events exposing (on, onClick) -import Json.Decode as Json -import Task -import VirtualDom - - - --- Model - - -type Model msg - = Opening (Config msg) - | Opened (Config msg) - | Closing (Config msg) - | Closed - - -initModel : Model msg -initModel = - Closed - - - --- Commands - - -cmdGetWindowSize : Cmd (Msg msg) -cmdGetWindowSize = - Task.perform - (\{ viewport } -> - GetWindowSize viewport.width viewport.height - ) - getViewport - - - --- Subscriptions - - -subscriptions : Sub (Msg msg) -subscriptions = - onResize (\x y -> GetWindowSize (toFloat x) (toFloat y)) - - - --- Msg - - -type Msg msg - = OpenModal (Config msg) - | CloseModal - | AnimationEnd - | GetWindowSize Float Float - - - --- Types - - -type OpeningAnimation - = FromTop - | FromRight - | FromBottom - | FromLeft - - -type OpenedAnimation - = OpenFromTop - | OpenFromRight - | OpenFromBottom - | OpenFromLeft - - -type ClosingAnimation - = ToTop - | ToRight - | ToBottom - | ToLeft - - -type ClosingEffect - = WithoutAnimate - | WithAnimate - - -type BodySettings - = BodySettings - { borderRadius : Css.Px - , fromTop : Css.Px - , width : History.History Css.Px - , height : Css.Px - - -- Internal settings - , bottomClosingTop : Css.Pct - , center : Css.Pct - } - - -type History a - = History a (List a) - - -type alias Header msg = - Styled.Html msg - - -type alias Body msg = - Styled.Html msg - - -type alias Footer msg = - Styled.Html msg - - -type Config msg - = Config - { openingAnimation : OpeningAnimation - , openedAnimation : OpenedAnimation - , closingAnimation : ClosingAnimation - , closingEffect : ClosingEffect - , headerCss : String - , header : Header msg - , bodyCss : String - , body : Body msg - , footerCss : String - , footer : Footer msg - , modalBodySettings : BodySettings - , tagger : Msg msg -> msg - } - - -{-| Create a new configuration for Modal. - - newConfig |> setHeaderCss "header--bg-color" - --} -newConfig : (Msg msg -> msg) -> Config msg -newConfig tagger = - Config - { openingAnimation = FromTop - , openedAnimation = OpenFromTop - , closingAnimation = ToTop - , closingEffect = WithAnimate - , headerCss = "" - , header = text "" - , bodyCss = "" - , body = text "" - , footerCss = "" - , footer = text "" - , modalBodySettings = - BodySettings - { borderRadius = Css.px 5 - , fromTop = Css.px 200 - , width = create (Css.px 600) - , height = Css.px 185 - , bottomClosingTop = Css.pct 65 - , center = Css.pct 33 - } - , tagger = tagger - } - - -openModal : (Msg msg -> msg) -> Config msg -> msg -openModal fn config = - fn (OpenModal config) - - -closeModal : (Msg msg -> msg) -> msg -closeModal fn = - fn CloseModal - - -animationEnd : (Msg msg -> msg) -> msg -animationEnd fn = - fn AnimationEnd - - - --- Update - - -{-| Update the component state. In your parent update function -just add - - ModalMsg modalMsg -> - let - ( updatedModal, cmdModal ) = - Modal.update modalMsg model.modal - in - ( { model | modal = updatedModal } - , Cmd.map ModalMsg cmdModal - ) - --} -update : Msg msg -> Model msg -> ( Model msg, Cmd (Msg msg) ) -update msg model = - case msg of - OpenModal config -> - ( Opening config - , cmdGetWindowSize - ) - - CloseModal -> - ( setModalState model - , Cmd.none - ) - - AnimationEnd -> - ( setModalState model - , Cmd.none - ) - - GetWindowSize width height -> - ( setModalBodyPosition width height model - , Cmd.none - ) - - - --- View - - -{-| Render a view of modal window - - Modal.view model.modal - --} -view : Model msg -> Html.Html msg -view modal = - Styled.toUnstyled <| - case modal of - Opening (Config config) -> - div [ modalFade ] - [ modalBodyView - (Just AnimationEnd) - (openingAnimationClass config.openingAnimation config.modalBodySettings) - (Config config) - ] - - Opened (Config config) -> - div [ modalFade ] - [ modalBodyView - Nothing - (openedAnimationClass config.openedAnimation config.modalBodySettings) - (Config config) - ] - - Closing (Config config) -> - div - [ modalFade - , modalClose - , closingEffectClass config.closingEffect - ] - [ modalBodyView - (Just AnimationEnd) - (closingAnimationClass config.closingAnimation config.modalBodySettings) - (Config config) - ] - - Closed -> - text "" - - -{-| @priv -View of modal body --} -modalBodyView : Maybe (Msg msg) -> Attribute msg -> Config msg -> Html msg -modalBodyView animationTagger animation (Config config) = - let - animTgr = - case animationTagger of - Just msg -> - onAnimationEnd (config.tagger msg) - - Nothing -> - css [] - in - div - [ modalBody config.modalBodySettings - , animation - , animTgr - ] - [ div [ class config.headerCss ] - [ config.header ] - , div [ class config.bodyCss ] - [ config.body ] - , div [ class config.footerCss ] - [ config.footer ] - ] - - - --- Setters for Config - - -{-| Set styles as css of modal's body - - Modal.setBodyCss "body--bg-color" config - --} -setBodyCss : String -> Config msg -> Config msg -setBodyCss newBodyCss config = - let - fn (Config c) = - Config { c | bodyCss = newBodyCss } - in - mapConfig fn config - - -{-| Set styles and msg to the body of modal - - bodySuccess : Html msg - bodySuccess = - div [][ text "Hello from body modal." ] - - Modal.setBody bodySuccess config - --} -setBody : VirtualDom.Node msg -> Config msg -> Config msg -setBody newBody config = - let - fn (Config c) = - Config { c | body = Styled.fromUnstyled newBody } - in - mapConfig fn config - - -setHeaderCss : String -> Config msg -> Config msg -setHeaderCss newHeaderCss config = - let - fn (Config c) = - Config { c | headerCss = newHeaderCss } - in - mapConfig fn config - - -setHeader : VirtualDom.Node msg -> Config msg -> Config msg -setHeader newHeader config = - let - fn (Config c) = - Config { c | header = Styled.fromUnstyled newHeader } - in - mapConfig fn config - - -setFooterCss : String -> Config msg -> Config msg -setFooterCss newFooterCss config = - let - fn (Config c) = - Config { c | footerCss = newFooterCss } - in - mapConfig fn config - - -setFooter : VirtualDom.Node msg -> Config msg -> Config msg -setFooter newFooter config = - let - fn (Config c) = - Config { c | footer = Styled.fromUnstyled newFooter } - in - mapConfig fn config - - -setOpeningAnimation : OpeningAnimation -> Config msg -> Config msg -setOpeningAnimation opening config = - let - fn (Config c) = - Config { c | openingAnimation = opening } - in - mapConfig fn config - - -setOpenedAnimation : OpenedAnimation -> Config msg -> Config msg -setOpenedAnimation opened config = - let - fn (Config c) = - Config { c | openedAnimation = opened } - in - mapConfig fn config - - -setClosingAnimation : ClosingAnimation -> Config msg -> Config msg -setClosingAnimation closing config = - let - fn (Config c) = - Config { c | closingAnimation = closing } - in - mapConfig fn config - - -setClosingEffect : ClosingEffect -> Config msg -> Config msg -setClosingEffect newClosingEffect config = - let - fn (Config c) = - Config { c | closingEffect = newClosingEffect } - in - mapConfig fn config - - -setBodyWidth : Float -> Config msg -> Config msg -setBodyWidth width config = - let - setWidth : Float -> BodySettings -> BodySettings - setWidth newWidth (BodySettings bs) = - BodySettings { bs | width = create (Css.px newWidth) } - in - setBodySettings (setWidth width) config - - -setBodyHeight : Float -> Config msg -> Config msg -setBodyHeight height config = - let - setHeight : Float -> BodySettings -> BodySettings - setHeight newHeight (BodySettings bs) = - BodySettings { bs | height = Css.px newHeight } - in - setBodySettings (setHeight height) config - - -setBodyFromTop : Float -> Config msg -> Config msg -setBodyFromTop top config = - let - setTop : Float -> BodySettings -> BodySettings - setTop newTop (BodySettings bs) = - BodySettings { bs | fromTop = Css.px newTop } - in - setBodySettings (setTop top) config - - -setBodyBorderRadius : Float -> Config msg -> Config msg -setBodyBorderRadius borderRad config = - let - setBorderRadius : Float -> BodySettings -> BodySettings - setBorderRadius newBorderRad (BodySettings bs) = - BodySettings { bs | borderRadius = Css.px newBorderRad } - in - setBodySettings (setBorderRadius borderRad) config - - -setBodySettings : (BodySettings -> BodySettings) -> Config msg -> Config msg -setBodySettings updateFn (Config c) = - Config { c | modalBodySettings = updateFn c.modalBodySettings } - - - --- Private setters for modal window - - -openingAnimationClass : OpeningAnimation -> BodySettings -> Attribute msg -openingAnimationClass animation bodySettings = - case animation of - FromTop -> - modalTopOpening bodySettings - - FromRight -> - modalRigthOpening bodySettings - - FromBottom -> - modalBottomOpening bodySettings - - FromLeft -> - modalLeftOpening bodySettings - - -openedAnimationClass : OpenedAnimation -> BodySettings -> Attribute msg -openedAnimationClass animation bodySettings = - case animation of - OpenFromTop -> - modalOpenFromTop bodySettings - - OpenFromRight -> - modalOpenFromRigth bodySettings - - OpenFromBottom -> - modalOpenFromBottom bodySettings - - OpenFromLeft -> - modalOpenFromLeft bodySettings - - -closingAnimationClass : ClosingAnimation -> BodySettings -> Attribute msg -closingAnimationClass animation bodySettings = - case animation of - ToTop -> - modalTopClosing bodySettings - - ToRight -> - modalRightClosing bodySettings - - ToBottom -> - modalBottomClosing bodySettings - - ToLeft -> - modalLeftClosing bodySettings - - -closingEffectClass : ClosingEffect -> Attribute msg -closingEffectClass animation = - case animation of - WithoutAnimate -> - css [ Css.display Css.none ] - - WithAnimate -> - css [] - - - --- Private css style and animations for modal window - - -modalFade : Attribute msg -modalFade = - css - [ Css.position Css.absolute - , Css.top (Css.px 0) - , Css.bottom (Css.px 0) - , Css.left (Css.px 0) - , Css.right (Css.px 0) - , Css.backgroundColor (Css.rgba 0 0 0 0.5) - , Css.zIndex (Css.int 999) - , Css.overflow Css.hidden - ] - - -modalBody : BodySettings -> Attribute msg -modalBody (BodySettings body) = - css - [ Css.displayFlex - , Css.flexDirection Css.column - , Css.alignItems Css.center - , Css.property "align-content" "space-between" - , Css.width (current body.width) - , Css.minHeight body.height - , Css.borderRadius body.borderRadius - , Css.boxShadow4 (Css.px 0) (Css.px 3) (Css.px 7) (Css.rgba 0 0 0 0.75) - , Css.backgroundColor (Css.rgb 255 255 255) - ] - - - --- Opening animations - - -modalTopOpening : BodySettings -> Attribute msg -modalTopOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "0" - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalRigthOpening : BodySettings -> Attribute msg -modalRigthOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "right" "0" - , Animations.property "top" body.fromTop.value - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "right" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "right" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalBottomOpening : BodySettings -> Attribute msg -modalBottomOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "75%" - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalLeftOpening : BodySettings -> Attribute msg -modalLeftOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" "0" - , Animations.property "top" body.fromTop.value - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - - --- Open animations - - -modalOpenFromTop : BodySettings -> Attribute msg -modalOpenFromTop (BodySettings body) = - css - [ --Css.animationName - -- (Animations.keyframes - -- [ ( 0 - -- , [ Animations.property "opacity" "0" - -- , Animations.property "left" body.center.value - -- , Animations.property "top" "0" - -- ] - -- ) - -- , ( 75 - -- , [ Animations.property "opacity" "1" - -- , Animations.property "left" body.center.value - -- , Animations.property "top" body.fromTop.value - -- ] - -- ) - -- ] - -- ) - -- , Css.property "animation-duration" "1s" - Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromRigth : BodySettings -> Attribute msg -modalOpenFromRigth (BodySettings body) = - css - [ Css.property "right" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromBottom : BodySettings -> Attribute msg -modalOpenFromBottom (BodySettings body) = - css - [ Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromLeft : BodySettings -> Attribute msg -modalOpenFromLeft (BodySettings body) = - css - [ Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - - --- Closing animations - - -modalTopClosing : BodySettings -> Attribute msg -modalTopClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "20px" - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.property "left" body.center.value - , Css.top (Css.px 0) - ] - - -modalRightClosing : BodySettings -> Attribute msg -modalRightClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "right" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "right" "0" - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.right (Css.px 0) - , Css.top body.fromTop - ] - - -modalBottomClosing : BodySettings -> Attribute msg -modalBottomClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" body.bottomClosingTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.property "left" body.center.value - ] - - -modalLeftClosing : BodySettings -> Attribute msg -modalLeftClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" "0" - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.left (Css.px 0) - , Css.top body.fromTop - ] - - - --- Modal close animation - - -modalClose : Attribute msg -modalClose = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "background" "rgba(0, 0, 0, 0.5)" - ] - ) - , ( 100 - , [ Animations.property "background" "rgba(0, 0, 0, 0)" - , Animations.property "display" "none" - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.backgroundColor (Css.rgba 0 0 0 0) - ] - - - --- Internal - - -onAnimationStart : msg -> Attribute msg -onAnimationStart msg = - on "animationstart" (Json.succeed msg) - - -onAnimationIteration : msg -> Attribute msg -onAnimationIteration msg = - on "animationiteration" (Json.succeed msg) - - -onAnimationEnd : msg -> Attribute msg -onAnimationEnd msg = - on "animationend" (Json.succeed msg) - - -setModalBodyPosition : Float -> Float -> Model msg -> Model msg -setModalBodyPosition width height model = - case model of - Opening config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Opening - - Opened config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Opened - - Closing config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Closing - - Closed -> - Closed - - - --- Private helpers - - -{-| @priv -Helper for update config --} -mapConfig : (Config msg -> Config msg) -> Config msg -> Config msg -mapConfig fn config = - fn config - - -{-| @priv -Helper for update function --} -setModalState : Model msg -> Model msg -setModalState modal = - case modal of - Opening config -> - Opened config - - Opened config -> - Closing config - - Closing _ -> - Closed - - Closed -> - Closed - - -{-| @priv -Centering and adaptive width for modal body --} -recalcBodyModalProperties : Float -> Float -> BodySettings -> BodySettings -recalcBodyModalProperties windowWidth windowHeight (BodySettings bs) = - let - defaultBodyWidth = - bs.width - |> rewindAll - |> current - |> .numericValue - - setB = - 100 * (1 - bs.height.numericValue / (0.8 * windowHeight)) - - setC = - if defaultBodyWidth >= windowWidth then - 0 - - else - 50 * (1 - defaultBodyWidth / windowWidth) - - setW = - if defaultBodyWidth < windowWidth then - bs.width |> rewindAll - - else - bs.width |> forward (Css.px windowWidth) - in - BodySettings - { bs - | bottomClosingTop = Css.pct setB - , center = Css.pct setC - , width = setW - } +module Modal exposing + ( ClosingAnimation(..) + , ClosingEffect(..) + , Config + , Model(..) + , Msg + , OpenedAnimation(..) + , OpeningAnimation(..) + , animationEnd + , closeModal + , cancelModal + , cmdGetWindowSize + , initModel + , newConfig + , openModal + , setBody + , setBodyBorderRadius + , setBodyCss + , setBodyFromTop + , setBodyHeight + , setBodyWidth + , setClosingAnimation + , setClosingEffect + , setFooter + , setFooterCss + , setHeader + , setHeaderCss + , setOpenedAnimation + , setOpeningAnimation + , subscriptions + , update + , view + ) + +import Browser.Dom exposing (getViewport) +import Browser.Events exposing (onResize) +import Css +import Css.Animations as Animations +import Css.Transitions as Transitions +import History exposing (History(..), create, current, forward, rewind, rewindAll) +import Html +import Html.Styled as Styled exposing (..) +import Html.Styled.Attributes exposing (..) +import Html.Styled.Events exposing (on, onClick) +import Json.Decode as Json +import Task +import VirtualDom + + + +-- Model + + +type Model msg + = Opening (Config msg) + | Opened (Config msg) + | Closing (Config msg) + | Closed + | Canceling (Config msg) + | Canceled + + +initModel : Model msg +initModel = + Closed + + + +-- Commands + + +cmdGetWindowSize : Cmd (Msg msg) +cmdGetWindowSize = + Task.perform + (\{ viewport } -> + GetWindowSize viewport.width viewport.height + ) + getViewport + + + +-- Subscriptions + + +subscriptions : Sub (Msg msg) +subscriptions = + onResize (\x y -> GetWindowSize (toFloat x) (toFloat y)) + + + +-- Msg + + +type Msg msg + = OpenModal (Config msg) + | CloseModal + | CancelModal + | AnimationEnd + | GetWindowSize Float Float + + + +-- Types + + +type OpeningAnimation + = FromTop + | FromRight + | FromBottom + | FromLeft + + +type OpenedAnimation + = OpenFromTop + | OpenFromRight + | OpenFromBottom + | OpenFromLeft + + +type ClosingAnimation + = ToTop + | ToRight + | ToBottom + | ToLeft + + +type ClosingEffect + = WithoutAnimate + | WithAnimate + + +type BodySettings + = BodySettings + { borderRadius : Css.Px + , fromTop : Css.Px + , width : History.History Css.Px + , height : Css.Px + + -- Internal settings + , bottomClosingTop : Css.Pct + , center : Css.Pct + } + + +type History a + = History a (List a) + + +type alias Header msg = + Styled.Html msg + + +type alias Body msg = + Styled.Html msg + + +type alias Footer msg = + Styled.Html msg + + +type Config msg + = Config + { openingAnimation : OpeningAnimation + , openedAnimation : OpenedAnimation + , closingAnimation : ClosingAnimation + , closingEffect : ClosingEffect + , headerCss : String + , header : Header msg + , bodyCss : String + , body : Body msg + , footerCss : String + , footer : Footer msg + , modalBodySettings : BodySettings + , tagger : Msg msg -> msg + } + + +{-| Create a new configuration for Modal. + + newConfig |> setHeaderCss "header--bg-color" + +-} +newConfig : (Msg msg -> msg) -> Config msg +newConfig tagger = + Config + { openingAnimation = FromTop + , openedAnimation = OpenFromTop + , closingAnimation = ToTop + , closingEffect = WithAnimate + , headerCss = "" + , header = text "" + , bodyCss = "" + , body = text "" + , footerCss = "" + , footer = text "" + , modalBodySettings = + BodySettings + { borderRadius = Css.px 5 + , fromTop = Css.px 200 + , width = create (Css.px 600) + , height = Css.px 185 + , bottomClosingTop = Css.pct 65 + , center = Css.pct 33 + } + , tagger = tagger + } + + +openModal : (Msg msg -> msg) -> Config msg -> msg +openModal fn config = + fn (OpenModal config) + + +closeModal : (Msg msg -> msg) -> msg +closeModal fn = + fn CloseModal + +cancelModal : (Msg msg -> msg) -> msg +cancelModal fn = + fn CancelModal + +animationEnd : (Msg msg -> msg) -> msg +animationEnd fn = + fn AnimationEnd + + + +-- Update + + +{-| Update the component state. In your parent update function +just add + + ModalMsg modalMsg -> + let + ( updatedModal, cmdModal ) = + Modal.update modalMsg model.modal + in + case updatedModal of + Canceled -> + ( { model | modal = updatedModal } -- Model was canceled + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + Closed -> + ( { model | modal = updatedModal, confirmed = True } -- Modal was confirmed/acknowledged + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + _ -> + ( { model | modal = updatedModal } + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + +-} +update : Msg msg -> Model msg -> ( Model msg, Cmd (Msg msg) ) +update msg model = + case msg of + OpenModal config -> + ( Opening config + , cmdGetWindowSize + ) + + CloseModal -> + ( setModalState msg model + , Cmd.none + ) + + CancelModal -> + ( setModalState msg model + , Cmd.none + ) + + AnimationEnd -> + ( setModalState msg model + , Cmd.none + ) + + GetWindowSize width height -> + ( setModalBodyPosition width height model + , Cmd.none + ) + + + +-- View + + +{-| Render a view of modal window + + Modal.view model.modal + +-} +view : Model msg -> Html.Html msg +view modal = + Styled.toUnstyled <| + case modal of + Opening (Config config) -> + div [ modalFade ] + [ modalBodyView + (Just AnimationEnd) + (openingAnimationClass config.openingAnimation config.modalBodySettings) + (Config config) + ] + + Opened (Config config) -> + div [ modalFade ] + [ modalBodyView + Nothing + (openedAnimationClass config.openedAnimation config.modalBodySettings) + (Config config) + ] + + Closing (Config config) -> + div + [ modalFade + , modalClose + , closingEffectClass config.closingEffect + ] + [ modalBodyView + (Just AnimationEnd) + (closingAnimationClass config.closingAnimation config.modalBodySettings) + (Config config) + ] + + Closed -> + text "" + + Canceling (Config config) -> + div + [ modalFade + , modalClose + , closingEffectClass config.closingEffect + ] + [ modalBodyView + (Just AnimationEnd) + (closingAnimationClass config.closingAnimation config.modalBodySettings) + (Config config) + ] + + Canceled -> + text "" + + +{-| @priv +View of modal body +-} +modalBodyView : Maybe (Msg msg) -> Attribute msg -> Config msg -> Html msg +modalBodyView animationTagger animation (Config config) = + let + animTgr = + case animationTagger of + Just msg -> + onAnimationEnd (config.tagger msg) + + Nothing -> + css [] + in + div + [ modalBody config.modalBodySettings + , animation + , animTgr + ] + [ div [ class config.headerCss ] + [ config.header ] + , div [ class config.bodyCss ] + [ config.body ] + , div [ class config.footerCss ] + [ config.footer ] + ] + + + +-- Setters for Config + + +{-| Set styles as css of modal's body + + Modal.setBodyCss "body--bg-color" config + +-} +setBodyCss : String -> Config msg -> Config msg +setBodyCss newBodyCss config = + let + fn (Config c) = + Config { c | bodyCss = newBodyCss } + in + mapConfig fn config + + +{-| Set styles and msg to the body of modal + + bodySuccess : Html msg + bodySuccess = + div [][ text "Hello from body modal." ] + + Modal.setBody bodySuccess config + +-} +setBody : VirtualDom.Node msg -> Config msg -> Config msg +setBody newBody config = + let + fn (Config c) = + Config { c | body = Styled.fromUnstyled newBody } + in + mapConfig fn config + + +setHeaderCss : String -> Config msg -> Config msg +setHeaderCss newHeaderCss config = + let + fn (Config c) = + Config { c | headerCss = newHeaderCss } + in + mapConfig fn config + + +setHeader : VirtualDom.Node msg -> Config msg -> Config msg +setHeader newHeader config = + let + fn (Config c) = + Config { c | header = Styled.fromUnstyled newHeader } + in + mapConfig fn config + + +setFooterCss : String -> Config msg -> Config msg +setFooterCss newFooterCss config = + let + fn (Config c) = + Config { c | footerCss = newFooterCss } + in + mapConfig fn config + + +setFooter : VirtualDom.Node msg -> Config msg -> Config msg +setFooter newFooter config = + let + fn (Config c) = + Config { c | footer = Styled.fromUnstyled newFooter } + in + mapConfig fn config + + +setOpeningAnimation : OpeningAnimation -> Config msg -> Config msg +setOpeningAnimation opening config = + let + fn (Config c) = + Config { c | openingAnimation = opening } + in + mapConfig fn config + + +setOpenedAnimation : OpenedAnimation -> Config msg -> Config msg +setOpenedAnimation opened config = + let + fn (Config c) = + Config { c | openedAnimation = opened } + in + mapConfig fn config + + +setClosingAnimation : ClosingAnimation -> Config msg -> Config msg +setClosingAnimation closing config = + let + fn (Config c) = + Config { c | closingAnimation = closing } + in + mapConfig fn config + + +setClosingEffect : ClosingEffect -> Config msg -> Config msg +setClosingEffect newClosingEffect config = + let + fn (Config c) = + Config { c | closingEffect = newClosingEffect } + in + mapConfig fn config + + +setBodyWidth : Float -> Config msg -> Config msg +setBodyWidth width config = + let + setWidth : Float -> BodySettings -> BodySettings + setWidth newWidth (BodySettings bs) = + BodySettings { bs | width = create (Css.px newWidth) } + in + setBodySettings (setWidth width) config + + +setBodyHeight : Float -> Config msg -> Config msg +setBodyHeight height config = + let + setHeight : Float -> BodySettings -> BodySettings + setHeight newHeight (BodySettings bs) = + BodySettings { bs | height = Css.px newHeight } + in + setBodySettings (setHeight height) config + + +setBodyFromTop : Float -> Config msg -> Config msg +setBodyFromTop top config = + let + setTop : Float -> BodySettings -> BodySettings + setTop newTop (BodySettings bs) = + BodySettings { bs | fromTop = Css.px newTop } + in + setBodySettings (setTop top) config + + +setBodyBorderRadius : Float -> Config msg -> Config msg +setBodyBorderRadius borderRad config = + let + setBorderRadius : Float -> BodySettings -> BodySettings + setBorderRadius newBorderRad (BodySettings bs) = + BodySettings { bs | borderRadius = Css.px newBorderRad } + in + setBodySettings (setBorderRadius borderRad) config + + +setBodySettings : (BodySettings -> BodySettings) -> Config msg -> Config msg +setBodySettings updateFn (Config c) = + Config { c | modalBodySettings = updateFn c.modalBodySettings } + + + +-- Private setters for modal window + + +openingAnimationClass : OpeningAnimation -> BodySettings -> Attribute msg +openingAnimationClass animation bodySettings = + case animation of + FromTop -> + modalTopOpening bodySettings + + FromRight -> + modalRigthOpening bodySettings + + FromBottom -> + modalBottomOpening bodySettings + + FromLeft -> + modalLeftOpening bodySettings + + +openedAnimationClass : OpenedAnimation -> BodySettings -> Attribute msg +openedAnimationClass animation bodySettings = + case animation of + OpenFromTop -> + modalOpenFromTop bodySettings + + OpenFromRight -> + modalOpenFromRigth bodySettings + + OpenFromBottom -> + modalOpenFromBottom bodySettings + + OpenFromLeft -> + modalOpenFromLeft bodySettings + + +closingAnimationClass : ClosingAnimation -> BodySettings -> Attribute msg +closingAnimationClass animation bodySettings = + case animation of + ToTop -> + modalTopClosing bodySettings + + ToRight -> + modalRightClosing bodySettings + + ToBottom -> + modalBottomClosing bodySettings + + ToLeft -> + modalLeftClosing bodySettings + + +closingEffectClass : ClosingEffect -> Attribute msg +closingEffectClass animation = + case animation of + WithoutAnimate -> + css [ Css.display Css.none ] + + WithAnimate -> + css [] + + + +-- Private css style and animations for modal window + + +modalFade : Attribute msg +modalFade = + css + [ Css.position Css.absolute + , Css.top (Css.px 0) + , Css.bottom (Css.px 0) + , Css.left (Css.px 0) + , Css.right (Css.px 0) + , Css.backgroundColor (Css.rgba 0 0 0 0.5) + , Css.zIndex (Css.int 999) + , Css.overflow Css.hidden + ] + + +modalBody : BodySettings -> Attribute msg +modalBody (BodySettings body) = + css + [ Css.displayFlex + , Css.flexDirection Css.column + , Css.alignItems Css.center + , Css.property "align-content" "space-between" + , Css.width (current body.width) + , Css.minHeight body.height + , Css.borderRadius body.borderRadius + , Css.boxShadow4 (Css.px 0) (Css.px 3) (Css.px 7) (Css.rgba 0 0 0 0.75) + , Css.backgroundColor (Css.rgb 255 255 255) + ] + + + +-- Opening animations + + +modalTopOpening : BodySettings -> Attribute msg +modalTopOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "0" + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalRigthOpening : BodySettings -> Attribute msg +modalRigthOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "right" "0" + , Animations.property "top" body.fromTop.value + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "right" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "right" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalBottomOpening : BodySettings -> Attribute msg +modalBottomOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "75%" + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalLeftOpening : BodySettings -> Attribute msg +modalLeftOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" "0" + , Animations.property "top" body.fromTop.value + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + + +-- Open animations + + +modalOpenFromTop : BodySettings -> Attribute msg +modalOpenFromTop (BodySettings body) = + css + [ --Css.animationName + -- (Animations.keyframes + -- [ ( 0 + -- , [ Animations.property "opacity" "0" + -- , Animations.property "left" body.center.value + -- , Animations.property "top" "0" + -- ] + -- ) + -- , ( 75 + -- , [ Animations.property "opacity" "1" + -- , Animations.property "left" body.center.value + -- , Animations.property "top" body.fromTop.value + -- ] + -- ) + -- ] + -- ) + -- , Css.property "animation-duration" "1s" + Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromRigth : BodySettings -> Attribute msg +modalOpenFromRigth (BodySettings body) = + css + [ Css.property "right" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromBottom : BodySettings -> Attribute msg +modalOpenFromBottom (BodySettings body) = + css + [ Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromLeft : BodySettings -> Attribute msg +modalOpenFromLeft (BodySettings body) = + css + [ Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + + +-- Closing animations + + +modalTopClosing : BodySettings -> Attribute msg +modalTopClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "20px" + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.property "left" body.center.value + , Css.top (Css.px 0) + ] + + +modalRightClosing : BodySettings -> Attribute msg +modalRightClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "right" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "right" "0" + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.right (Css.px 0) + , Css.top body.fromTop + ] + + +modalBottomClosing : BodySettings -> Attribute msg +modalBottomClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" body.bottomClosingTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.property "left" body.center.value + ] + + +modalLeftClosing : BodySettings -> Attribute msg +modalLeftClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" "0" + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.left (Css.px 0) + , Css.top body.fromTop + ] + + + +-- Modal close animation + + +modalClose : Attribute msg +modalClose = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "background" "rgba(0, 0, 0, 0.5)" + ] + ) + , ( 100 + , [ Animations.property "background" "rgba(0, 0, 0, 0)" + , Animations.property "display" "none" + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.backgroundColor (Css.rgba 0 0 0 0) + ] + + + +-- Internal + + +onAnimationStart : msg -> Attribute msg +onAnimationStart msg = + on "animationstart" (Json.succeed msg) + + +onAnimationIteration : msg -> Attribute msg +onAnimationIteration msg = + on "animationiteration" (Json.succeed msg) + + +onAnimationEnd : msg -> Attribute msg +onAnimationEnd msg = + on "animationend" (Json.succeed msg) + + +setModalBodyPosition : Float -> Float -> Model msg -> Model msg +setModalBodyPosition width height model = + case model of + Opening config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Opening + + Opened config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Opened + + Closing config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Closing + + Closed -> + Closed + + Canceling config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Canceling + + Canceled -> + Canceled + + + +-- Private helpers + + +{-| @priv +Helper for update config +-} +mapConfig : (Config msg -> Config msg) -> Config msg -> Config msg +mapConfig fn config = + fn config + + +{-| @priv +Helper for update function +-} +setModalState : Msg msg -> Model msg -> Model msg +setModalState msg modal = + case modal of + Opening config -> + Opened config + + Opened config -> + if msg == CancelModal then + Canceling config + + else + Closing config + + Closing _ -> + Closed + + Closed -> + Closed + + Canceling _ -> + Canceled + + Canceled -> + Canceled + +{-| @priv +Centering and adaptive width for modal body +-} +recalcBodyModalProperties : Float -> Float -> BodySettings -> BodySettings +recalcBodyModalProperties windowWidth windowHeight (BodySettings bs) = + let + defaultBodyWidth = + bs.width + |> rewindAll + |> current + |> .numericValue + + setB = + 100 * (1 - bs.height.numericValue / (0.8 * windowHeight)) + + setC = + if defaultBodyWidth >= windowWidth then + 0 + + else + 50 * (1 - defaultBodyWidth / windowWidth) + + setW = + if defaultBodyWidth < windowWidth then + bs.width |> rewindAll + + else + bs.width |> forward (Css.px windowWidth) + in + BodySettings + { bs + | bottomClosingTop = Css.pct setB + , center = Css.pct setC + , width = setW + } From 28828e951c79f850cac53d8fa3a5f18fabf55499 Mon Sep 17 00:00:00 2001 From: anotherhale Date: Mon, 11 Jan 2021 13:01:20 -0800 Subject: [PATCH 2/6] updated eol character --- src/Modal.elm | 2086 ++++++++++++++++++++++++------------------------- 1 file changed, 1043 insertions(+), 1043 deletions(-) diff --git a/src/Modal.elm b/src/Modal.elm index b864b39..c998b99 100755 --- a/src/Modal.elm +++ b/src/Modal.elm @@ -1,1043 +1,1043 @@ -module Modal exposing - ( ClosingAnimation(..) - , ClosingEffect(..) - , Config - , Model(..) - , Msg - , OpenedAnimation(..) - , OpeningAnimation(..) - , animationEnd - , closeModal - , cancelModal - , cmdGetWindowSize - , initModel - , newConfig - , openModal - , setBody - , setBodyBorderRadius - , setBodyCss - , setBodyFromTop - , setBodyHeight - , setBodyWidth - , setClosingAnimation - , setClosingEffect - , setFooter - , setFooterCss - , setHeader - , setHeaderCss - , setOpenedAnimation - , setOpeningAnimation - , subscriptions - , update - , view - ) - -import Browser.Dom exposing (getViewport) -import Browser.Events exposing (onResize) -import Css -import Css.Animations as Animations -import Css.Transitions as Transitions -import History exposing (History(..), create, current, forward, rewind, rewindAll) -import Html -import Html.Styled as Styled exposing (..) -import Html.Styled.Attributes exposing (..) -import Html.Styled.Events exposing (on, onClick) -import Json.Decode as Json -import Task -import VirtualDom - - - --- Model - - -type Model msg - = Opening (Config msg) - | Opened (Config msg) - | Closing (Config msg) - | Closed - | Canceling (Config msg) - | Canceled - - -initModel : Model msg -initModel = - Closed - - - --- Commands - - -cmdGetWindowSize : Cmd (Msg msg) -cmdGetWindowSize = - Task.perform - (\{ viewport } -> - GetWindowSize viewport.width viewport.height - ) - getViewport - - - --- Subscriptions - - -subscriptions : Sub (Msg msg) -subscriptions = - onResize (\x y -> GetWindowSize (toFloat x) (toFloat y)) - - - --- Msg - - -type Msg msg - = OpenModal (Config msg) - | CloseModal - | CancelModal - | AnimationEnd - | GetWindowSize Float Float - - - --- Types - - -type OpeningAnimation - = FromTop - | FromRight - | FromBottom - | FromLeft - - -type OpenedAnimation - = OpenFromTop - | OpenFromRight - | OpenFromBottom - | OpenFromLeft - - -type ClosingAnimation - = ToTop - | ToRight - | ToBottom - | ToLeft - - -type ClosingEffect - = WithoutAnimate - | WithAnimate - - -type BodySettings - = BodySettings - { borderRadius : Css.Px - , fromTop : Css.Px - , width : History.History Css.Px - , height : Css.Px - - -- Internal settings - , bottomClosingTop : Css.Pct - , center : Css.Pct - } - - -type History a - = History a (List a) - - -type alias Header msg = - Styled.Html msg - - -type alias Body msg = - Styled.Html msg - - -type alias Footer msg = - Styled.Html msg - - -type Config msg - = Config - { openingAnimation : OpeningAnimation - , openedAnimation : OpenedAnimation - , closingAnimation : ClosingAnimation - , closingEffect : ClosingEffect - , headerCss : String - , header : Header msg - , bodyCss : String - , body : Body msg - , footerCss : String - , footer : Footer msg - , modalBodySettings : BodySettings - , tagger : Msg msg -> msg - } - - -{-| Create a new configuration for Modal. - - newConfig |> setHeaderCss "header--bg-color" - --} -newConfig : (Msg msg -> msg) -> Config msg -newConfig tagger = - Config - { openingAnimation = FromTop - , openedAnimation = OpenFromTop - , closingAnimation = ToTop - , closingEffect = WithAnimate - , headerCss = "" - , header = text "" - , bodyCss = "" - , body = text "" - , footerCss = "" - , footer = text "" - , modalBodySettings = - BodySettings - { borderRadius = Css.px 5 - , fromTop = Css.px 200 - , width = create (Css.px 600) - , height = Css.px 185 - , bottomClosingTop = Css.pct 65 - , center = Css.pct 33 - } - , tagger = tagger - } - - -openModal : (Msg msg -> msg) -> Config msg -> msg -openModal fn config = - fn (OpenModal config) - - -closeModal : (Msg msg -> msg) -> msg -closeModal fn = - fn CloseModal - -cancelModal : (Msg msg -> msg) -> msg -cancelModal fn = - fn CancelModal - -animationEnd : (Msg msg -> msg) -> msg -animationEnd fn = - fn AnimationEnd - - - --- Update - - -{-| Update the component state. In your parent update function -just add - - ModalMsg modalMsg -> - let - ( updatedModal, cmdModal ) = - Modal.update modalMsg model.modal - in - case updatedModal of - Canceled -> - ( { model | modal = updatedModal } -- Model was canceled - , Cmd.batch [Cmd.map ModalMsg cmdModal] - ) - Closed -> - ( { model | modal = updatedModal, confirmed = True } -- Modal was confirmed/acknowledged - , Cmd.batch [Cmd.map ModalMsg cmdModal] - ) - _ -> - ( { model | modal = updatedModal } - , Cmd.batch [Cmd.map ModalMsg cmdModal] - ) - --} -update : Msg msg -> Model msg -> ( Model msg, Cmd (Msg msg) ) -update msg model = - case msg of - OpenModal config -> - ( Opening config - , cmdGetWindowSize - ) - - CloseModal -> - ( setModalState msg model - , Cmd.none - ) - - CancelModal -> - ( setModalState msg model - , Cmd.none - ) - - AnimationEnd -> - ( setModalState msg model - , Cmd.none - ) - - GetWindowSize width height -> - ( setModalBodyPosition width height model - , Cmd.none - ) - - - --- View - - -{-| Render a view of modal window - - Modal.view model.modal - --} -view : Model msg -> Html.Html msg -view modal = - Styled.toUnstyled <| - case modal of - Opening (Config config) -> - div [ modalFade ] - [ modalBodyView - (Just AnimationEnd) - (openingAnimationClass config.openingAnimation config.modalBodySettings) - (Config config) - ] - - Opened (Config config) -> - div [ modalFade ] - [ modalBodyView - Nothing - (openedAnimationClass config.openedAnimation config.modalBodySettings) - (Config config) - ] - - Closing (Config config) -> - div - [ modalFade - , modalClose - , closingEffectClass config.closingEffect - ] - [ modalBodyView - (Just AnimationEnd) - (closingAnimationClass config.closingAnimation config.modalBodySettings) - (Config config) - ] - - Closed -> - text "" - - Canceling (Config config) -> - div - [ modalFade - , modalClose - , closingEffectClass config.closingEffect - ] - [ modalBodyView - (Just AnimationEnd) - (closingAnimationClass config.closingAnimation config.modalBodySettings) - (Config config) - ] - - Canceled -> - text "" - - -{-| @priv -View of modal body --} -modalBodyView : Maybe (Msg msg) -> Attribute msg -> Config msg -> Html msg -modalBodyView animationTagger animation (Config config) = - let - animTgr = - case animationTagger of - Just msg -> - onAnimationEnd (config.tagger msg) - - Nothing -> - css [] - in - div - [ modalBody config.modalBodySettings - , animation - , animTgr - ] - [ div [ class config.headerCss ] - [ config.header ] - , div [ class config.bodyCss ] - [ config.body ] - , div [ class config.footerCss ] - [ config.footer ] - ] - - - --- Setters for Config - - -{-| Set styles as css of modal's body - - Modal.setBodyCss "body--bg-color" config - --} -setBodyCss : String -> Config msg -> Config msg -setBodyCss newBodyCss config = - let - fn (Config c) = - Config { c | bodyCss = newBodyCss } - in - mapConfig fn config - - -{-| Set styles and msg to the body of modal - - bodySuccess : Html msg - bodySuccess = - div [][ text "Hello from body modal." ] - - Modal.setBody bodySuccess config - --} -setBody : VirtualDom.Node msg -> Config msg -> Config msg -setBody newBody config = - let - fn (Config c) = - Config { c | body = Styled.fromUnstyled newBody } - in - mapConfig fn config - - -setHeaderCss : String -> Config msg -> Config msg -setHeaderCss newHeaderCss config = - let - fn (Config c) = - Config { c | headerCss = newHeaderCss } - in - mapConfig fn config - - -setHeader : VirtualDom.Node msg -> Config msg -> Config msg -setHeader newHeader config = - let - fn (Config c) = - Config { c | header = Styled.fromUnstyled newHeader } - in - mapConfig fn config - - -setFooterCss : String -> Config msg -> Config msg -setFooterCss newFooterCss config = - let - fn (Config c) = - Config { c | footerCss = newFooterCss } - in - mapConfig fn config - - -setFooter : VirtualDom.Node msg -> Config msg -> Config msg -setFooter newFooter config = - let - fn (Config c) = - Config { c | footer = Styled.fromUnstyled newFooter } - in - mapConfig fn config - - -setOpeningAnimation : OpeningAnimation -> Config msg -> Config msg -setOpeningAnimation opening config = - let - fn (Config c) = - Config { c | openingAnimation = opening } - in - mapConfig fn config - - -setOpenedAnimation : OpenedAnimation -> Config msg -> Config msg -setOpenedAnimation opened config = - let - fn (Config c) = - Config { c | openedAnimation = opened } - in - mapConfig fn config - - -setClosingAnimation : ClosingAnimation -> Config msg -> Config msg -setClosingAnimation closing config = - let - fn (Config c) = - Config { c | closingAnimation = closing } - in - mapConfig fn config - - -setClosingEffect : ClosingEffect -> Config msg -> Config msg -setClosingEffect newClosingEffect config = - let - fn (Config c) = - Config { c | closingEffect = newClosingEffect } - in - mapConfig fn config - - -setBodyWidth : Float -> Config msg -> Config msg -setBodyWidth width config = - let - setWidth : Float -> BodySettings -> BodySettings - setWidth newWidth (BodySettings bs) = - BodySettings { bs | width = create (Css.px newWidth) } - in - setBodySettings (setWidth width) config - - -setBodyHeight : Float -> Config msg -> Config msg -setBodyHeight height config = - let - setHeight : Float -> BodySettings -> BodySettings - setHeight newHeight (BodySettings bs) = - BodySettings { bs | height = Css.px newHeight } - in - setBodySettings (setHeight height) config - - -setBodyFromTop : Float -> Config msg -> Config msg -setBodyFromTop top config = - let - setTop : Float -> BodySettings -> BodySettings - setTop newTop (BodySettings bs) = - BodySettings { bs | fromTop = Css.px newTop } - in - setBodySettings (setTop top) config - - -setBodyBorderRadius : Float -> Config msg -> Config msg -setBodyBorderRadius borderRad config = - let - setBorderRadius : Float -> BodySettings -> BodySettings - setBorderRadius newBorderRad (BodySettings bs) = - BodySettings { bs | borderRadius = Css.px newBorderRad } - in - setBodySettings (setBorderRadius borderRad) config - - -setBodySettings : (BodySettings -> BodySettings) -> Config msg -> Config msg -setBodySettings updateFn (Config c) = - Config { c | modalBodySettings = updateFn c.modalBodySettings } - - - --- Private setters for modal window - - -openingAnimationClass : OpeningAnimation -> BodySettings -> Attribute msg -openingAnimationClass animation bodySettings = - case animation of - FromTop -> - modalTopOpening bodySettings - - FromRight -> - modalRigthOpening bodySettings - - FromBottom -> - modalBottomOpening bodySettings - - FromLeft -> - modalLeftOpening bodySettings - - -openedAnimationClass : OpenedAnimation -> BodySettings -> Attribute msg -openedAnimationClass animation bodySettings = - case animation of - OpenFromTop -> - modalOpenFromTop bodySettings - - OpenFromRight -> - modalOpenFromRigth bodySettings - - OpenFromBottom -> - modalOpenFromBottom bodySettings - - OpenFromLeft -> - modalOpenFromLeft bodySettings - - -closingAnimationClass : ClosingAnimation -> BodySettings -> Attribute msg -closingAnimationClass animation bodySettings = - case animation of - ToTop -> - modalTopClosing bodySettings - - ToRight -> - modalRightClosing bodySettings - - ToBottom -> - modalBottomClosing bodySettings - - ToLeft -> - modalLeftClosing bodySettings - - -closingEffectClass : ClosingEffect -> Attribute msg -closingEffectClass animation = - case animation of - WithoutAnimate -> - css [ Css.display Css.none ] - - WithAnimate -> - css [] - - - --- Private css style and animations for modal window - - -modalFade : Attribute msg -modalFade = - css - [ Css.position Css.absolute - , Css.top (Css.px 0) - , Css.bottom (Css.px 0) - , Css.left (Css.px 0) - , Css.right (Css.px 0) - , Css.backgroundColor (Css.rgba 0 0 0 0.5) - , Css.zIndex (Css.int 999) - , Css.overflow Css.hidden - ] - - -modalBody : BodySettings -> Attribute msg -modalBody (BodySettings body) = - css - [ Css.displayFlex - , Css.flexDirection Css.column - , Css.alignItems Css.center - , Css.property "align-content" "space-between" - , Css.width (current body.width) - , Css.minHeight body.height - , Css.borderRadius body.borderRadius - , Css.boxShadow4 (Css.px 0) (Css.px 3) (Css.px 7) (Css.rgba 0 0 0 0.75) - , Css.backgroundColor (Css.rgb 255 255 255) - ] - - - --- Opening animations - - -modalTopOpening : BodySettings -> Attribute msg -modalTopOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "0" - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalRigthOpening : BodySettings -> Attribute msg -modalRigthOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "right" "0" - , Animations.property "top" body.fromTop.value - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "right" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "right" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalBottomOpening : BodySettings -> Attribute msg -modalBottomOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "75%" - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalLeftOpening : BodySettings -> Attribute msg -modalLeftOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" "0" - , Animations.property "top" body.fromTop.value - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - - --- Open animations - - -modalOpenFromTop : BodySettings -> Attribute msg -modalOpenFromTop (BodySettings body) = - css - [ --Css.animationName - -- (Animations.keyframes - -- [ ( 0 - -- , [ Animations.property "opacity" "0" - -- , Animations.property "left" body.center.value - -- , Animations.property "top" "0" - -- ] - -- ) - -- , ( 75 - -- , [ Animations.property "opacity" "1" - -- , Animations.property "left" body.center.value - -- , Animations.property "top" body.fromTop.value - -- ] - -- ) - -- ] - -- ) - -- , Css.property "animation-duration" "1s" - Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromRigth : BodySettings -> Attribute msg -modalOpenFromRigth (BodySettings body) = - css - [ Css.property "right" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromBottom : BodySettings -> Attribute msg -modalOpenFromBottom (BodySettings body) = - css - [ Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromLeft : BodySettings -> Attribute msg -modalOpenFromLeft (BodySettings body) = - css - [ Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - - --- Closing animations - - -modalTopClosing : BodySettings -> Attribute msg -modalTopClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "20px" - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.property "left" body.center.value - , Css.top (Css.px 0) - ] - - -modalRightClosing : BodySettings -> Attribute msg -modalRightClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "right" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "right" "0" - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.right (Css.px 0) - , Css.top body.fromTop - ] - - -modalBottomClosing : BodySettings -> Attribute msg -modalBottomClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" body.bottomClosingTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.property "left" body.center.value - ] - - -modalLeftClosing : BodySettings -> Attribute msg -modalLeftClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" "0" - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.left (Css.px 0) - , Css.top body.fromTop - ] - - - --- Modal close animation - - -modalClose : Attribute msg -modalClose = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "background" "rgba(0, 0, 0, 0.5)" - ] - ) - , ( 100 - , [ Animations.property "background" "rgba(0, 0, 0, 0)" - , Animations.property "display" "none" - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.backgroundColor (Css.rgba 0 0 0 0) - ] - - - --- Internal - - -onAnimationStart : msg -> Attribute msg -onAnimationStart msg = - on "animationstart" (Json.succeed msg) - - -onAnimationIteration : msg -> Attribute msg -onAnimationIteration msg = - on "animationiteration" (Json.succeed msg) - - -onAnimationEnd : msg -> Attribute msg -onAnimationEnd msg = - on "animationend" (Json.succeed msg) - - -setModalBodyPosition : Float -> Float -> Model msg -> Model msg -setModalBodyPosition width height model = - case model of - Opening config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Opening - - Opened config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Opened - - Closing config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Closing - - Closed -> - Closed - - Canceling config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Canceling - - Canceled -> - Canceled - - - --- Private helpers - - -{-| @priv -Helper for update config --} -mapConfig : (Config msg -> Config msg) -> Config msg -> Config msg -mapConfig fn config = - fn config - - -{-| @priv -Helper for update function --} -setModalState : Msg msg -> Model msg -> Model msg -setModalState msg modal = - case modal of - Opening config -> - Opened config - - Opened config -> - if msg == CancelModal then - Canceling config - - else - Closing config - - Closing _ -> - Closed - - Closed -> - Closed - - Canceling _ -> - Canceled - - Canceled -> - Canceled - -{-| @priv -Centering and adaptive width for modal body --} -recalcBodyModalProperties : Float -> Float -> BodySettings -> BodySettings -recalcBodyModalProperties windowWidth windowHeight (BodySettings bs) = - let - defaultBodyWidth = - bs.width - |> rewindAll - |> current - |> .numericValue - - setB = - 100 * (1 - bs.height.numericValue / (0.8 * windowHeight)) - - setC = - if defaultBodyWidth >= windowWidth then - 0 - - else - 50 * (1 - defaultBodyWidth / windowWidth) - - setW = - if defaultBodyWidth < windowWidth then - bs.width |> rewindAll - - else - bs.width |> forward (Css.px windowWidth) - in - BodySettings - { bs - | bottomClosingTop = Css.pct setB - , center = Css.pct setC - , width = setW - } +module Modal exposing + ( ClosingAnimation(..) + , ClosingEffect(..) + , Config + , Model(..) + , Msg + , OpenedAnimation(..) + , OpeningAnimation(..) + , animationEnd + , closeModal + , cancelModal + , cmdGetWindowSize + , initModel + , newConfig + , openModal + , setBody + , setBodyBorderRadius + , setBodyCss + , setBodyFromTop + , setBodyHeight + , setBodyWidth + , setClosingAnimation + , setClosingEffect + , setFooter + , setFooterCss + , setHeader + , setHeaderCss + , setOpenedAnimation + , setOpeningAnimation + , subscriptions + , update + , view + ) + +import Browser.Dom exposing (getViewport) +import Browser.Events exposing (onResize) +import Css +import Css.Animations as Animations +import Css.Transitions as Transitions +import History exposing (History(..), create, current, forward, rewind, rewindAll) +import Html +import Html.Styled as Styled exposing (..) +import Html.Styled.Attributes exposing (..) +import Html.Styled.Events exposing (on, onClick) +import Json.Decode as Json +import Task +import VirtualDom + + + +-- Model + + +type Model msg + = Opening (Config msg) + | Opened (Config msg) + | Closing (Config msg) + | Closed + | Canceling (Config msg) + | Canceled + + +initModel : Model msg +initModel = + Closed + + + +-- Commands + + +cmdGetWindowSize : Cmd (Msg msg) +cmdGetWindowSize = + Task.perform + (\{ viewport } -> + GetWindowSize viewport.width viewport.height + ) + getViewport + + + +-- Subscriptions + + +subscriptions : Sub (Msg msg) +subscriptions = + onResize (\x y -> GetWindowSize (toFloat x) (toFloat y)) + + + +-- Msg + + +type Msg msg + = OpenModal (Config msg) + | CloseModal + | CancelModal + | AnimationEnd + | GetWindowSize Float Float + + + +-- Types + + +type OpeningAnimation + = FromTop + | FromRight + | FromBottom + | FromLeft + + +type OpenedAnimation + = OpenFromTop + | OpenFromRight + | OpenFromBottom + | OpenFromLeft + + +type ClosingAnimation + = ToTop + | ToRight + | ToBottom + | ToLeft + + +type ClosingEffect + = WithoutAnimate + | WithAnimate + + +type BodySettings + = BodySettings + { borderRadius : Css.Px + , fromTop : Css.Px + , width : History.History Css.Px + , height : Css.Px + + -- Internal settings + , bottomClosingTop : Css.Pct + , center : Css.Pct + } + + +type History a + = History a (List a) + + +type alias Header msg = + Styled.Html msg + + +type alias Body msg = + Styled.Html msg + + +type alias Footer msg = + Styled.Html msg + + +type Config msg + = Config + { openingAnimation : OpeningAnimation + , openedAnimation : OpenedAnimation + , closingAnimation : ClosingAnimation + , closingEffect : ClosingEffect + , headerCss : String + , header : Header msg + , bodyCss : String + , body : Body msg + , footerCss : String + , footer : Footer msg + , modalBodySettings : BodySettings + , tagger : Msg msg -> msg + } + + +{-| Create a new configuration for Modal. + + newConfig |> setHeaderCss "header--bg-color" + +-} +newConfig : (Msg msg -> msg) -> Config msg +newConfig tagger = + Config + { openingAnimation = FromTop + , openedAnimation = OpenFromTop + , closingAnimation = ToTop + , closingEffect = WithAnimate + , headerCss = "" + , header = text "" + , bodyCss = "" + , body = text "" + , footerCss = "" + , footer = text "" + , modalBodySettings = + BodySettings + { borderRadius = Css.px 5 + , fromTop = Css.px 200 + , width = create (Css.px 600) + , height = Css.px 185 + , bottomClosingTop = Css.pct 65 + , center = Css.pct 33 + } + , tagger = tagger + } + + +openModal : (Msg msg -> msg) -> Config msg -> msg +openModal fn config = + fn (OpenModal config) + + +closeModal : (Msg msg -> msg) -> msg +closeModal fn = + fn CloseModal + +cancelModal : (Msg msg -> msg) -> msg +cancelModal fn = + fn CancelModal + +animationEnd : (Msg msg -> msg) -> msg +animationEnd fn = + fn AnimationEnd + + + +-- Update + + +{-| Update the component state. In your parent update function +just add + + ModalMsg modalMsg -> + let + ( updatedModal, cmdModal ) = + Modal.update modalMsg model.modal + in + case updatedModal of + Canceled -> + ( { model | modal = updatedModal } -- Model was canceled + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + Closed -> + ( { model | modal = updatedModal, confirmed = True } -- Modal was confirmed/acknowledged + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + _ -> + ( { model | modal = updatedModal } + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + +-} +update : Msg msg -> Model msg -> ( Model msg, Cmd (Msg msg) ) +update msg model = + case msg of + OpenModal config -> + ( Opening config + , cmdGetWindowSize + ) + + CloseModal -> + ( setModalState msg model + , Cmd.none + ) + + CancelModal -> + ( setModalState msg model + , Cmd.none + ) + + AnimationEnd -> + ( setModalState msg model + , Cmd.none + ) + + GetWindowSize width height -> + ( setModalBodyPosition width height model + , Cmd.none + ) + + + +-- View + + +{-| Render a view of modal window + + Modal.view model.modal + +-} +view : Model msg -> Html.Html msg +view modal = + Styled.toUnstyled <| + case modal of + Opening (Config config) -> + div [ modalFade ] + [ modalBodyView + (Just AnimationEnd) + (openingAnimationClass config.openingAnimation config.modalBodySettings) + (Config config) + ] + + Opened (Config config) -> + div [ modalFade ] + [ modalBodyView + Nothing + (openedAnimationClass config.openedAnimation config.modalBodySettings) + (Config config) + ] + + Closing (Config config) -> + div + [ modalFade + , modalClose + , closingEffectClass config.closingEffect + ] + [ modalBodyView + (Just AnimationEnd) + (closingAnimationClass config.closingAnimation config.modalBodySettings) + (Config config) + ] + + Closed -> + text "" + + Canceling (Config config) -> + div + [ modalFade + , modalClose + , closingEffectClass config.closingEffect + ] + [ modalBodyView + (Just AnimationEnd) + (closingAnimationClass config.closingAnimation config.modalBodySettings) + (Config config) + ] + + Canceled -> + text "" + + +{-| @priv +View of modal body +-} +modalBodyView : Maybe (Msg msg) -> Attribute msg -> Config msg -> Html msg +modalBodyView animationTagger animation (Config config) = + let + animTgr = + case animationTagger of + Just msg -> + onAnimationEnd (config.tagger msg) + + Nothing -> + css [] + in + div + [ modalBody config.modalBodySettings + , animation + , animTgr + ] + [ div [ class config.headerCss ] + [ config.header ] + , div [ class config.bodyCss ] + [ config.body ] + , div [ class config.footerCss ] + [ config.footer ] + ] + + + +-- Setters for Config + + +{-| Set styles as css of modal's body + + Modal.setBodyCss "body--bg-color" config + +-} +setBodyCss : String -> Config msg -> Config msg +setBodyCss newBodyCss config = + let + fn (Config c) = + Config { c | bodyCss = newBodyCss } + in + mapConfig fn config + + +{-| Set styles and msg to the body of modal + + bodySuccess : Html msg + bodySuccess = + div [][ text "Hello from body modal." ] + + Modal.setBody bodySuccess config + +-} +setBody : VirtualDom.Node msg -> Config msg -> Config msg +setBody newBody config = + let + fn (Config c) = + Config { c | body = Styled.fromUnstyled newBody } + in + mapConfig fn config + + +setHeaderCss : String -> Config msg -> Config msg +setHeaderCss newHeaderCss config = + let + fn (Config c) = + Config { c | headerCss = newHeaderCss } + in + mapConfig fn config + + +setHeader : VirtualDom.Node msg -> Config msg -> Config msg +setHeader newHeader config = + let + fn (Config c) = + Config { c | header = Styled.fromUnstyled newHeader } + in + mapConfig fn config + + +setFooterCss : String -> Config msg -> Config msg +setFooterCss newFooterCss config = + let + fn (Config c) = + Config { c | footerCss = newFooterCss } + in + mapConfig fn config + + +setFooter : VirtualDom.Node msg -> Config msg -> Config msg +setFooter newFooter config = + let + fn (Config c) = + Config { c | footer = Styled.fromUnstyled newFooter } + in + mapConfig fn config + + +setOpeningAnimation : OpeningAnimation -> Config msg -> Config msg +setOpeningAnimation opening config = + let + fn (Config c) = + Config { c | openingAnimation = opening } + in + mapConfig fn config + + +setOpenedAnimation : OpenedAnimation -> Config msg -> Config msg +setOpenedAnimation opened config = + let + fn (Config c) = + Config { c | openedAnimation = opened } + in + mapConfig fn config + + +setClosingAnimation : ClosingAnimation -> Config msg -> Config msg +setClosingAnimation closing config = + let + fn (Config c) = + Config { c | closingAnimation = closing } + in + mapConfig fn config + + +setClosingEffect : ClosingEffect -> Config msg -> Config msg +setClosingEffect newClosingEffect config = + let + fn (Config c) = + Config { c | closingEffect = newClosingEffect } + in + mapConfig fn config + + +setBodyWidth : Float -> Config msg -> Config msg +setBodyWidth width config = + let + setWidth : Float -> BodySettings -> BodySettings + setWidth newWidth (BodySettings bs) = + BodySettings { bs | width = create (Css.px newWidth) } + in + setBodySettings (setWidth width) config + + +setBodyHeight : Float -> Config msg -> Config msg +setBodyHeight height config = + let + setHeight : Float -> BodySettings -> BodySettings + setHeight newHeight (BodySettings bs) = + BodySettings { bs | height = Css.px newHeight } + in + setBodySettings (setHeight height) config + + +setBodyFromTop : Float -> Config msg -> Config msg +setBodyFromTop top config = + let + setTop : Float -> BodySettings -> BodySettings + setTop newTop (BodySettings bs) = + BodySettings { bs | fromTop = Css.px newTop } + in + setBodySettings (setTop top) config + + +setBodyBorderRadius : Float -> Config msg -> Config msg +setBodyBorderRadius borderRad config = + let + setBorderRadius : Float -> BodySettings -> BodySettings + setBorderRadius newBorderRad (BodySettings bs) = + BodySettings { bs | borderRadius = Css.px newBorderRad } + in + setBodySettings (setBorderRadius borderRad) config + + +setBodySettings : (BodySettings -> BodySettings) -> Config msg -> Config msg +setBodySettings updateFn (Config c) = + Config { c | modalBodySettings = updateFn c.modalBodySettings } + + + +-- Private setters for modal window + + +openingAnimationClass : OpeningAnimation -> BodySettings -> Attribute msg +openingAnimationClass animation bodySettings = + case animation of + FromTop -> + modalTopOpening bodySettings + + FromRight -> + modalRigthOpening bodySettings + + FromBottom -> + modalBottomOpening bodySettings + + FromLeft -> + modalLeftOpening bodySettings + + +openedAnimationClass : OpenedAnimation -> BodySettings -> Attribute msg +openedAnimationClass animation bodySettings = + case animation of + OpenFromTop -> + modalOpenFromTop bodySettings + + OpenFromRight -> + modalOpenFromRigth bodySettings + + OpenFromBottom -> + modalOpenFromBottom bodySettings + + OpenFromLeft -> + modalOpenFromLeft bodySettings + + +closingAnimationClass : ClosingAnimation -> BodySettings -> Attribute msg +closingAnimationClass animation bodySettings = + case animation of + ToTop -> + modalTopClosing bodySettings + + ToRight -> + modalRightClosing bodySettings + + ToBottom -> + modalBottomClosing bodySettings + + ToLeft -> + modalLeftClosing bodySettings + + +closingEffectClass : ClosingEffect -> Attribute msg +closingEffectClass animation = + case animation of + WithoutAnimate -> + css [ Css.display Css.none ] + + WithAnimate -> + css [] + + + +-- Private css style and animations for modal window + + +modalFade : Attribute msg +modalFade = + css + [ Css.position Css.absolute + , Css.top (Css.px 0) + , Css.bottom (Css.px 0) + , Css.left (Css.px 0) + , Css.right (Css.px 0) + , Css.backgroundColor (Css.rgba 0 0 0 0.5) + , Css.zIndex (Css.int 999) + , Css.overflow Css.hidden + ] + + +modalBody : BodySettings -> Attribute msg +modalBody (BodySettings body) = + css + [ Css.displayFlex + , Css.flexDirection Css.column + , Css.alignItems Css.center + , Css.property "align-content" "space-between" + , Css.width (current body.width) + , Css.minHeight body.height + , Css.borderRadius body.borderRadius + , Css.boxShadow4 (Css.px 0) (Css.px 3) (Css.px 7) (Css.rgba 0 0 0 0.75) + , Css.backgroundColor (Css.rgb 255 255 255) + ] + + + +-- Opening animations + + +modalTopOpening : BodySettings -> Attribute msg +modalTopOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "0" + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalRigthOpening : BodySettings -> Attribute msg +modalRigthOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "right" "0" + , Animations.property "top" body.fromTop.value + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "right" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "right" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalBottomOpening : BodySettings -> Attribute msg +modalBottomOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "75%" + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalLeftOpening : BodySettings -> Attribute msg +modalLeftOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" "0" + , Animations.property "top" body.fromTop.value + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + + +-- Open animations + + +modalOpenFromTop : BodySettings -> Attribute msg +modalOpenFromTop (BodySettings body) = + css + [ --Css.animationName + -- (Animations.keyframes + -- [ ( 0 + -- , [ Animations.property "opacity" "0" + -- , Animations.property "left" body.center.value + -- , Animations.property "top" "0" + -- ] + -- ) + -- , ( 75 + -- , [ Animations.property "opacity" "1" + -- , Animations.property "left" body.center.value + -- , Animations.property "top" body.fromTop.value + -- ] + -- ) + -- ] + -- ) + -- , Css.property "animation-duration" "1s" + Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromRigth : BodySettings -> Attribute msg +modalOpenFromRigth (BodySettings body) = + css + [ Css.property "right" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromBottom : BodySettings -> Attribute msg +modalOpenFromBottom (BodySettings body) = + css + [ Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromLeft : BodySettings -> Attribute msg +modalOpenFromLeft (BodySettings body) = + css + [ Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + + +-- Closing animations + + +modalTopClosing : BodySettings -> Attribute msg +modalTopClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "20px" + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.property "left" body.center.value + , Css.top (Css.px 0) + ] + + +modalRightClosing : BodySettings -> Attribute msg +modalRightClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "right" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "right" "0" + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.right (Css.px 0) + , Css.top body.fromTop + ] + + +modalBottomClosing : BodySettings -> Attribute msg +modalBottomClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" body.bottomClosingTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.property "left" body.center.value + ] + + +modalLeftClosing : BodySettings -> Attribute msg +modalLeftClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" "0" + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.left (Css.px 0) + , Css.top body.fromTop + ] + + + +-- Modal close animation + + +modalClose : Attribute msg +modalClose = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "background" "rgba(0, 0, 0, 0.5)" + ] + ) + , ( 100 + , [ Animations.property "background" "rgba(0, 0, 0, 0)" + , Animations.property "display" "none" + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.backgroundColor (Css.rgba 0 0 0 0) + ] + + + +-- Internal + + +onAnimationStart : msg -> Attribute msg +onAnimationStart msg = + on "animationstart" (Json.succeed msg) + + +onAnimationIteration : msg -> Attribute msg +onAnimationIteration msg = + on "animationiteration" (Json.succeed msg) + + +onAnimationEnd : msg -> Attribute msg +onAnimationEnd msg = + on "animationend" (Json.succeed msg) + + +setModalBodyPosition : Float -> Float -> Model msg -> Model msg +setModalBodyPosition width height model = + case model of + Opening config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Opening + + Opened config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Opened + + Closing config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Closing + + Closed -> + Closed + + Canceling config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Canceling + + Canceled -> + Canceled + + + +-- Private helpers + + +{-| @priv +Helper for update config +-} +mapConfig : (Config msg -> Config msg) -> Config msg -> Config msg +mapConfig fn config = + fn config + + +{-| @priv +Helper for update function +-} +setModalState : Msg msg -> Model msg -> Model msg +setModalState msg modal = + case modal of + Opening config -> + Opened config + + Opened config -> + if msg == CancelModal then + Canceling config + + else + Closing config + + Closing _ -> + Closed + + Closed -> + Closed + + Canceling _ -> + Canceled + + Canceled -> + Canceled + +{-| @priv +Centering and adaptive width for modal body +-} +recalcBodyModalProperties : Float -> Float -> BodySettings -> BodySettings +recalcBodyModalProperties windowWidth windowHeight (BodySettings bs) = + let + defaultBodyWidth = + bs.width + |> rewindAll + |> current + |> .numericValue + + setB = + 100 * (1 - bs.height.numericValue / (0.8 * windowHeight)) + + setC = + if defaultBodyWidth >= windowWidth then + 0 + + else + 50 * (1 - defaultBodyWidth / windowWidth) + + setW = + if defaultBodyWidth < windowWidth then + bs.width |> rewindAll + + else + bs.width |> forward (Css.px windowWidth) + in + BodySettings + { bs + | bottomClosingTop = Css.pct setB + , center = Css.pct setC + , width = setW + } From 37c2bcdec79861186712db3fa3020f043e7c80f3 Mon Sep 17 00:00:00 2001 From: anotherhale Date: Sun, 10 Jan 2021 14:46:53 -0800 Subject: [PATCH 3/6] Added support for Cancel event and update for Elm 19.1 --- elm.json | 2 +- src/Modal.elm | 2032 +++++++++++++++++++++++++------------------------ 2 files changed, 1044 insertions(+), 990 deletions(-) diff --git a/elm.json b/elm.json index 83033b8..6bfd09a 100644 --- a/elm.json +++ b/elm.json @@ -16,7 +16,7 @@ "elm/json": "1.0.0 <= v < 2.0.0", "elm/virtual-dom": "1.0.2 <= v < 2.0.0", "iodevs/elm-history": "1.0.0 <= v < 2.0.0", - "rtfeldman/elm-css": "15.1.0 <= v < 16.0.0" + "rtfeldman/elm-css": "16.0.0 <= v < 17.0.0" }, "test-dependencies": {} } diff --git a/src/Modal.elm b/src/Modal.elm index f1905f6..b864b39 100755 --- a/src/Modal.elm +++ b/src/Modal.elm @@ -1,989 +1,1043 @@ -module Modal exposing - ( ClosingAnimation(..) - , ClosingEffect(..) - , Config - , Model - , Msg - , OpenedAnimation(..) - , OpeningAnimation(..) - , animationEnd - , closeModal - , cmdGetWindowSize - , initModel - , newConfig - , openModal - , setBody - , setBodyBorderRadius - , setBodyCss - , setBodyFromTop - , setBodyHeight - , setBodyWidth - , setClosingAnimation - , setClosingEffect - , setFooter - , setFooterCss - , setHeader - , setHeaderCss - , setOpenedAnimation - , setOpeningAnimation - , subscriptions - , update - , view - ) - -import Browser.Dom exposing (getViewport) -import Browser.Events exposing (onResize) -import Css -import Css.Animations as Animations -import Css.Transitions as Transitions -import History exposing (History(..), create, current, forward, rewind, rewindAll) -import Html -import Html.Styled as Styled exposing (..) -import Html.Styled.Attributes exposing (..) -import Html.Styled.Events exposing (on, onClick) -import Json.Decode as Json -import Task -import VirtualDom - - - --- Model - - -type Model msg - = Opening (Config msg) - | Opened (Config msg) - | Closing (Config msg) - | Closed - - -initModel : Model msg -initModel = - Closed - - - --- Commands - - -cmdGetWindowSize : Cmd (Msg msg) -cmdGetWindowSize = - Task.perform - (\{ viewport } -> - GetWindowSize viewport.width viewport.height - ) - getViewport - - - --- Subscriptions - - -subscriptions : Sub (Msg msg) -subscriptions = - onResize (\x y -> GetWindowSize (toFloat x) (toFloat y)) - - - --- Msg - - -type Msg msg - = OpenModal (Config msg) - | CloseModal - | AnimationEnd - | GetWindowSize Float Float - - - --- Types - - -type OpeningAnimation - = FromTop - | FromRight - | FromBottom - | FromLeft - - -type OpenedAnimation - = OpenFromTop - | OpenFromRight - | OpenFromBottom - | OpenFromLeft - - -type ClosingAnimation - = ToTop - | ToRight - | ToBottom - | ToLeft - - -type ClosingEffect - = WithoutAnimate - | WithAnimate - - -type BodySettings - = BodySettings - { borderRadius : Css.Px - , fromTop : Css.Px - , width : History.History Css.Px - , height : Css.Px - - -- Internal settings - , bottomClosingTop : Css.Pct - , center : Css.Pct - } - - -type History a - = History a (List a) - - -type alias Header msg = - Styled.Html msg - - -type alias Body msg = - Styled.Html msg - - -type alias Footer msg = - Styled.Html msg - - -type Config msg - = Config - { openingAnimation : OpeningAnimation - , openedAnimation : OpenedAnimation - , closingAnimation : ClosingAnimation - , closingEffect : ClosingEffect - , headerCss : String - , header : Header msg - , bodyCss : String - , body : Body msg - , footerCss : String - , footer : Footer msg - , modalBodySettings : BodySettings - , tagger : Msg msg -> msg - } - - -{-| Create a new configuration for Modal. - - newConfig |> setHeaderCss "header--bg-color" - --} -newConfig : (Msg msg -> msg) -> Config msg -newConfig tagger = - Config - { openingAnimation = FromTop - , openedAnimation = OpenFromTop - , closingAnimation = ToTop - , closingEffect = WithAnimate - , headerCss = "" - , header = text "" - , bodyCss = "" - , body = text "" - , footerCss = "" - , footer = text "" - , modalBodySettings = - BodySettings - { borderRadius = Css.px 5 - , fromTop = Css.px 200 - , width = create (Css.px 600) - , height = Css.px 185 - , bottomClosingTop = Css.pct 65 - , center = Css.pct 33 - } - , tagger = tagger - } - - -openModal : (Msg msg -> msg) -> Config msg -> msg -openModal fn config = - fn (OpenModal config) - - -closeModal : (Msg msg -> msg) -> msg -closeModal fn = - fn CloseModal - - -animationEnd : (Msg msg -> msg) -> msg -animationEnd fn = - fn AnimationEnd - - - --- Update - - -{-| Update the component state. In your parent update function -just add - - ModalMsg modalMsg -> - let - ( updatedModal, cmdModal ) = - Modal.update modalMsg model.modal - in - ( { model | modal = updatedModal } - , Cmd.map ModalMsg cmdModal - ) - --} -update : Msg msg -> Model msg -> ( Model msg, Cmd (Msg msg) ) -update msg model = - case msg of - OpenModal config -> - ( Opening config - , cmdGetWindowSize - ) - - CloseModal -> - ( setModalState model - , Cmd.none - ) - - AnimationEnd -> - ( setModalState model - , Cmd.none - ) - - GetWindowSize width height -> - ( setModalBodyPosition width height model - , Cmd.none - ) - - - --- View - - -{-| Render a view of modal window - - Modal.view model.modal - --} -view : Model msg -> Html.Html msg -view modal = - Styled.toUnstyled <| - case modal of - Opening (Config config) -> - div [ modalFade ] - [ modalBodyView - (Just AnimationEnd) - (openingAnimationClass config.openingAnimation config.modalBodySettings) - (Config config) - ] - - Opened (Config config) -> - div [ modalFade ] - [ modalBodyView - Nothing - (openedAnimationClass config.openedAnimation config.modalBodySettings) - (Config config) - ] - - Closing (Config config) -> - div - [ modalFade - , modalClose - , closingEffectClass config.closingEffect - ] - [ modalBodyView - (Just AnimationEnd) - (closingAnimationClass config.closingAnimation config.modalBodySettings) - (Config config) - ] - - Closed -> - text "" - - -{-| @priv -View of modal body --} -modalBodyView : Maybe (Msg msg) -> Attribute msg -> Config msg -> Html msg -modalBodyView animationTagger animation (Config config) = - let - animTgr = - case animationTagger of - Just msg -> - onAnimationEnd (config.tagger msg) - - Nothing -> - css [] - in - div - [ modalBody config.modalBodySettings - , animation - , animTgr - ] - [ div [ class config.headerCss ] - [ config.header ] - , div [ class config.bodyCss ] - [ config.body ] - , div [ class config.footerCss ] - [ config.footer ] - ] - - - --- Setters for Config - - -{-| Set styles as css of modal's body - - Modal.setBodyCss "body--bg-color" config - --} -setBodyCss : String -> Config msg -> Config msg -setBodyCss newBodyCss config = - let - fn (Config c) = - Config { c | bodyCss = newBodyCss } - in - mapConfig fn config - - -{-| Set styles and msg to the body of modal - - bodySuccess : Html msg - bodySuccess = - div [][ text "Hello from body modal." ] - - Modal.setBody bodySuccess config - --} -setBody : VirtualDom.Node msg -> Config msg -> Config msg -setBody newBody config = - let - fn (Config c) = - Config { c | body = Styled.fromUnstyled newBody } - in - mapConfig fn config - - -setHeaderCss : String -> Config msg -> Config msg -setHeaderCss newHeaderCss config = - let - fn (Config c) = - Config { c | headerCss = newHeaderCss } - in - mapConfig fn config - - -setHeader : VirtualDom.Node msg -> Config msg -> Config msg -setHeader newHeader config = - let - fn (Config c) = - Config { c | header = Styled.fromUnstyled newHeader } - in - mapConfig fn config - - -setFooterCss : String -> Config msg -> Config msg -setFooterCss newFooterCss config = - let - fn (Config c) = - Config { c | footerCss = newFooterCss } - in - mapConfig fn config - - -setFooter : VirtualDom.Node msg -> Config msg -> Config msg -setFooter newFooter config = - let - fn (Config c) = - Config { c | footer = Styled.fromUnstyled newFooter } - in - mapConfig fn config - - -setOpeningAnimation : OpeningAnimation -> Config msg -> Config msg -setOpeningAnimation opening config = - let - fn (Config c) = - Config { c | openingAnimation = opening } - in - mapConfig fn config - - -setOpenedAnimation : OpenedAnimation -> Config msg -> Config msg -setOpenedAnimation opened config = - let - fn (Config c) = - Config { c | openedAnimation = opened } - in - mapConfig fn config - - -setClosingAnimation : ClosingAnimation -> Config msg -> Config msg -setClosingAnimation closing config = - let - fn (Config c) = - Config { c | closingAnimation = closing } - in - mapConfig fn config - - -setClosingEffect : ClosingEffect -> Config msg -> Config msg -setClosingEffect newClosingEffect config = - let - fn (Config c) = - Config { c | closingEffect = newClosingEffect } - in - mapConfig fn config - - -setBodyWidth : Float -> Config msg -> Config msg -setBodyWidth width config = - let - setWidth : Float -> BodySettings -> BodySettings - setWidth newWidth (BodySettings bs) = - BodySettings { bs | width = create (Css.px newWidth) } - in - setBodySettings (setWidth width) config - - -setBodyHeight : Float -> Config msg -> Config msg -setBodyHeight height config = - let - setHeight : Float -> BodySettings -> BodySettings - setHeight newHeight (BodySettings bs) = - BodySettings { bs | height = Css.px newHeight } - in - setBodySettings (setHeight height) config - - -setBodyFromTop : Float -> Config msg -> Config msg -setBodyFromTop top config = - let - setTop : Float -> BodySettings -> BodySettings - setTop newTop (BodySettings bs) = - BodySettings { bs | fromTop = Css.px newTop } - in - setBodySettings (setTop top) config - - -setBodyBorderRadius : Float -> Config msg -> Config msg -setBodyBorderRadius borderRad config = - let - setBorderRadius : Float -> BodySettings -> BodySettings - setBorderRadius newBorderRad (BodySettings bs) = - BodySettings { bs | borderRadius = Css.px newBorderRad } - in - setBodySettings (setBorderRadius borderRad) config - - -setBodySettings : (BodySettings -> BodySettings) -> Config msg -> Config msg -setBodySettings updateFn (Config c) = - Config { c | modalBodySettings = updateFn c.modalBodySettings } - - - --- Private setters for modal window - - -openingAnimationClass : OpeningAnimation -> BodySettings -> Attribute msg -openingAnimationClass animation bodySettings = - case animation of - FromTop -> - modalTopOpening bodySettings - - FromRight -> - modalRigthOpening bodySettings - - FromBottom -> - modalBottomOpening bodySettings - - FromLeft -> - modalLeftOpening bodySettings - - -openedAnimationClass : OpenedAnimation -> BodySettings -> Attribute msg -openedAnimationClass animation bodySettings = - case animation of - OpenFromTop -> - modalOpenFromTop bodySettings - - OpenFromRight -> - modalOpenFromRigth bodySettings - - OpenFromBottom -> - modalOpenFromBottom bodySettings - - OpenFromLeft -> - modalOpenFromLeft bodySettings - - -closingAnimationClass : ClosingAnimation -> BodySettings -> Attribute msg -closingAnimationClass animation bodySettings = - case animation of - ToTop -> - modalTopClosing bodySettings - - ToRight -> - modalRightClosing bodySettings - - ToBottom -> - modalBottomClosing bodySettings - - ToLeft -> - modalLeftClosing bodySettings - - -closingEffectClass : ClosingEffect -> Attribute msg -closingEffectClass animation = - case animation of - WithoutAnimate -> - css [ Css.display Css.none ] - - WithAnimate -> - css [] - - - --- Private css style and animations for modal window - - -modalFade : Attribute msg -modalFade = - css - [ Css.position Css.absolute - , Css.top (Css.px 0) - , Css.bottom (Css.px 0) - , Css.left (Css.px 0) - , Css.right (Css.px 0) - , Css.backgroundColor (Css.rgba 0 0 0 0.5) - , Css.zIndex (Css.int 999) - , Css.overflow Css.hidden - ] - - -modalBody : BodySettings -> Attribute msg -modalBody (BodySettings body) = - css - [ Css.displayFlex - , Css.flexDirection Css.column - , Css.alignItems Css.center - , Css.property "align-content" "space-between" - , Css.width (current body.width) - , Css.minHeight body.height - , Css.borderRadius body.borderRadius - , Css.boxShadow4 (Css.px 0) (Css.px 3) (Css.px 7) (Css.rgba 0 0 0 0.75) - , Css.backgroundColor (Css.rgb 255 255 255) - ] - - - --- Opening animations - - -modalTopOpening : BodySettings -> Attribute msg -modalTopOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "0" - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalRigthOpening : BodySettings -> Attribute msg -modalRigthOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "right" "0" - , Animations.property "top" body.fromTop.value - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "right" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "right" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalBottomOpening : BodySettings -> Attribute msg -modalBottomOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "75%" - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalLeftOpening : BodySettings -> Attribute msg -modalLeftOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" "0" - , Animations.property "top" body.fromTop.value - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - - --- Open animations - - -modalOpenFromTop : BodySettings -> Attribute msg -modalOpenFromTop (BodySettings body) = - css - [ --Css.animationName - -- (Animations.keyframes - -- [ ( 0 - -- , [ Animations.property "opacity" "0" - -- , Animations.property "left" body.center.value - -- , Animations.property "top" "0" - -- ] - -- ) - -- , ( 75 - -- , [ Animations.property "opacity" "1" - -- , Animations.property "left" body.center.value - -- , Animations.property "top" body.fromTop.value - -- ] - -- ) - -- ] - -- ) - -- , Css.property "animation-duration" "1s" - Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromRigth : BodySettings -> Attribute msg -modalOpenFromRigth (BodySettings body) = - css - [ Css.property "right" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromBottom : BodySettings -> Attribute msg -modalOpenFromBottom (BodySettings body) = - css - [ Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromLeft : BodySettings -> Attribute msg -modalOpenFromLeft (BodySettings body) = - css - [ Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - - --- Closing animations - - -modalTopClosing : BodySettings -> Attribute msg -modalTopClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "20px" - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.property "left" body.center.value - , Css.top (Css.px 0) - ] - - -modalRightClosing : BodySettings -> Attribute msg -modalRightClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "right" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "right" "0" - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.right (Css.px 0) - , Css.top body.fromTop - ] - - -modalBottomClosing : BodySettings -> Attribute msg -modalBottomClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" body.bottomClosingTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.property "left" body.center.value - ] - - -modalLeftClosing : BodySettings -> Attribute msg -modalLeftClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" "0" - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.left (Css.px 0) - , Css.top body.fromTop - ] - - - --- Modal close animation - - -modalClose : Attribute msg -modalClose = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "background" "rgba(0, 0, 0, 0.5)" - ] - ) - , ( 100 - , [ Animations.property "background" "rgba(0, 0, 0, 0)" - , Animations.property "display" "none" - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.backgroundColor (Css.rgba 0 0 0 0) - ] - - - --- Internal - - -onAnimationStart : msg -> Attribute msg -onAnimationStart msg = - on "animationstart" (Json.succeed msg) - - -onAnimationIteration : msg -> Attribute msg -onAnimationIteration msg = - on "animationiteration" (Json.succeed msg) - - -onAnimationEnd : msg -> Attribute msg -onAnimationEnd msg = - on "animationend" (Json.succeed msg) - - -setModalBodyPosition : Float -> Float -> Model msg -> Model msg -setModalBodyPosition width height model = - case model of - Opening config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Opening - - Opened config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Opened - - Closing config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Closing - - Closed -> - Closed - - - --- Private helpers - - -{-| @priv -Helper for update config --} -mapConfig : (Config msg -> Config msg) -> Config msg -> Config msg -mapConfig fn config = - fn config - - -{-| @priv -Helper for update function --} -setModalState : Model msg -> Model msg -setModalState modal = - case modal of - Opening config -> - Opened config - - Opened config -> - Closing config - - Closing _ -> - Closed - - Closed -> - Closed - - -{-| @priv -Centering and adaptive width for modal body --} -recalcBodyModalProperties : Float -> Float -> BodySettings -> BodySettings -recalcBodyModalProperties windowWidth windowHeight (BodySettings bs) = - let - defaultBodyWidth = - bs.width - |> rewindAll - |> current - |> .numericValue - - setB = - 100 * (1 - bs.height.numericValue / (0.8 * windowHeight)) - - setC = - if defaultBodyWidth >= windowWidth then - 0 - - else - 50 * (1 - defaultBodyWidth / windowWidth) - - setW = - if defaultBodyWidth < windowWidth then - bs.width |> rewindAll - - else - bs.width |> forward (Css.px windowWidth) - in - BodySettings - { bs - | bottomClosingTop = Css.pct setB - , center = Css.pct setC - , width = setW - } +module Modal exposing + ( ClosingAnimation(..) + , ClosingEffect(..) + , Config + , Model(..) + , Msg + , OpenedAnimation(..) + , OpeningAnimation(..) + , animationEnd + , closeModal + , cancelModal + , cmdGetWindowSize + , initModel + , newConfig + , openModal + , setBody + , setBodyBorderRadius + , setBodyCss + , setBodyFromTop + , setBodyHeight + , setBodyWidth + , setClosingAnimation + , setClosingEffect + , setFooter + , setFooterCss + , setHeader + , setHeaderCss + , setOpenedAnimation + , setOpeningAnimation + , subscriptions + , update + , view + ) + +import Browser.Dom exposing (getViewport) +import Browser.Events exposing (onResize) +import Css +import Css.Animations as Animations +import Css.Transitions as Transitions +import History exposing (History(..), create, current, forward, rewind, rewindAll) +import Html +import Html.Styled as Styled exposing (..) +import Html.Styled.Attributes exposing (..) +import Html.Styled.Events exposing (on, onClick) +import Json.Decode as Json +import Task +import VirtualDom + + + +-- Model + + +type Model msg + = Opening (Config msg) + | Opened (Config msg) + | Closing (Config msg) + | Closed + | Canceling (Config msg) + | Canceled + + +initModel : Model msg +initModel = + Closed + + + +-- Commands + + +cmdGetWindowSize : Cmd (Msg msg) +cmdGetWindowSize = + Task.perform + (\{ viewport } -> + GetWindowSize viewport.width viewport.height + ) + getViewport + + + +-- Subscriptions + + +subscriptions : Sub (Msg msg) +subscriptions = + onResize (\x y -> GetWindowSize (toFloat x) (toFloat y)) + + + +-- Msg + + +type Msg msg + = OpenModal (Config msg) + | CloseModal + | CancelModal + | AnimationEnd + | GetWindowSize Float Float + + + +-- Types + + +type OpeningAnimation + = FromTop + | FromRight + | FromBottom + | FromLeft + + +type OpenedAnimation + = OpenFromTop + | OpenFromRight + | OpenFromBottom + | OpenFromLeft + + +type ClosingAnimation + = ToTop + | ToRight + | ToBottom + | ToLeft + + +type ClosingEffect + = WithoutAnimate + | WithAnimate + + +type BodySettings + = BodySettings + { borderRadius : Css.Px + , fromTop : Css.Px + , width : History.History Css.Px + , height : Css.Px + + -- Internal settings + , bottomClosingTop : Css.Pct + , center : Css.Pct + } + + +type History a + = History a (List a) + + +type alias Header msg = + Styled.Html msg + + +type alias Body msg = + Styled.Html msg + + +type alias Footer msg = + Styled.Html msg + + +type Config msg + = Config + { openingAnimation : OpeningAnimation + , openedAnimation : OpenedAnimation + , closingAnimation : ClosingAnimation + , closingEffect : ClosingEffect + , headerCss : String + , header : Header msg + , bodyCss : String + , body : Body msg + , footerCss : String + , footer : Footer msg + , modalBodySettings : BodySettings + , tagger : Msg msg -> msg + } + + +{-| Create a new configuration for Modal. + + newConfig |> setHeaderCss "header--bg-color" + +-} +newConfig : (Msg msg -> msg) -> Config msg +newConfig tagger = + Config + { openingAnimation = FromTop + , openedAnimation = OpenFromTop + , closingAnimation = ToTop + , closingEffect = WithAnimate + , headerCss = "" + , header = text "" + , bodyCss = "" + , body = text "" + , footerCss = "" + , footer = text "" + , modalBodySettings = + BodySettings + { borderRadius = Css.px 5 + , fromTop = Css.px 200 + , width = create (Css.px 600) + , height = Css.px 185 + , bottomClosingTop = Css.pct 65 + , center = Css.pct 33 + } + , tagger = tagger + } + + +openModal : (Msg msg -> msg) -> Config msg -> msg +openModal fn config = + fn (OpenModal config) + + +closeModal : (Msg msg -> msg) -> msg +closeModal fn = + fn CloseModal + +cancelModal : (Msg msg -> msg) -> msg +cancelModal fn = + fn CancelModal + +animationEnd : (Msg msg -> msg) -> msg +animationEnd fn = + fn AnimationEnd + + + +-- Update + + +{-| Update the component state. In your parent update function +just add + + ModalMsg modalMsg -> + let + ( updatedModal, cmdModal ) = + Modal.update modalMsg model.modal + in + case updatedModal of + Canceled -> + ( { model | modal = updatedModal } -- Model was canceled + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + Closed -> + ( { model | modal = updatedModal, confirmed = True } -- Modal was confirmed/acknowledged + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + _ -> + ( { model | modal = updatedModal } + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + +-} +update : Msg msg -> Model msg -> ( Model msg, Cmd (Msg msg) ) +update msg model = + case msg of + OpenModal config -> + ( Opening config + , cmdGetWindowSize + ) + + CloseModal -> + ( setModalState msg model + , Cmd.none + ) + + CancelModal -> + ( setModalState msg model + , Cmd.none + ) + + AnimationEnd -> + ( setModalState msg model + , Cmd.none + ) + + GetWindowSize width height -> + ( setModalBodyPosition width height model + , Cmd.none + ) + + + +-- View + + +{-| Render a view of modal window + + Modal.view model.modal + +-} +view : Model msg -> Html.Html msg +view modal = + Styled.toUnstyled <| + case modal of + Opening (Config config) -> + div [ modalFade ] + [ modalBodyView + (Just AnimationEnd) + (openingAnimationClass config.openingAnimation config.modalBodySettings) + (Config config) + ] + + Opened (Config config) -> + div [ modalFade ] + [ modalBodyView + Nothing + (openedAnimationClass config.openedAnimation config.modalBodySettings) + (Config config) + ] + + Closing (Config config) -> + div + [ modalFade + , modalClose + , closingEffectClass config.closingEffect + ] + [ modalBodyView + (Just AnimationEnd) + (closingAnimationClass config.closingAnimation config.modalBodySettings) + (Config config) + ] + + Closed -> + text "" + + Canceling (Config config) -> + div + [ modalFade + , modalClose + , closingEffectClass config.closingEffect + ] + [ modalBodyView + (Just AnimationEnd) + (closingAnimationClass config.closingAnimation config.modalBodySettings) + (Config config) + ] + + Canceled -> + text "" + + +{-| @priv +View of modal body +-} +modalBodyView : Maybe (Msg msg) -> Attribute msg -> Config msg -> Html msg +modalBodyView animationTagger animation (Config config) = + let + animTgr = + case animationTagger of + Just msg -> + onAnimationEnd (config.tagger msg) + + Nothing -> + css [] + in + div + [ modalBody config.modalBodySettings + , animation + , animTgr + ] + [ div [ class config.headerCss ] + [ config.header ] + , div [ class config.bodyCss ] + [ config.body ] + , div [ class config.footerCss ] + [ config.footer ] + ] + + + +-- Setters for Config + + +{-| Set styles as css of modal's body + + Modal.setBodyCss "body--bg-color" config + +-} +setBodyCss : String -> Config msg -> Config msg +setBodyCss newBodyCss config = + let + fn (Config c) = + Config { c | bodyCss = newBodyCss } + in + mapConfig fn config + + +{-| Set styles and msg to the body of modal + + bodySuccess : Html msg + bodySuccess = + div [][ text "Hello from body modal." ] + + Modal.setBody bodySuccess config + +-} +setBody : VirtualDom.Node msg -> Config msg -> Config msg +setBody newBody config = + let + fn (Config c) = + Config { c | body = Styled.fromUnstyled newBody } + in + mapConfig fn config + + +setHeaderCss : String -> Config msg -> Config msg +setHeaderCss newHeaderCss config = + let + fn (Config c) = + Config { c | headerCss = newHeaderCss } + in + mapConfig fn config + + +setHeader : VirtualDom.Node msg -> Config msg -> Config msg +setHeader newHeader config = + let + fn (Config c) = + Config { c | header = Styled.fromUnstyled newHeader } + in + mapConfig fn config + + +setFooterCss : String -> Config msg -> Config msg +setFooterCss newFooterCss config = + let + fn (Config c) = + Config { c | footerCss = newFooterCss } + in + mapConfig fn config + + +setFooter : VirtualDom.Node msg -> Config msg -> Config msg +setFooter newFooter config = + let + fn (Config c) = + Config { c | footer = Styled.fromUnstyled newFooter } + in + mapConfig fn config + + +setOpeningAnimation : OpeningAnimation -> Config msg -> Config msg +setOpeningAnimation opening config = + let + fn (Config c) = + Config { c | openingAnimation = opening } + in + mapConfig fn config + + +setOpenedAnimation : OpenedAnimation -> Config msg -> Config msg +setOpenedAnimation opened config = + let + fn (Config c) = + Config { c | openedAnimation = opened } + in + mapConfig fn config + + +setClosingAnimation : ClosingAnimation -> Config msg -> Config msg +setClosingAnimation closing config = + let + fn (Config c) = + Config { c | closingAnimation = closing } + in + mapConfig fn config + + +setClosingEffect : ClosingEffect -> Config msg -> Config msg +setClosingEffect newClosingEffect config = + let + fn (Config c) = + Config { c | closingEffect = newClosingEffect } + in + mapConfig fn config + + +setBodyWidth : Float -> Config msg -> Config msg +setBodyWidth width config = + let + setWidth : Float -> BodySettings -> BodySettings + setWidth newWidth (BodySettings bs) = + BodySettings { bs | width = create (Css.px newWidth) } + in + setBodySettings (setWidth width) config + + +setBodyHeight : Float -> Config msg -> Config msg +setBodyHeight height config = + let + setHeight : Float -> BodySettings -> BodySettings + setHeight newHeight (BodySettings bs) = + BodySettings { bs | height = Css.px newHeight } + in + setBodySettings (setHeight height) config + + +setBodyFromTop : Float -> Config msg -> Config msg +setBodyFromTop top config = + let + setTop : Float -> BodySettings -> BodySettings + setTop newTop (BodySettings bs) = + BodySettings { bs | fromTop = Css.px newTop } + in + setBodySettings (setTop top) config + + +setBodyBorderRadius : Float -> Config msg -> Config msg +setBodyBorderRadius borderRad config = + let + setBorderRadius : Float -> BodySettings -> BodySettings + setBorderRadius newBorderRad (BodySettings bs) = + BodySettings { bs | borderRadius = Css.px newBorderRad } + in + setBodySettings (setBorderRadius borderRad) config + + +setBodySettings : (BodySettings -> BodySettings) -> Config msg -> Config msg +setBodySettings updateFn (Config c) = + Config { c | modalBodySettings = updateFn c.modalBodySettings } + + + +-- Private setters for modal window + + +openingAnimationClass : OpeningAnimation -> BodySettings -> Attribute msg +openingAnimationClass animation bodySettings = + case animation of + FromTop -> + modalTopOpening bodySettings + + FromRight -> + modalRigthOpening bodySettings + + FromBottom -> + modalBottomOpening bodySettings + + FromLeft -> + modalLeftOpening bodySettings + + +openedAnimationClass : OpenedAnimation -> BodySettings -> Attribute msg +openedAnimationClass animation bodySettings = + case animation of + OpenFromTop -> + modalOpenFromTop bodySettings + + OpenFromRight -> + modalOpenFromRigth bodySettings + + OpenFromBottom -> + modalOpenFromBottom bodySettings + + OpenFromLeft -> + modalOpenFromLeft bodySettings + + +closingAnimationClass : ClosingAnimation -> BodySettings -> Attribute msg +closingAnimationClass animation bodySettings = + case animation of + ToTop -> + modalTopClosing bodySettings + + ToRight -> + modalRightClosing bodySettings + + ToBottom -> + modalBottomClosing bodySettings + + ToLeft -> + modalLeftClosing bodySettings + + +closingEffectClass : ClosingEffect -> Attribute msg +closingEffectClass animation = + case animation of + WithoutAnimate -> + css [ Css.display Css.none ] + + WithAnimate -> + css [] + + + +-- Private css style and animations for modal window + + +modalFade : Attribute msg +modalFade = + css + [ Css.position Css.absolute + , Css.top (Css.px 0) + , Css.bottom (Css.px 0) + , Css.left (Css.px 0) + , Css.right (Css.px 0) + , Css.backgroundColor (Css.rgba 0 0 0 0.5) + , Css.zIndex (Css.int 999) + , Css.overflow Css.hidden + ] + + +modalBody : BodySettings -> Attribute msg +modalBody (BodySettings body) = + css + [ Css.displayFlex + , Css.flexDirection Css.column + , Css.alignItems Css.center + , Css.property "align-content" "space-between" + , Css.width (current body.width) + , Css.minHeight body.height + , Css.borderRadius body.borderRadius + , Css.boxShadow4 (Css.px 0) (Css.px 3) (Css.px 7) (Css.rgba 0 0 0 0.75) + , Css.backgroundColor (Css.rgb 255 255 255) + ] + + + +-- Opening animations + + +modalTopOpening : BodySettings -> Attribute msg +modalTopOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "0" + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalRigthOpening : BodySettings -> Attribute msg +modalRigthOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "right" "0" + , Animations.property "top" body.fromTop.value + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "right" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "right" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalBottomOpening : BodySettings -> Attribute msg +modalBottomOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "75%" + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalLeftOpening : BodySettings -> Attribute msg +modalLeftOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" "0" + , Animations.property "top" body.fromTop.value + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + + +-- Open animations + + +modalOpenFromTop : BodySettings -> Attribute msg +modalOpenFromTop (BodySettings body) = + css + [ --Css.animationName + -- (Animations.keyframes + -- [ ( 0 + -- , [ Animations.property "opacity" "0" + -- , Animations.property "left" body.center.value + -- , Animations.property "top" "0" + -- ] + -- ) + -- , ( 75 + -- , [ Animations.property "opacity" "1" + -- , Animations.property "left" body.center.value + -- , Animations.property "top" body.fromTop.value + -- ] + -- ) + -- ] + -- ) + -- , Css.property "animation-duration" "1s" + Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromRigth : BodySettings -> Attribute msg +modalOpenFromRigth (BodySettings body) = + css + [ Css.property "right" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromBottom : BodySettings -> Attribute msg +modalOpenFromBottom (BodySettings body) = + css + [ Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromLeft : BodySettings -> Attribute msg +modalOpenFromLeft (BodySettings body) = + css + [ Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + + +-- Closing animations + + +modalTopClosing : BodySettings -> Attribute msg +modalTopClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "20px" + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.property "left" body.center.value + , Css.top (Css.px 0) + ] + + +modalRightClosing : BodySettings -> Attribute msg +modalRightClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "right" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "right" "0" + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.right (Css.px 0) + , Css.top body.fromTop + ] + + +modalBottomClosing : BodySettings -> Attribute msg +modalBottomClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" body.bottomClosingTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.property "left" body.center.value + ] + + +modalLeftClosing : BodySettings -> Attribute msg +modalLeftClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" "0" + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.left (Css.px 0) + , Css.top body.fromTop + ] + + + +-- Modal close animation + + +modalClose : Attribute msg +modalClose = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "background" "rgba(0, 0, 0, 0.5)" + ] + ) + , ( 100 + , [ Animations.property "background" "rgba(0, 0, 0, 0)" + , Animations.property "display" "none" + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.backgroundColor (Css.rgba 0 0 0 0) + ] + + + +-- Internal + + +onAnimationStart : msg -> Attribute msg +onAnimationStart msg = + on "animationstart" (Json.succeed msg) + + +onAnimationIteration : msg -> Attribute msg +onAnimationIteration msg = + on "animationiteration" (Json.succeed msg) + + +onAnimationEnd : msg -> Attribute msg +onAnimationEnd msg = + on "animationend" (Json.succeed msg) + + +setModalBodyPosition : Float -> Float -> Model msg -> Model msg +setModalBodyPosition width height model = + case model of + Opening config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Opening + + Opened config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Opened + + Closing config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Closing + + Closed -> + Closed + + Canceling config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Canceling + + Canceled -> + Canceled + + + +-- Private helpers + + +{-| @priv +Helper for update config +-} +mapConfig : (Config msg -> Config msg) -> Config msg -> Config msg +mapConfig fn config = + fn config + + +{-| @priv +Helper for update function +-} +setModalState : Msg msg -> Model msg -> Model msg +setModalState msg modal = + case modal of + Opening config -> + Opened config + + Opened config -> + if msg == CancelModal then + Canceling config + + else + Closing config + + Closing _ -> + Closed + + Closed -> + Closed + + Canceling _ -> + Canceled + + Canceled -> + Canceled + +{-| @priv +Centering and adaptive width for modal body +-} +recalcBodyModalProperties : Float -> Float -> BodySettings -> BodySettings +recalcBodyModalProperties windowWidth windowHeight (BodySettings bs) = + let + defaultBodyWidth = + bs.width + |> rewindAll + |> current + |> .numericValue + + setB = + 100 * (1 - bs.height.numericValue / (0.8 * windowHeight)) + + setC = + if defaultBodyWidth >= windowWidth then + 0 + + else + 50 * (1 - defaultBodyWidth / windowWidth) + + setW = + if defaultBodyWidth < windowWidth then + bs.width |> rewindAll + + else + bs.width |> forward (Css.px windowWidth) + in + BodySettings + { bs + | bottomClosingTop = Css.pct setB + , center = Css.pct setC + , width = setW + } From 7c5c78a86e08fbb951775ab416b73076f0e22b0f Mon Sep 17 00:00:00 2001 From: anotherhale Date: Mon, 11 Jan 2021 13:01:20 -0800 Subject: [PATCH 4/6] updated eol character --- src/Modal.elm | 2086 ++++++++++++++++++++++++------------------------- 1 file changed, 1043 insertions(+), 1043 deletions(-) diff --git a/src/Modal.elm b/src/Modal.elm index b864b39..c998b99 100755 --- a/src/Modal.elm +++ b/src/Modal.elm @@ -1,1043 +1,1043 @@ -module Modal exposing - ( ClosingAnimation(..) - , ClosingEffect(..) - , Config - , Model(..) - , Msg - , OpenedAnimation(..) - , OpeningAnimation(..) - , animationEnd - , closeModal - , cancelModal - , cmdGetWindowSize - , initModel - , newConfig - , openModal - , setBody - , setBodyBorderRadius - , setBodyCss - , setBodyFromTop - , setBodyHeight - , setBodyWidth - , setClosingAnimation - , setClosingEffect - , setFooter - , setFooterCss - , setHeader - , setHeaderCss - , setOpenedAnimation - , setOpeningAnimation - , subscriptions - , update - , view - ) - -import Browser.Dom exposing (getViewport) -import Browser.Events exposing (onResize) -import Css -import Css.Animations as Animations -import Css.Transitions as Transitions -import History exposing (History(..), create, current, forward, rewind, rewindAll) -import Html -import Html.Styled as Styled exposing (..) -import Html.Styled.Attributes exposing (..) -import Html.Styled.Events exposing (on, onClick) -import Json.Decode as Json -import Task -import VirtualDom - - - --- Model - - -type Model msg - = Opening (Config msg) - | Opened (Config msg) - | Closing (Config msg) - | Closed - | Canceling (Config msg) - | Canceled - - -initModel : Model msg -initModel = - Closed - - - --- Commands - - -cmdGetWindowSize : Cmd (Msg msg) -cmdGetWindowSize = - Task.perform - (\{ viewport } -> - GetWindowSize viewport.width viewport.height - ) - getViewport - - - --- Subscriptions - - -subscriptions : Sub (Msg msg) -subscriptions = - onResize (\x y -> GetWindowSize (toFloat x) (toFloat y)) - - - --- Msg - - -type Msg msg - = OpenModal (Config msg) - | CloseModal - | CancelModal - | AnimationEnd - | GetWindowSize Float Float - - - --- Types - - -type OpeningAnimation - = FromTop - | FromRight - | FromBottom - | FromLeft - - -type OpenedAnimation - = OpenFromTop - | OpenFromRight - | OpenFromBottom - | OpenFromLeft - - -type ClosingAnimation - = ToTop - | ToRight - | ToBottom - | ToLeft - - -type ClosingEffect - = WithoutAnimate - | WithAnimate - - -type BodySettings - = BodySettings - { borderRadius : Css.Px - , fromTop : Css.Px - , width : History.History Css.Px - , height : Css.Px - - -- Internal settings - , bottomClosingTop : Css.Pct - , center : Css.Pct - } - - -type History a - = History a (List a) - - -type alias Header msg = - Styled.Html msg - - -type alias Body msg = - Styled.Html msg - - -type alias Footer msg = - Styled.Html msg - - -type Config msg - = Config - { openingAnimation : OpeningAnimation - , openedAnimation : OpenedAnimation - , closingAnimation : ClosingAnimation - , closingEffect : ClosingEffect - , headerCss : String - , header : Header msg - , bodyCss : String - , body : Body msg - , footerCss : String - , footer : Footer msg - , modalBodySettings : BodySettings - , tagger : Msg msg -> msg - } - - -{-| Create a new configuration for Modal. - - newConfig |> setHeaderCss "header--bg-color" - --} -newConfig : (Msg msg -> msg) -> Config msg -newConfig tagger = - Config - { openingAnimation = FromTop - , openedAnimation = OpenFromTop - , closingAnimation = ToTop - , closingEffect = WithAnimate - , headerCss = "" - , header = text "" - , bodyCss = "" - , body = text "" - , footerCss = "" - , footer = text "" - , modalBodySettings = - BodySettings - { borderRadius = Css.px 5 - , fromTop = Css.px 200 - , width = create (Css.px 600) - , height = Css.px 185 - , bottomClosingTop = Css.pct 65 - , center = Css.pct 33 - } - , tagger = tagger - } - - -openModal : (Msg msg -> msg) -> Config msg -> msg -openModal fn config = - fn (OpenModal config) - - -closeModal : (Msg msg -> msg) -> msg -closeModal fn = - fn CloseModal - -cancelModal : (Msg msg -> msg) -> msg -cancelModal fn = - fn CancelModal - -animationEnd : (Msg msg -> msg) -> msg -animationEnd fn = - fn AnimationEnd - - - --- Update - - -{-| Update the component state. In your parent update function -just add - - ModalMsg modalMsg -> - let - ( updatedModal, cmdModal ) = - Modal.update modalMsg model.modal - in - case updatedModal of - Canceled -> - ( { model | modal = updatedModal } -- Model was canceled - , Cmd.batch [Cmd.map ModalMsg cmdModal] - ) - Closed -> - ( { model | modal = updatedModal, confirmed = True } -- Modal was confirmed/acknowledged - , Cmd.batch [Cmd.map ModalMsg cmdModal] - ) - _ -> - ( { model | modal = updatedModal } - , Cmd.batch [Cmd.map ModalMsg cmdModal] - ) - --} -update : Msg msg -> Model msg -> ( Model msg, Cmd (Msg msg) ) -update msg model = - case msg of - OpenModal config -> - ( Opening config - , cmdGetWindowSize - ) - - CloseModal -> - ( setModalState msg model - , Cmd.none - ) - - CancelModal -> - ( setModalState msg model - , Cmd.none - ) - - AnimationEnd -> - ( setModalState msg model - , Cmd.none - ) - - GetWindowSize width height -> - ( setModalBodyPosition width height model - , Cmd.none - ) - - - --- View - - -{-| Render a view of modal window - - Modal.view model.modal - --} -view : Model msg -> Html.Html msg -view modal = - Styled.toUnstyled <| - case modal of - Opening (Config config) -> - div [ modalFade ] - [ modalBodyView - (Just AnimationEnd) - (openingAnimationClass config.openingAnimation config.modalBodySettings) - (Config config) - ] - - Opened (Config config) -> - div [ modalFade ] - [ modalBodyView - Nothing - (openedAnimationClass config.openedAnimation config.modalBodySettings) - (Config config) - ] - - Closing (Config config) -> - div - [ modalFade - , modalClose - , closingEffectClass config.closingEffect - ] - [ modalBodyView - (Just AnimationEnd) - (closingAnimationClass config.closingAnimation config.modalBodySettings) - (Config config) - ] - - Closed -> - text "" - - Canceling (Config config) -> - div - [ modalFade - , modalClose - , closingEffectClass config.closingEffect - ] - [ modalBodyView - (Just AnimationEnd) - (closingAnimationClass config.closingAnimation config.modalBodySettings) - (Config config) - ] - - Canceled -> - text "" - - -{-| @priv -View of modal body --} -modalBodyView : Maybe (Msg msg) -> Attribute msg -> Config msg -> Html msg -modalBodyView animationTagger animation (Config config) = - let - animTgr = - case animationTagger of - Just msg -> - onAnimationEnd (config.tagger msg) - - Nothing -> - css [] - in - div - [ modalBody config.modalBodySettings - , animation - , animTgr - ] - [ div [ class config.headerCss ] - [ config.header ] - , div [ class config.bodyCss ] - [ config.body ] - , div [ class config.footerCss ] - [ config.footer ] - ] - - - --- Setters for Config - - -{-| Set styles as css of modal's body - - Modal.setBodyCss "body--bg-color" config - --} -setBodyCss : String -> Config msg -> Config msg -setBodyCss newBodyCss config = - let - fn (Config c) = - Config { c | bodyCss = newBodyCss } - in - mapConfig fn config - - -{-| Set styles and msg to the body of modal - - bodySuccess : Html msg - bodySuccess = - div [][ text "Hello from body modal." ] - - Modal.setBody bodySuccess config - --} -setBody : VirtualDom.Node msg -> Config msg -> Config msg -setBody newBody config = - let - fn (Config c) = - Config { c | body = Styled.fromUnstyled newBody } - in - mapConfig fn config - - -setHeaderCss : String -> Config msg -> Config msg -setHeaderCss newHeaderCss config = - let - fn (Config c) = - Config { c | headerCss = newHeaderCss } - in - mapConfig fn config - - -setHeader : VirtualDom.Node msg -> Config msg -> Config msg -setHeader newHeader config = - let - fn (Config c) = - Config { c | header = Styled.fromUnstyled newHeader } - in - mapConfig fn config - - -setFooterCss : String -> Config msg -> Config msg -setFooterCss newFooterCss config = - let - fn (Config c) = - Config { c | footerCss = newFooterCss } - in - mapConfig fn config - - -setFooter : VirtualDom.Node msg -> Config msg -> Config msg -setFooter newFooter config = - let - fn (Config c) = - Config { c | footer = Styled.fromUnstyled newFooter } - in - mapConfig fn config - - -setOpeningAnimation : OpeningAnimation -> Config msg -> Config msg -setOpeningAnimation opening config = - let - fn (Config c) = - Config { c | openingAnimation = opening } - in - mapConfig fn config - - -setOpenedAnimation : OpenedAnimation -> Config msg -> Config msg -setOpenedAnimation opened config = - let - fn (Config c) = - Config { c | openedAnimation = opened } - in - mapConfig fn config - - -setClosingAnimation : ClosingAnimation -> Config msg -> Config msg -setClosingAnimation closing config = - let - fn (Config c) = - Config { c | closingAnimation = closing } - in - mapConfig fn config - - -setClosingEffect : ClosingEffect -> Config msg -> Config msg -setClosingEffect newClosingEffect config = - let - fn (Config c) = - Config { c | closingEffect = newClosingEffect } - in - mapConfig fn config - - -setBodyWidth : Float -> Config msg -> Config msg -setBodyWidth width config = - let - setWidth : Float -> BodySettings -> BodySettings - setWidth newWidth (BodySettings bs) = - BodySettings { bs | width = create (Css.px newWidth) } - in - setBodySettings (setWidth width) config - - -setBodyHeight : Float -> Config msg -> Config msg -setBodyHeight height config = - let - setHeight : Float -> BodySettings -> BodySettings - setHeight newHeight (BodySettings bs) = - BodySettings { bs | height = Css.px newHeight } - in - setBodySettings (setHeight height) config - - -setBodyFromTop : Float -> Config msg -> Config msg -setBodyFromTop top config = - let - setTop : Float -> BodySettings -> BodySettings - setTop newTop (BodySettings bs) = - BodySettings { bs | fromTop = Css.px newTop } - in - setBodySettings (setTop top) config - - -setBodyBorderRadius : Float -> Config msg -> Config msg -setBodyBorderRadius borderRad config = - let - setBorderRadius : Float -> BodySettings -> BodySettings - setBorderRadius newBorderRad (BodySettings bs) = - BodySettings { bs | borderRadius = Css.px newBorderRad } - in - setBodySettings (setBorderRadius borderRad) config - - -setBodySettings : (BodySettings -> BodySettings) -> Config msg -> Config msg -setBodySettings updateFn (Config c) = - Config { c | modalBodySettings = updateFn c.modalBodySettings } - - - --- Private setters for modal window - - -openingAnimationClass : OpeningAnimation -> BodySettings -> Attribute msg -openingAnimationClass animation bodySettings = - case animation of - FromTop -> - modalTopOpening bodySettings - - FromRight -> - modalRigthOpening bodySettings - - FromBottom -> - modalBottomOpening bodySettings - - FromLeft -> - modalLeftOpening bodySettings - - -openedAnimationClass : OpenedAnimation -> BodySettings -> Attribute msg -openedAnimationClass animation bodySettings = - case animation of - OpenFromTop -> - modalOpenFromTop bodySettings - - OpenFromRight -> - modalOpenFromRigth bodySettings - - OpenFromBottom -> - modalOpenFromBottom bodySettings - - OpenFromLeft -> - modalOpenFromLeft bodySettings - - -closingAnimationClass : ClosingAnimation -> BodySettings -> Attribute msg -closingAnimationClass animation bodySettings = - case animation of - ToTop -> - modalTopClosing bodySettings - - ToRight -> - modalRightClosing bodySettings - - ToBottom -> - modalBottomClosing bodySettings - - ToLeft -> - modalLeftClosing bodySettings - - -closingEffectClass : ClosingEffect -> Attribute msg -closingEffectClass animation = - case animation of - WithoutAnimate -> - css [ Css.display Css.none ] - - WithAnimate -> - css [] - - - --- Private css style and animations for modal window - - -modalFade : Attribute msg -modalFade = - css - [ Css.position Css.absolute - , Css.top (Css.px 0) - , Css.bottom (Css.px 0) - , Css.left (Css.px 0) - , Css.right (Css.px 0) - , Css.backgroundColor (Css.rgba 0 0 0 0.5) - , Css.zIndex (Css.int 999) - , Css.overflow Css.hidden - ] - - -modalBody : BodySettings -> Attribute msg -modalBody (BodySettings body) = - css - [ Css.displayFlex - , Css.flexDirection Css.column - , Css.alignItems Css.center - , Css.property "align-content" "space-between" - , Css.width (current body.width) - , Css.minHeight body.height - , Css.borderRadius body.borderRadius - , Css.boxShadow4 (Css.px 0) (Css.px 3) (Css.px 7) (Css.rgba 0 0 0 0.75) - , Css.backgroundColor (Css.rgb 255 255 255) - ] - - - --- Opening animations - - -modalTopOpening : BodySettings -> Attribute msg -modalTopOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "0" - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalRigthOpening : BodySettings -> Attribute msg -modalRigthOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "right" "0" - , Animations.property "top" body.fromTop.value - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "right" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "right" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalBottomOpening : BodySettings -> Attribute msg -modalBottomOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "75%" - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalLeftOpening : BodySettings -> Attribute msg -modalLeftOpening (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "0" - , Animations.property "left" "0" - , Animations.property "top" body.fromTop.value - ] - ) - , ( 75 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - - --- Open animations - - -modalOpenFromTop : BodySettings -> Attribute msg -modalOpenFromTop (BodySettings body) = - css - [ --Css.animationName - -- (Animations.keyframes - -- [ ( 0 - -- , [ Animations.property "opacity" "0" - -- , Animations.property "left" body.center.value - -- , Animations.property "top" "0" - -- ] - -- ) - -- , ( 75 - -- , [ Animations.property "opacity" "1" - -- , Animations.property "left" body.center.value - -- , Animations.property "top" body.fromTop.value - -- ] - -- ) - -- ] - -- ) - -- , Css.property "animation-duration" "1s" - Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromRigth : BodySettings -> Attribute msg -modalOpenFromRigth (BodySettings body) = - css - [ Css.property "right" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromBottom : BodySettings -> Attribute msg -modalOpenFromBottom (BodySettings body) = - css - [ Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - -modalOpenFromLeft : BodySettings -> Attribute msg -modalOpenFromLeft (BodySettings body) = - css - [ Css.property "left" body.center.value - , Css.top body.fromTop - , Css.position Css.absolute - ] - - - --- Closing animations - - -modalTopClosing : BodySettings -> Attribute msg -modalTopClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" "20px" - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.property "left" body.center.value - , Css.top (Css.px 0) - ] - - -modalRightClosing : BodySettings -> Attribute msg -modalRightClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "right" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "right" "0" - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.right (Css.px 0) - , Css.top body.fromTop - ] - - -modalBottomClosing : BodySettings -> Attribute msg -modalBottomClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value - , Animations.property "top" body.bottomClosingTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.property "left" body.center.value - ] - - -modalLeftClosing : BodySettings -> Attribute msg -modalLeftClosing (BodySettings body) = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value - ] - ) - , ( 100 - , [ Animations.property "opacity" "0" - , Animations.property "left" "0" - , Animations.property "top" body.fromTop.value - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.opacity (Css.int 0) - , Css.position Css.absolute - , Css.left (Css.px 0) - , Css.top body.fromTop - ] - - - --- Modal close animation - - -modalClose : Attribute msg -modalClose = - css - [ Css.animationName - (Animations.keyframes - [ ( 0 - , [ Animations.property "background" "rgba(0, 0, 0, 0.5)" - ] - ) - , ( 100 - , [ Animations.property "background" "rgba(0, 0, 0, 0)" - , Animations.property "display" "none" - ] - ) - ] - ) - , Css.property "animation-duration" "0.5s" - , Css.backgroundColor (Css.rgba 0 0 0 0) - ] - - - --- Internal - - -onAnimationStart : msg -> Attribute msg -onAnimationStart msg = - on "animationstart" (Json.succeed msg) - - -onAnimationIteration : msg -> Attribute msg -onAnimationIteration msg = - on "animationiteration" (Json.succeed msg) - - -onAnimationEnd : msg -> Attribute msg -onAnimationEnd msg = - on "animationend" (Json.succeed msg) - - -setModalBodyPosition : Float -> Float -> Model msg -> Model msg -setModalBodyPosition width height model = - case model of - Opening config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Opening - - Opened config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Opened - - Closing config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Closing - - Closed -> - Closed - - Canceling config -> - config - |> setBodySettings (recalcBodyModalProperties width height) - |> Canceling - - Canceled -> - Canceled - - - --- Private helpers - - -{-| @priv -Helper for update config --} -mapConfig : (Config msg -> Config msg) -> Config msg -> Config msg -mapConfig fn config = - fn config - - -{-| @priv -Helper for update function --} -setModalState : Msg msg -> Model msg -> Model msg -setModalState msg modal = - case modal of - Opening config -> - Opened config - - Opened config -> - if msg == CancelModal then - Canceling config - - else - Closing config - - Closing _ -> - Closed - - Closed -> - Closed - - Canceling _ -> - Canceled - - Canceled -> - Canceled - -{-| @priv -Centering and adaptive width for modal body --} -recalcBodyModalProperties : Float -> Float -> BodySettings -> BodySettings -recalcBodyModalProperties windowWidth windowHeight (BodySettings bs) = - let - defaultBodyWidth = - bs.width - |> rewindAll - |> current - |> .numericValue - - setB = - 100 * (1 - bs.height.numericValue / (0.8 * windowHeight)) - - setC = - if defaultBodyWidth >= windowWidth then - 0 - - else - 50 * (1 - defaultBodyWidth / windowWidth) - - setW = - if defaultBodyWidth < windowWidth then - bs.width |> rewindAll - - else - bs.width |> forward (Css.px windowWidth) - in - BodySettings - { bs - | bottomClosingTop = Css.pct setB - , center = Css.pct setC - , width = setW - } +module Modal exposing + ( ClosingAnimation(..) + , ClosingEffect(..) + , Config + , Model(..) + , Msg + , OpenedAnimation(..) + , OpeningAnimation(..) + , animationEnd + , closeModal + , cancelModal + , cmdGetWindowSize + , initModel + , newConfig + , openModal + , setBody + , setBodyBorderRadius + , setBodyCss + , setBodyFromTop + , setBodyHeight + , setBodyWidth + , setClosingAnimation + , setClosingEffect + , setFooter + , setFooterCss + , setHeader + , setHeaderCss + , setOpenedAnimation + , setOpeningAnimation + , subscriptions + , update + , view + ) + +import Browser.Dom exposing (getViewport) +import Browser.Events exposing (onResize) +import Css +import Css.Animations as Animations +import Css.Transitions as Transitions +import History exposing (History(..), create, current, forward, rewind, rewindAll) +import Html +import Html.Styled as Styled exposing (..) +import Html.Styled.Attributes exposing (..) +import Html.Styled.Events exposing (on, onClick) +import Json.Decode as Json +import Task +import VirtualDom + + + +-- Model + + +type Model msg + = Opening (Config msg) + | Opened (Config msg) + | Closing (Config msg) + | Closed + | Canceling (Config msg) + | Canceled + + +initModel : Model msg +initModel = + Closed + + + +-- Commands + + +cmdGetWindowSize : Cmd (Msg msg) +cmdGetWindowSize = + Task.perform + (\{ viewport } -> + GetWindowSize viewport.width viewport.height + ) + getViewport + + + +-- Subscriptions + + +subscriptions : Sub (Msg msg) +subscriptions = + onResize (\x y -> GetWindowSize (toFloat x) (toFloat y)) + + + +-- Msg + + +type Msg msg + = OpenModal (Config msg) + | CloseModal + | CancelModal + | AnimationEnd + | GetWindowSize Float Float + + + +-- Types + + +type OpeningAnimation + = FromTop + | FromRight + | FromBottom + | FromLeft + + +type OpenedAnimation + = OpenFromTop + | OpenFromRight + | OpenFromBottom + | OpenFromLeft + + +type ClosingAnimation + = ToTop + | ToRight + | ToBottom + | ToLeft + + +type ClosingEffect + = WithoutAnimate + | WithAnimate + + +type BodySettings + = BodySettings + { borderRadius : Css.Px + , fromTop : Css.Px + , width : History.History Css.Px + , height : Css.Px + + -- Internal settings + , bottomClosingTop : Css.Pct + , center : Css.Pct + } + + +type History a + = History a (List a) + + +type alias Header msg = + Styled.Html msg + + +type alias Body msg = + Styled.Html msg + + +type alias Footer msg = + Styled.Html msg + + +type Config msg + = Config + { openingAnimation : OpeningAnimation + , openedAnimation : OpenedAnimation + , closingAnimation : ClosingAnimation + , closingEffect : ClosingEffect + , headerCss : String + , header : Header msg + , bodyCss : String + , body : Body msg + , footerCss : String + , footer : Footer msg + , modalBodySettings : BodySettings + , tagger : Msg msg -> msg + } + + +{-| Create a new configuration for Modal. + + newConfig |> setHeaderCss "header--bg-color" + +-} +newConfig : (Msg msg -> msg) -> Config msg +newConfig tagger = + Config + { openingAnimation = FromTop + , openedAnimation = OpenFromTop + , closingAnimation = ToTop + , closingEffect = WithAnimate + , headerCss = "" + , header = text "" + , bodyCss = "" + , body = text "" + , footerCss = "" + , footer = text "" + , modalBodySettings = + BodySettings + { borderRadius = Css.px 5 + , fromTop = Css.px 200 + , width = create (Css.px 600) + , height = Css.px 185 + , bottomClosingTop = Css.pct 65 + , center = Css.pct 33 + } + , tagger = tagger + } + + +openModal : (Msg msg -> msg) -> Config msg -> msg +openModal fn config = + fn (OpenModal config) + + +closeModal : (Msg msg -> msg) -> msg +closeModal fn = + fn CloseModal + +cancelModal : (Msg msg -> msg) -> msg +cancelModal fn = + fn CancelModal + +animationEnd : (Msg msg -> msg) -> msg +animationEnd fn = + fn AnimationEnd + + + +-- Update + + +{-| Update the component state. In your parent update function +just add + + ModalMsg modalMsg -> + let + ( updatedModal, cmdModal ) = + Modal.update modalMsg model.modal + in + case updatedModal of + Canceled -> + ( { model | modal = updatedModal } -- Model was canceled + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + Closed -> + ( { model | modal = updatedModal, confirmed = True } -- Modal was confirmed/acknowledged + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + _ -> + ( { model | modal = updatedModal } + , Cmd.batch [Cmd.map ModalMsg cmdModal] + ) + +-} +update : Msg msg -> Model msg -> ( Model msg, Cmd (Msg msg) ) +update msg model = + case msg of + OpenModal config -> + ( Opening config + , cmdGetWindowSize + ) + + CloseModal -> + ( setModalState msg model + , Cmd.none + ) + + CancelModal -> + ( setModalState msg model + , Cmd.none + ) + + AnimationEnd -> + ( setModalState msg model + , Cmd.none + ) + + GetWindowSize width height -> + ( setModalBodyPosition width height model + , Cmd.none + ) + + + +-- View + + +{-| Render a view of modal window + + Modal.view model.modal + +-} +view : Model msg -> Html.Html msg +view modal = + Styled.toUnstyled <| + case modal of + Opening (Config config) -> + div [ modalFade ] + [ modalBodyView + (Just AnimationEnd) + (openingAnimationClass config.openingAnimation config.modalBodySettings) + (Config config) + ] + + Opened (Config config) -> + div [ modalFade ] + [ modalBodyView + Nothing + (openedAnimationClass config.openedAnimation config.modalBodySettings) + (Config config) + ] + + Closing (Config config) -> + div + [ modalFade + , modalClose + , closingEffectClass config.closingEffect + ] + [ modalBodyView + (Just AnimationEnd) + (closingAnimationClass config.closingAnimation config.modalBodySettings) + (Config config) + ] + + Closed -> + text "" + + Canceling (Config config) -> + div + [ modalFade + , modalClose + , closingEffectClass config.closingEffect + ] + [ modalBodyView + (Just AnimationEnd) + (closingAnimationClass config.closingAnimation config.modalBodySettings) + (Config config) + ] + + Canceled -> + text "" + + +{-| @priv +View of modal body +-} +modalBodyView : Maybe (Msg msg) -> Attribute msg -> Config msg -> Html msg +modalBodyView animationTagger animation (Config config) = + let + animTgr = + case animationTagger of + Just msg -> + onAnimationEnd (config.tagger msg) + + Nothing -> + css [] + in + div + [ modalBody config.modalBodySettings + , animation + , animTgr + ] + [ div [ class config.headerCss ] + [ config.header ] + , div [ class config.bodyCss ] + [ config.body ] + , div [ class config.footerCss ] + [ config.footer ] + ] + + + +-- Setters for Config + + +{-| Set styles as css of modal's body + + Modal.setBodyCss "body--bg-color" config + +-} +setBodyCss : String -> Config msg -> Config msg +setBodyCss newBodyCss config = + let + fn (Config c) = + Config { c | bodyCss = newBodyCss } + in + mapConfig fn config + + +{-| Set styles and msg to the body of modal + + bodySuccess : Html msg + bodySuccess = + div [][ text "Hello from body modal." ] + + Modal.setBody bodySuccess config + +-} +setBody : VirtualDom.Node msg -> Config msg -> Config msg +setBody newBody config = + let + fn (Config c) = + Config { c | body = Styled.fromUnstyled newBody } + in + mapConfig fn config + + +setHeaderCss : String -> Config msg -> Config msg +setHeaderCss newHeaderCss config = + let + fn (Config c) = + Config { c | headerCss = newHeaderCss } + in + mapConfig fn config + + +setHeader : VirtualDom.Node msg -> Config msg -> Config msg +setHeader newHeader config = + let + fn (Config c) = + Config { c | header = Styled.fromUnstyled newHeader } + in + mapConfig fn config + + +setFooterCss : String -> Config msg -> Config msg +setFooterCss newFooterCss config = + let + fn (Config c) = + Config { c | footerCss = newFooterCss } + in + mapConfig fn config + + +setFooter : VirtualDom.Node msg -> Config msg -> Config msg +setFooter newFooter config = + let + fn (Config c) = + Config { c | footer = Styled.fromUnstyled newFooter } + in + mapConfig fn config + + +setOpeningAnimation : OpeningAnimation -> Config msg -> Config msg +setOpeningAnimation opening config = + let + fn (Config c) = + Config { c | openingAnimation = opening } + in + mapConfig fn config + + +setOpenedAnimation : OpenedAnimation -> Config msg -> Config msg +setOpenedAnimation opened config = + let + fn (Config c) = + Config { c | openedAnimation = opened } + in + mapConfig fn config + + +setClosingAnimation : ClosingAnimation -> Config msg -> Config msg +setClosingAnimation closing config = + let + fn (Config c) = + Config { c | closingAnimation = closing } + in + mapConfig fn config + + +setClosingEffect : ClosingEffect -> Config msg -> Config msg +setClosingEffect newClosingEffect config = + let + fn (Config c) = + Config { c | closingEffect = newClosingEffect } + in + mapConfig fn config + + +setBodyWidth : Float -> Config msg -> Config msg +setBodyWidth width config = + let + setWidth : Float -> BodySettings -> BodySettings + setWidth newWidth (BodySettings bs) = + BodySettings { bs | width = create (Css.px newWidth) } + in + setBodySettings (setWidth width) config + + +setBodyHeight : Float -> Config msg -> Config msg +setBodyHeight height config = + let + setHeight : Float -> BodySettings -> BodySettings + setHeight newHeight (BodySettings bs) = + BodySettings { bs | height = Css.px newHeight } + in + setBodySettings (setHeight height) config + + +setBodyFromTop : Float -> Config msg -> Config msg +setBodyFromTop top config = + let + setTop : Float -> BodySettings -> BodySettings + setTop newTop (BodySettings bs) = + BodySettings { bs | fromTop = Css.px newTop } + in + setBodySettings (setTop top) config + + +setBodyBorderRadius : Float -> Config msg -> Config msg +setBodyBorderRadius borderRad config = + let + setBorderRadius : Float -> BodySettings -> BodySettings + setBorderRadius newBorderRad (BodySettings bs) = + BodySettings { bs | borderRadius = Css.px newBorderRad } + in + setBodySettings (setBorderRadius borderRad) config + + +setBodySettings : (BodySettings -> BodySettings) -> Config msg -> Config msg +setBodySettings updateFn (Config c) = + Config { c | modalBodySettings = updateFn c.modalBodySettings } + + + +-- Private setters for modal window + + +openingAnimationClass : OpeningAnimation -> BodySettings -> Attribute msg +openingAnimationClass animation bodySettings = + case animation of + FromTop -> + modalTopOpening bodySettings + + FromRight -> + modalRigthOpening bodySettings + + FromBottom -> + modalBottomOpening bodySettings + + FromLeft -> + modalLeftOpening bodySettings + + +openedAnimationClass : OpenedAnimation -> BodySettings -> Attribute msg +openedAnimationClass animation bodySettings = + case animation of + OpenFromTop -> + modalOpenFromTop bodySettings + + OpenFromRight -> + modalOpenFromRigth bodySettings + + OpenFromBottom -> + modalOpenFromBottom bodySettings + + OpenFromLeft -> + modalOpenFromLeft bodySettings + + +closingAnimationClass : ClosingAnimation -> BodySettings -> Attribute msg +closingAnimationClass animation bodySettings = + case animation of + ToTop -> + modalTopClosing bodySettings + + ToRight -> + modalRightClosing bodySettings + + ToBottom -> + modalBottomClosing bodySettings + + ToLeft -> + modalLeftClosing bodySettings + + +closingEffectClass : ClosingEffect -> Attribute msg +closingEffectClass animation = + case animation of + WithoutAnimate -> + css [ Css.display Css.none ] + + WithAnimate -> + css [] + + + +-- Private css style and animations for modal window + + +modalFade : Attribute msg +modalFade = + css + [ Css.position Css.absolute + , Css.top (Css.px 0) + , Css.bottom (Css.px 0) + , Css.left (Css.px 0) + , Css.right (Css.px 0) + , Css.backgroundColor (Css.rgba 0 0 0 0.5) + , Css.zIndex (Css.int 999) + , Css.overflow Css.hidden + ] + + +modalBody : BodySettings -> Attribute msg +modalBody (BodySettings body) = + css + [ Css.displayFlex + , Css.flexDirection Css.column + , Css.alignItems Css.center + , Css.property "align-content" "space-between" + , Css.width (current body.width) + , Css.minHeight body.height + , Css.borderRadius body.borderRadius + , Css.boxShadow4 (Css.px 0) (Css.px 3) (Css.px 7) (Css.rgba 0 0 0 0.75) + , Css.backgroundColor (Css.rgb 255 255 255) + ] + + + +-- Opening animations + + +modalTopOpening : BodySettings -> Attribute msg +modalTopOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "0" + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalRigthOpening : BodySettings -> Attribute msg +modalRigthOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "right" "0" + , Animations.property "top" body.fromTop.value + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "right" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "right" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalBottomOpening : BodySettings -> Attribute msg +modalBottomOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "75%" + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalLeftOpening : BodySettings -> Attribute msg +modalLeftOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + , Animations.property "left" "0" + , Animations.property "top" body.fromTop.value + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "1s" + , Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + + +-- Open animations + + +modalOpenFromTop : BodySettings -> Attribute msg +modalOpenFromTop (BodySettings body) = + css + [ --Css.animationName + -- (Animations.keyframes + -- [ ( 0 + -- , [ Animations.property "opacity" "0" + -- , Animations.property "left" body.center.value + -- , Animations.property "top" "0" + -- ] + -- ) + -- , ( 75 + -- , [ Animations.property "opacity" "1" + -- , Animations.property "left" body.center.value + -- , Animations.property "top" body.fromTop.value + -- ] + -- ) + -- ] + -- ) + -- , Css.property "animation-duration" "1s" + Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromRigth : BodySettings -> Attribute msg +modalOpenFromRigth (BodySettings body) = + css + [ Css.property "right" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromBottom : BodySettings -> Attribute msg +modalOpenFromBottom (BodySettings body) = + css + [ Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + +modalOpenFromLeft : BodySettings -> Attribute msg +modalOpenFromLeft (BodySettings body) = + css + [ Css.property "left" body.center.value + , Css.top body.fromTop + , Css.position Css.absolute + ] + + + +-- Closing animations + + +modalTopClosing : BodySettings -> Attribute msg +modalTopClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" "20px" + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.property "left" body.center.value + , Css.top (Css.px 0) + ] + + +modalRightClosing : BodySettings -> Attribute msg +modalRightClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "right" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "right" "0" + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.right (Css.px 0) + , Css.top body.fromTop + ] + + +modalBottomClosing : BodySettings -> Attribute msg +modalBottomClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" body.center.value + , Animations.property "top" body.bottomClosingTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.property "left" body.center.value + ] + + +modalLeftClosing : BodySettings -> Attribute msg +modalLeftClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" "0" + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.left (Css.px 0) + , Css.top body.fromTop + ] + + + +-- Modal close animation + + +modalClose : Attribute msg +modalClose = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "background" "rgba(0, 0, 0, 0.5)" + ] + ) + , ( 100 + , [ Animations.property "background" "rgba(0, 0, 0, 0)" + , Animations.property "display" "none" + ] + ) + ] + ) + , Css.property "animation-duration" "0.5s" + , Css.backgroundColor (Css.rgba 0 0 0 0) + ] + + + +-- Internal + + +onAnimationStart : msg -> Attribute msg +onAnimationStart msg = + on "animationstart" (Json.succeed msg) + + +onAnimationIteration : msg -> Attribute msg +onAnimationIteration msg = + on "animationiteration" (Json.succeed msg) + + +onAnimationEnd : msg -> Attribute msg +onAnimationEnd msg = + on "animationend" (Json.succeed msg) + + +setModalBodyPosition : Float -> Float -> Model msg -> Model msg +setModalBodyPosition width height model = + case model of + Opening config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Opening + + Opened config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Opened + + Closing config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Closing + + Closed -> + Closed + + Canceling config -> + config + |> setBodySettings (recalcBodyModalProperties width height) + |> Canceling + + Canceled -> + Canceled + + + +-- Private helpers + + +{-| @priv +Helper for update config +-} +mapConfig : (Config msg -> Config msg) -> Config msg -> Config msg +mapConfig fn config = + fn config + + +{-| @priv +Helper for update function +-} +setModalState : Msg msg -> Model msg -> Model msg +setModalState msg modal = + case modal of + Opening config -> + Opened config + + Opened config -> + if msg == CancelModal then + Canceling config + + else + Closing config + + Closing _ -> + Closed + + Closed -> + Closed + + Canceling _ -> + Canceled + + Canceled -> + Canceled + +{-| @priv +Centering and adaptive width for modal body +-} +recalcBodyModalProperties : Float -> Float -> BodySettings -> BodySettings +recalcBodyModalProperties windowWidth windowHeight (BodySettings bs) = + let + defaultBodyWidth = + bs.width + |> rewindAll + |> current + |> .numericValue + + setB = + 100 * (1 - bs.height.numericValue / (0.8 * windowHeight)) + + setC = + if defaultBodyWidth >= windowWidth then + 0 + + else + 50 * (1 - defaultBodyWidth / windowWidth) + + setW = + if defaultBodyWidth < windowWidth then + bs.width |> rewindAll + + else + bs.width |> forward (Css.px windowWidth) + in + BodySettings + { bs + | bottomClosingTop = Css.pct setB + , center = Css.pct setC + , width = setW + } From 3305d9eec0c40d6545ad08c2b038ff3c2e3bb189 Mon Sep 17 00:00:00 2001 From: anotherhale Date: Fri, 15 Jan 2021 21:53:20 -0700 Subject: [PATCH 5/6] #10 - FromNone and ToNone --- src/Modal.elm | 83 ++++++++++++++++++++++++++++++++------------------- 1 file changed, 53 insertions(+), 30 deletions(-) diff --git a/src/Modal.elm b/src/Modal.elm index c998b99..3394afc 100755 --- a/src/Modal.elm +++ b/src/Modal.elm @@ -1,6 +1,5 @@ module Modal exposing ( ClosingAnimation(..) - , ClosingEffect(..) , Config , Model(..) , Msg @@ -20,7 +19,6 @@ module Modal exposing , setBodyHeight , setBodyWidth , setClosingAnimation - , setClosingEffect , setFooter , setFooterCss , setHeader @@ -108,6 +106,7 @@ type OpeningAnimation | FromRight | FromBottom | FromLeft + | FromNone type OpenedAnimation @@ -122,12 +121,8 @@ type ClosingAnimation | ToRight | ToBottom | ToLeft - - -type ClosingEffect - = WithoutAnimate - | WithAnimate - + | ToNone + type BodySettings = BodySettings @@ -163,7 +158,6 @@ type Config msg { openingAnimation : OpeningAnimation , openedAnimation : OpenedAnimation , closingAnimation : ClosingAnimation - , closingEffect : ClosingEffect , headerCss : String , header : Header msg , bodyCss : String @@ -186,7 +180,6 @@ newConfig tagger = { openingAnimation = FromTop , openedAnimation = OpenFromTop , closingAnimation = ToTop - , closingEffect = WithAnimate , headerCss = "" , header = text "" , bodyCss = "" @@ -295,7 +288,7 @@ view modal = case modal of Opening (Config config) -> div [ modalFade ] - [ modalBodyView + [ modalBodyView (Just AnimationEnd) (openingAnimationClass config.openingAnimation config.modalBodySettings) (Config config) @@ -313,7 +306,6 @@ view modal = div [ modalFade , modalClose - , closingEffectClass config.closingEffect ] [ modalBodyView (Just AnimationEnd) @@ -328,7 +320,6 @@ view modal = div [ modalFade , modalClose - , closingEffectClass config.closingEffect ] [ modalBodyView (Just AnimationEnd) @@ -467,13 +458,6 @@ setClosingAnimation closing config = mapConfig fn config -setClosingEffect : ClosingEffect -> Config msg -> Config msg -setClosingEffect newClosingEffect config = - let - fn (Config c) = - Config { c | closingEffect = newClosingEffect } - in - mapConfig fn config setBodyWidth : Float -> Config msg -> Config msg @@ -540,6 +524,9 @@ openingAnimationClass animation bodySettings = FromLeft -> modalLeftOpening bodySettings + FromNone -> + modalNoneOpening bodySettings + openedAnimationClass : OpenedAnimation -> BodySettings -> Attribute msg openedAnimationClass animation bodySettings = @@ -572,16 +559,8 @@ closingAnimationClass animation bodySettings = ToLeft -> modalLeftClosing bodySettings - -closingEffectClass : ClosingEffect -> Attribute msg -closingEffectClass animation = - case animation of - WithoutAnimate -> - css [ Css.display Css.none ] - - WithAnimate -> - css [] - + ToNone -> + modalNoneClosing bodySettings -- Private css style and animations for modal window @@ -724,6 +703,25 @@ modalLeftOpening (BodySettings body) = ] +modalNoneOpening : BodySettings -> Attribute msg +modalNoneOpening (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "0" + ] + ) + , ( 75 + , [ Animations.property "opacity" "1" + ] + ) + ] + ) + , Css.property "animation-duration" "0s" + , Css.property "top" body.center.value + , Css.position Css.absolute + ] -- Open animations @@ -892,6 +890,31 @@ modalLeftClosing (BodySettings body) = ] +modalNoneClosing : BodySettings -> Attribute msg +modalNoneClosing (BodySettings body) = + css + [ Css.animationName + (Animations.keyframes + [ ( 0 + , [ Animations.property "opacity" "1" + , Animations.property "left" body.center.value + , Animations.property "top" body.fromTop.value + ] + ) + , ( 100 + , [ Animations.property "opacity" "0" + , Animations.property "left" "0" + , Animations.property "top" body.fromTop.value + ] + ) + ] + ) + , Css.property "animation-duration" "0.0s" + , Css.opacity (Css.int 0) + , Css.position Css.absolute + , Css.top body.fromTop + ] + -- Modal close animation From 622ded2d4760f510af45a271eb5288ead97dc137 Mon Sep 17 00:00:00 2001 From: anotherhale Date: Sat, 16 Jan 2021 09:04:28 -0700 Subject: [PATCH 6/6] ran elm-format --- src/Modal.elm | 143 +++++++++++++++++++++++++++++++++++++++++--------- 1 file changed, 118 insertions(+), 25 deletions(-) diff --git a/src/Modal.elm b/src/Modal.elm index 3394afc..3ef3a3d 100755 --- a/src/Modal.elm +++ b/src/Modal.elm @@ -1,13 +1,14 @@ module Modal exposing - ( ClosingAnimation(..) + ( BodySettings + , ClosingAnimation(..) , Config , Model(..) , Msg , OpenedAnimation(..) , OpeningAnimation(..) , animationEnd - , closeModal , cancelModal + , closeModal , cmdGetWindowSize , initModel , newConfig @@ -23,6 +24,11 @@ module Modal exposing , setFooterCss , setHeader , setHeaderCss + , setModalCloseFn + , setModalClosingFn + , setModalFadeFn + , setModalOpenedFn + , setModalOpeningFn , setOpenedAnimation , setOpeningAnimation , subscriptions @@ -122,7 +128,7 @@ type ClosingAnimation | ToBottom | ToLeft | ToNone - + type BodySettings = BodySettings @@ -158,6 +164,11 @@ type Config msg { openingAnimation : OpeningAnimation , openedAnimation : OpenedAnimation , closingAnimation : ClosingAnimation + , modalOpeningFn : Maybe (Attribute msg) + , modalOpenedFn : Maybe (Attribute msg) + , modalClosingFn : Maybe (Attribute msg) + , modalCloseFn : Maybe (Attribute msg) + , modalFadeFn : Maybe (Attribute msg) , headerCss : String , header : Header msg , bodyCss : String @@ -180,6 +191,11 @@ newConfig tagger = { openingAnimation = FromTop , openedAnimation = OpenFromTop , closingAnimation = ToTop + , modalOpeningFn = Nothing + , modalOpenedFn = Nothing + , modalClosingFn = Nothing + , modalCloseFn = Nothing + , modalFadeFn = Nothing , headerCss = "" , header = text "" , bodyCss = "" @@ -208,10 +224,12 @@ closeModal : (Msg msg -> msg) -> msg closeModal fn = fn CloseModal + cancelModal : (Msg msg -> msg) -> msg cancelModal fn = fn CancelModal + animationEnd : (Msg msg -> msg) -> msg animationEnd fn = fn AnimationEnd @@ -230,11 +248,11 @@ just add Modal.update modalMsg model.modal in case updatedModal of - Canceled -> + Canceled -> ( { model | modal = updatedModal } -- Model was canceled , Cmd.batch [Cmd.map ModalMsg cmdModal] ) - Closed -> + Closed -> ( { model | modal = updatedModal, confirmed = True } -- Modal was confirmed/acknowledged , Cmd.batch [Cmd.map ModalMsg cmdModal] ) @@ -287,29 +305,56 @@ view modal = Styled.toUnstyled <| case modal of Opening (Config config) -> - div [ modalFade ] - [ modalBodyView + let + updatefun = + case config.modalOpeningFn of + Nothing -> + openingAnimationClass config.openingAnimation config.modalBodySettings + + Just fn -> + fn + in + div [ Maybe.withDefault modalFade config.modalFadeFn ] + [ modalBodyView (Just AnimationEnd) - (openingAnimationClass config.openingAnimation config.modalBodySettings) + updatefun (Config config) ] Opened (Config config) -> - div [ modalFade ] + let + updatedOpenedFn = + case config.modalOpenedFn of + Nothing -> + openedAnimationClass config.openedAnimation config.modalBodySettings + + Just openedFn -> + openedFn + in + div [ Maybe.withDefault modalFade config.modalFadeFn ] [ modalBodyView Nothing - (openedAnimationClass config.openedAnimation config.modalBodySettings) + updatedOpenedFn (Config config) ] Closing (Config config) -> + let + updatedClosingFn = + case config.modalClosingFn of + Nothing -> + closingAnimationClass config.closingAnimation config.modalBodySettings + + Just closingFn -> + closingFn + in div - [ modalFade + [ Maybe.withDefault modalFade config.modalFadeFn , modalClose ] [ modalBodyView (Just AnimationEnd) - (closingAnimationClass config.closingAnimation config.modalBodySettings) + updatedClosingFn (Config config) ] @@ -318,12 +363,12 @@ view modal = Canceling (Config config) -> div - [ modalFade + [ Maybe.withDefault modalFade config.modalFadeFn , modalClose ] [ modalBodyView (Just AnimationEnd) - (closingAnimationClass config.closingAnimation config.modalBodySettings) + (Maybe.withDefault (closingAnimationClass config.closingAnimation config.modalBodySettings) config.modalClosingFn) (Config config) ] @@ -431,6 +476,33 @@ setFooter newFooter config = mapConfig fn config +setModalOpeningFn : Attribute msg -> Config msg -> Config msg +setModalOpeningFn opening config = + let + fn (Config c) = + Config { c | modalOpeningFn = Just opening } + in + mapConfig fn config + + +setModalOpenedFn : Attribute msg -> Config msg -> Config msg +setModalOpenedFn opened config = + let + fn (Config c) = + Config { c | modalOpenedFn = Just opened } + in + mapConfig fn config + + +setModalFadeFn : Attribute msg -> Config msg -> Config msg +setModalFadeFn fade config = + let + fn (Config c) = + Config { c | modalFadeFn = Just fade } + in + mapConfig fn config + + setOpeningAnimation : OpeningAnimation -> Config msg -> Config msg setOpeningAnimation opening config = let @@ -458,6 +530,22 @@ setClosingAnimation closing config = mapConfig fn config +setModalClosingFn : Attribute msg -> Config msg -> Config msg +setModalClosingFn closing config = + let + fn (Config c) = + Config { c | modalClosingFn = Just closing } + in + mapConfig fn config + + +setModalCloseFn : Attribute msg -> Config msg -> Config msg +setModalCloseFn close config = + let + fn (Config c) = + Config { c | modalCloseFn = Just close } + in + mapConfig fn config setBodyWidth : Float -> Config msg -> Config msg @@ -535,7 +623,7 @@ openedAnimationClass animation bodySettings = modalOpenFromTop bodySettings OpenFromRight -> - modalOpenFromRigth bodySettings + modalOpenFromRight bodySettings OpenFromBottom -> modalOpenFromBottom bodySettings @@ -563,6 +651,7 @@ closingAnimationClass animation bodySettings = modalNoneClosing bodySettings + -- Private css style and animations for modal window @@ -606,20 +695,20 @@ modalTopOpening (BodySettings body) = (Animations.keyframes [ ( 0 , [ Animations.property "opacity" "0" - , Animations.property "left" body.center.value + , Animations.property "left" (Debug.log "center1" body.center.value) , Animations.property "top" "0" ] ) , ( 75 , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value + , Animations.property "left" (Debug.log "center2" body.center.value) + , Animations.property "top" (Debug.log "top1" body.fromTop.value) ] ) ] ) , Css.property "animation-duration" "1s" - , Css.property "left" body.center.value + , Css.property "left" (Debug.log "center3" body.center.value) , Css.top body.fromTop , Css.position Css.absolute ] @@ -723,6 +812,8 @@ modalNoneOpening (BodySettings body) = , Css.position Css.absolute ] + + -- Open animations @@ -752,8 +843,8 @@ modalOpenFromTop (BodySettings body) = ] -modalOpenFromRigth : BodySettings -> Attribute msg -modalOpenFromRigth (BodySettings body) = +modalOpenFromRight : BodySettings -> Attribute msg +modalOpenFromRight (BodySettings body) = css [ Css.property "right" body.center.value , Css.top body.fromTop @@ -870,14 +961,14 @@ modalLeftClosing (BodySettings body) = (Animations.keyframes [ ( 0 , [ Animations.property "opacity" "1" - , Animations.property "left" body.center.value - , Animations.property "top" body.fromTop.value + , Animations.property "left" (Debug.log "center:left" body.center.value) + , Animations.property "top" (Debug.log "top:left:0" body.fromTop.value) ] ) , ( 100 , [ Animations.property "opacity" "0" , Animations.property "left" "0" - , Animations.property "top" body.fromTop.value + , Animations.property "top" (Debug.log "top:left:1000" body.fromTop.value) ] ) ] @@ -916,6 +1007,7 @@ modalNoneClosing (BodySettings body) = ] + -- Modal close animation @@ -1014,7 +1106,7 @@ setModalState msg modal = if msg == CancelModal then Canceling config - else + else Closing config Closing _ -> @@ -1029,6 +1121,7 @@ setModalState msg modal = Canceled -> Canceled + {-| @priv Centering and adaptive width for modal body -}