{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeFamilies #-}
module Ditto.Lucid.Named where
import Data.Foldable (traverse_, fold)
import Ditto.Backend
import Ditto.Core
import Ditto.Lucid (encodeFormId)
import Ditto.Generalized.Named as G
import Ditto.Result (FormId, Result (Ok), unitRange)
import Lucid
import Web.PathPieces
import qualified Data.Text as T
foldTraverse_ :: (Foldable t, Applicative f, Monoid (f b)) => (a -> t (f b)) -> t a -> f ()
foldTraverse_ f = traverse_ (fold . f)
inputText
  :: (Monad m, FormError err input, PathPiece text, Applicative f)
  => (input -> Either err text)
  -> String
  -> text
  -> Form m input err (HtmlT f ()) text
inputText getInput name initialValue = G.input name getInput inputField initialValue
  where
  inputField i a = input_ [type_ "text", id_ (encodeFormId i), name_ (encodeFormId i), value_ (toPathPiece a)]
inputTextMReq
  :: (Monad m, FormError err input, PathPiece text, Applicative f)
  => (input -> Either err text)
  -> String
  -> Maybe text
  -> Form m input err (HtmlT f ()) text
inputTextMReq getInput name initialValue = G.inputMaybeReq name getInput inputField initialValue
  where
  inputField i Nothing = input_ [type_ "text", id_ (encodeFormId i), name_ (encodeFormId i), required_ "required"]
  inputField i (Just a) = input_ [type_ "text", id_ (encodeFormId i), name_ (encodeFormId i), value_ (toPathPiece a), required_ "required"]
inputPassword
  :: (Monad m, FormError err input, PathPiece text, Applicative f)
  => (input -> Either err text)
  -> String
  -> text
  -> Form m input err (HtmlT f ()) text
inputPassword getInput name initialValue = G.input name getInput inputField initialValue
  where
  inputField i a = input_ [type_ "password", id_ (encodeFormId i), name_ (encodeFormId i), value_ (toPathPiece a)]
inputSubmit
  :: (Monad m, FormError err input, PathPiece text, Applicative f)
  => (input -> Either err text)
  -> String
  -> text
  -> Form m input err (HtmlT f ()) (Maybe text)
inputSubmit getInput name initialValue = G.inputMaybe name getInput inputField (Just initialValue)
  where
  inputField i a = input_ [type_ "submit", id_ (encodeFormId i), name_ (encodeFormId i), value_ (toPathPiece a)]
inputReset
  :: (Monad m, FormError err input, PathPiece text, Applicative f)
  => String
  -> text
  -> Form m input err (HtmlT f ()) ()
inputReset name lbl = G.inputNoData name inputField
  where
  inputField i = input_ [type_ "submit", id_ (encodeFormId i), name_ (encodeFormId i), value_ (toPathPiece lbl)]
inputHidden
  :: (Monad m, FormError err input, PathPiece text, Applicative f)
  => (input -> Either err text)
  -> String
  -> text
  -> Form m input err (HtmlT f ()) text
inputHidden getInput name initialValue = G.input name getInput inputField initialValue
  where
  inputField i a = input_ [type_ "hidden", id_ (encodeFormId i), name_ (encodeFormId i), value_ (toPathPiece a)]
inputButton
  :: (Monad m, FormError err input, PathPiece text, Applicative f)
  => String
  -> text
  -> Form m input err (HtmlT f ()) ()
inputButton name lbl = G.inputNoData name inputField
  where
  inputField i = input_ [type_ "button", id_ (encodeFormId i), name_ (encodeFormId i), value_ (toPathPiece lbl)]
textarea
  :: (Monad m, FormError err input, ToHtml text, Monad f)
  => (input -> Either err text)
  -> Int 
  -> Int 
  -> String
  -> text 
  -> Form m input err (HtmlT f ()) text
textarea getInput cols rows name initialValue = G.input name getInput textareaView initialValue
  where
  textareaView i txt =
    textarea_
      [ rows_ (toPathPiece rows)
      , cols_ (toPathPiece cols)
      , id_ (encodeFormId i)
      , name_ (encodeFormId i)
      ] $
      toHtml txt
inputFile
  :: (Monad m, FormError err input, FormInput input, Applicative f)
  => String
  -> Form m input err (HtmlT f ()) (FileType input)
inputFile name = G.inputFile name fileView
  where
  fileView i = input_ [type_ "file", id_ (encodeFormId i), name_ (encodeFormId i)]
buttonSubmit
  :: (Monad m, FormError err input, PathPiece text, ToHtml children, Monad f)
  => (input -> Either err text)
  -> String
  -> text
  -> children
  -> Form m input err (HtmlT f ()) (Maybe text)
buttonSubmit getInput name text c = G.inputMaybe name getInput inputField (Just text)
  where
  inputField i a = button_ [type_ "submit", id_ (encodeFormId i), name_ (encodeFormId i), value_ (toPathPiece a)] $ toHtml c
buttonReset
  :: (Monad m, FormError err input, Monad f)
  => String
  -> HtmlT f ()
  -> Form m input err (HtmlT f ()) ()
buttonReset name c = G.inputNoData name inputField
  where
  inputField i = button_ [type_ "reset", id_ (encodeFormId i), name_ (encodeFormId i)] c
button
  :: (Monad m, FormError err input, Monad f)
  => String
  -> HtmlT f ()
  -> Form m input err (HtmlT f ()) ()
button name c = G.inputNoData name inputField
  where
  inputField i = button_ [type_ "button", id_ (encodeFormId i), name_ (encodeFormId i)] c
label
  :: (Monad m, Monad f)
  => HtmlT f ()
  -> String
  -> Form m input err (HtmlT f ()) ()
label c name = G.label name mkLabel
  where
  mkLabel i = label_ [for_ (encodeFormId i)] c
arbitraryHtml :: Monad m => view -> Form m input err view ()
arbitraryHtml = view
inputInt
  :: (Monad m, FormError err input, Applicative f)
  => (input -> Either err Int)
  -> String
  -> Int
  -> Form m input err (HtmlT f ()) Int
inputInt getInput name initialValue = G.input name getInput inputField initialValue
  where
  inputField i a =
    input_
      [ type_ "number"
      , id_ (encodeFormId i)
      , name_ (encodeFormId i)
      , value_ (toPathPiece a)
      ]
inputDouble
  :: (Monad m, FormError err input, Applicative f)
  => (input -> Either err Double)
  -> String
  -> Double
  -> Form m input err (HtmlT f ()) Double
inputDouble getInput name initialValue = G.input name getInput inputField initialValue
  where
  inputField i a = input_ [type_ "number", step_ "any", id_ (encodeFormId i), name_ (encodeFormId i), value_ (T.pack $ show a)]
inputCheckbox
  :: forall err input m f. (Monad m, FormError err input, Applicative f)
  => Bool 
  -> String 
  -> Form m input err (HtmlT f ()) Bool
inputCheckbox initiallyChecked name =
  Form $ do
    i <- getNamedFormId name
    v <- getFormInput' i
    case v of
      Default -> mkCheckbox i initiallyChecked
      Missing -> mkCheckbox i False 
      Found _ -> mkCheckbox i True
  where
  mkCheckbox i checked =
    let checkbox =
          input_ $
            (if checked then (:) checked_ else id)
              [type_ "checkbox", id_ (encodeFormId i), name_ (encodeFormId i), value_ (encodeFormId i)]
     in pure
          ( View $ const $ checkbox
          , pure $
            Ok
              ( Proved
                { pos = unitRange i
                , unProved = if checked then True else False
                }
              )
          )
inputCheckboxes
  :: (Functor m, Monad m, FormError err input, FormInput input, ToHtml lbl, Monad f)
  => String
  -> [(a, lbl)] 
  -> (a -> Bool) 
  -> Form m input err (HtmlT f ()) [a]
inputCheckboxes name choices isChecked = G.inputMulti name choices mkCheckboxes isChecked
  where
  mkCheckboxes nm choices' = foldTraverse_ (mkCheckbox nm) choices'
  mkCheckbox nm (i, val, lbl, checked) =
    [ input_ $
        ( (if checked then (checked_ :) else id)
          [type_ "checkbox", id_ (encodeFormId i), name_ (encodeFormId nm), value_ (toPathPiece val)]
        )
    , label_ [for_ (encodeFormId i)] $ toHtml lbl
    ]
inputRadio
  :: (Functor m, Monad m, FormError err input, FormInput input, Monad f)
  => String
  -> [(a, Html ())] 
  -> (a -> Bool) 
  -> Form m input err (HtmlT f ()) a
inputRadio name choices isDefault =
  G.inputChoice name isDefault choices mkRadios
  where
  mkRadios nm choices' = foldTraverse_ (mkRadio nm) choices'
  mkRadio nm (i, val, lbl, checked) =
    [ input_ $
        (if checked then (checked_ :) else id)
          [type_ "radio", id_ (encodeFormId i), name_ (encodeFormId nm), value_ (toPathPiece val)]
    , label_ [for_ (encodeFormId i)] $ toHtml lbl
    , br_ []
    ]
select
  :: (Functor m, Monad m, FormError err input, FormInput input, Monad f)
  => String
  -> [(a, Html ())] 
  -> (a -> Bool) 
  -> Form m input err (HtmlT f ()) a
select name choices isDefault = G.inputChoice name isDefault choices mkSelect
  where
  mkSelect :: (ToHtml lbl, Monad f) => FormId -> [(a, Int, lbl, Bool)] -> HtmlT f ()
  mkSelect nm choices' =
    select_ [name_ (encodeFormId nm)] $
      traverse_ mkOption choices'
  mkOption :: (ToHtml lbl, Monad f) => (a, Int, lbl, Bool) -> HtmlT f ()
  mkOption (_, val, lbl, selected) =
    option_
      ( (if selected then ((:) (selected_ "selected")) else id)
        [value_ (toPathPiece val)]
      )
      (toHtml lbl)
selectMultiple
  :: (Functor m, Monad m, FormError err input, FormInput input, Monad f)
  => String
  -> [(a, Html ())] 
  -> (a -> Bool) 
  -> Form m input err (HtmlT f ()) [a]
selectMultiple name choices isSelected = G.inputMulti name choices mkSelect isSelected
  where
  mkSelect :: (ToHtml lbl, Monad f) => FormId -> [(a, Int, lbl, Bool)] -> HtmlT f ()
  mkSelect nm choices' =
    select_ [name_ (encodeFormId nm), multiple_ "multiple"] $
      traverse_ mkOption choices'
  mkOption :: (ToHtml lbl, Monad f) => (a, Int, lbl, Bool) -> HtmlT f ()
  mkOption (_, val, lbl, selected) =
    option_
      ( (if selected then ((:) (selected_ "selected")) else id)
        [value_ (toPathPiece val)]
      )
      (toHtml lbl)