aboutsummaryrefslogtreecommitdiff
path: root/src/client/elm/Tooltip.elm
diff options
context:
space:
mode:
Diffstat (limited to 'src/client/elm/Tooltip.elm')
-rw-r--r--src/client/elm/Tooltip.elm113
1 files changed, 113 insertions, 0 deletions
diff --git a/src/client/elm/Tooltip.elm b/src/client/elm/Tooltip.elm
new file mode 100644
index 0000000..4f70cda
--- /dev/null
+++ b/src/client/elm/Tooltip.elm
@@ -0,0 +1,113 @@
+module Tooltip exposing
+ ( Msg(..)
+ , Model
+ , init
+ , subscription
+ , update
+ , view
+ , show
+ )
+
+import Platform.Cmd
+
+import Html exposing (..)
+import Html.Attributes exposing (..)
+import Html.Events exposing (..)
+
+import Mouse exposing (Position)
+import Window exposing (Size)
+
+type Msg =
+ UpdateMousePosition Position
+ | UpdateWindowSize Size
+ | ShowMessage String
+ | HideMessage
+
+type alias Model =
+ { mousePosition : Maybe Position
+ , windowSize : Size
+ , message : Maybe String
+ }
+
+init : Int -> Int -> Model
+init width height =
+ { mousePosition = Nothing
+ , windowSize =
+ { width = width
+ , height = height
+ }
+ , message = Nothing
+ }
+
+subscription : Sub Msg
+subscription =
+ Sub.batch
+ [ Mouse.moves UpdateMousePosition
+ , Window.resizes UpdateWindowSize
+ ]
+
+update : Msg -> Model -> (Model, Cmd Msg)
+update msg model =
+ case msg of
+ UpdateMousePosition position ->
+ ( { model | mousePosition = Just position }
+ , Cmd.none
+ )
+
+ UpdateWindowSize size ->
+ ( { model | windowSize = size }
+ , Cmd.none
+ )
+
+ ShowMessage message ->
+ ( { model | message = Just message }
+ , Cmd.none
+ )
+
+ HideMessage ->
+ ( { model | message = Nothing }
+ , Cmd.none
+ )
+
+view : Model -> Html Msg
+view { mousePosition, windowSize, message } =
+ case (mousePosition, message) of
+ (Just pos, Just msg) ->
+ div
+ [ class "tooltip"
+ , style
+ [ ("position", "absolute")
+ , horizontalPosition windowSize pos
+ , ("top", px <| pos.y + 15)
+ ]
+ ]
+ [ text msg ]
+ _ ->
+ text ""
+
+horizontalPosition : Size -> Position -> (String, String)
+horizontalPosition size position =
+ if isLeft size position
+ then ("left", px <| position.x + 5)
+ else ("right", px <| size.width - position.x)
+
+verticalPosition : Size -> Position -> (String, String)
+verticalPosition size position =
+ if isTop size position
+ then ("top", px <| position.y + 20)
+ else ("bottom", px <| size.height - position.y + 15)
+
+px : Int -> String
+px n = (toString n) ++ "px"
+
+isLeft : Size -> Position -> Bool
+isLeft { width } { x } = x < width // 2
+
+isTop : Size -> Position -> Bool
+isTop { height } { y } = y < height // 2
+
+show : (Msg -> msg) -> String -> List (Attribute msg)
+show mapMsg message =
+ [ onMouseEnter <| mapMsg <| ShowMessage message
+ , onMouseLeave <| mapMsg <| HideMessage
+ ]