svg.pablopie.xyz

A simple SVG markup editor for the web

NameSizeMode
..
src/Editor.elm 2086B -rw-r--r--
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
module Editor exposing (editor)

import Types exposing (..)
import Html exposing (Html, Attribute, div, pre, code, text, textarea)
import Html.Attributes exposing (id, value, style, spellcheck, alt, attribute)
import Html.Events exposing (onInput, on)
import Accessibility.Landmark exposing (region)
import Accessibility.Widget exposing (label, hidden)
import Json.Decode
import SyntaxHighlight exposing (xml, toBlockHtml)

editor : Model -> Html Msg
editor model =
  let
    -- Indent the editor according to the order of magnitude of the number of
    -- lines in the input file
    indent 
      = model.image
        |> String.lines
        |> List.length
        |> toFloat
        |> logBase 10
        |> floor
        |> (+) 1
        |> max 1
        |> String.fromInt
        |> \s -> "--indent: " ++ s ++"em;"
  in
    div 
      [ id "editor"
      , region
      , label "Text editor" 
      , attribute "style" indent
      ]
      [ div 
        [ hidden True, translate model.editorScroll ]
        [ codeDisplay model ]
      , textarea model
      ]

textarea : Model -> Html Msg
textarea model =
  Html.textarea
    [ value model.image
    , onInput Update
    , onScroll Scroll
    , spellcheck False
    , alt "Text Editor"
    ] []

codeDisplay : Model -> Html Msg
codeDisplay model =
  xml (if model.image == "" then placeholder else model.image)
    |> Result.map (toBlockHtml (Just 1))
    |> Result.withDefault (pre [] [ code [] [ text model.image ] ])

onScroll : ((Int, Int) -> msg) -> Attribute msg
onScroll f = 
  Json.Decode.map2 
    (\x -> \y -> (x, y))
    (Json.Decode.at [ "target", "scrollLeft" ] Json.Decode.int)
    (Json.Decode.at [ "target", "scrollTop"  ] Json.Decode.int)
    |> Json.Decode.map f
    |> on "scroll"

translate : (Int, Int) -> Attribute Msg
translate (x, y) =
  let 
    show = String.fromInt
  in
    style 
      "transform" 
      ("translate(" ++ (show -x) ++ "px, " ++ (show -y) ++ "px)")

placeholder : String
placeholder = "<svg> ... </svg>"