module IHP.IDE.Data.View.NewRow where

import IHP.ViewPrelude
import IHP.IDE.ToolServer.Types
import IHP.IDE.Data.View.ShowDatabase
import IHP.IDE.Data.View.Layout
import Data.Maybe
import qualified Data.Text as Text

data NewRowView = NewRowView
    { NewRowView -> [Text]
tableNames :: [Text]
    , NewRowView -> Text
tableName :: Text
    , NewRowView -> [[DynamicField]]
rows :: [[DynamicField]]
    , NewRowView -> [ColumnDefinition]
tableCols :: [ColumnDefinition]
    }

instance View NewRowView where
    html :: NewRowView -> Html
html NewRowView { [[DynamicField]]
[Text]
[ColumnDefinition]
Text
tableCols :: [ColumnDefinition]
rows :: [[DynamicField]]
tableName :: Text
tableNames :: [Text]
$sel:tableCols:NewRowView :: NewRowView -> [ColumnDefinition]
$sel:rows:NewRowView :: NewRowView -> [[DynamicField]]
$sel:tableName:NewRowView :: NewRowView -> Text
$sel:tableNames:NewRowView :: NewRowView -> [Text]
.. } = [hsx|
        <div class="h-100">
            {headerNav}
            <div class="h-100 row no-gutters">
                {renderTableSelector tableNames tableName}
                <div class="col" style="overflow: scroll; max-height: 80vh">
                    {renderRows rows tableBody tableName}
                </div>
            </div>
        </div>
        {renderModal modal}
    |]
        where
            tableBody :: Html
tableBody = [hsx|<tbody>{forEach rows renderRow}</tbody>|]
            renderRow :: [DynamicField] -> Html
renderRow [DynamicField]
fields = [hsx|<tr>{forEach fields (renderField id)}</tr>|]
                where
                    id :: Text
id = (ByteString -> Text
forall a b. ConvertibleStrings a b => a -> b
cs (ByteString -> Maybe ByteString -> ByteString
forall a. a -> Maybe a -> a
fromMaybe ByteString
"" (Proxy "fieldValue" -> DynamicField -> Maybe ByteString
forall model (name :: Symbol) value.
(KnownSymbol name, HasField name model value) =>
Proxy name -> model -> value
get IsLabel "fieldValue" (Proxy "fieldValue")
Proxy "fieldValue"
#fieldValue (Maybe DynamicField -> DynamicField
forall a. HasCallStack => Maybe a -> a
fromJust ([DynamicField] -> Maybe DynamicField
forall a. [a] -> Maybe a
headMay [DynamicField]
fields)))))
            renderField :: Text -> DynamicField -> Html
renderField Text
id DynamicField { Maybe ByteString
ByteString
$sel:fieldName:DynamicField :: DynamicField -> ByteString
$sel:fieldValue:DynamicField :: DynamicField -> Maybe ByteString
fieldName :: ByteString
fieldValue :: Maybe ByteString
.. } | ByteString
fieldName ByteString -> ByteString -> Bool
forall a. Eq a => a -> a -> Bool
== ByteString
"id" = [hsx|<td><span data-fieldname={fieldName}><a class="no-link border rounded p-1" href={EditRowValueAction tableName (cs fieldName) id}>{renderId (sqlValueToText fieldValue)}</a></span></td>|]
            renderField Text
id DynamicField { Maybe ByteString
ByteString
fieldName :: ByteString
fieldValue :: Maybe ByteString
$sel:fieldName:DynamicField :: DynamicField -> ByteString
$sel:fieldValue:DynamicField :: DynamicField -> Maybe ByteString
.. } | ByteString -> [ColumnDefinition] -> Bool
forall a1 a2 a3 (t :: * -> *) model.
(Eq a1, Eq a2, ConvertibleStrings a3 a2, Foldable t,
 HasField "columnName" model a2, HasField "columnType" model a1,
 IsString a1) =>
a3 -> t model -> Bool
isBoolField ByteString
fieldName [ColumnDefinition]
tableCols Bool -> Bool -> Bool
&& Bool -> Bool
not (Maybe ByteString -> Bool
forall a. Maybe a -> Bool
isNothing Maybe ByteString
fieldValue) = [hsx|<td><span data-fieldname={fieldName}><input type="checkbox" onclick={onClick tableName fieldName id} checked={sqlValueToText fieldValue == "t"} /></span></td>|]
            renderField Text
id DynamicField { Maybe ByteString
ByteString
fieldName :: ByteString
fieldValue :: Maybe ByteString
$sel:fieldName:DynamicField :: DynamicField -> ByteString
$sel:fieldValue:DynamicField :: DynamicField -> Maybe ByteString
.. } = [hsx|<td><span data-fieldname={fieldName}><a class="no-link" href={EditRowValueAction tableName (cs fieldName) id}>{sqlValueToText fieldValue}</a></span></td>|]

            modalContent :: Html
modalContent = [hsx|
                <form method="POST" action={CreateRowAction}>
                    <input type="hidden" name="tableName" value={tableName}/>
                    {forEach tableCols renderFormField}
                    {renderFlashMessages}
                    <div class="text-right">
                        <button type="submit" class="btn btn-primary">Add Row</button>
                    </div>
                </form>
            |]
            modalFooter :: Maybe Html
modalFooter = Maybe Html
forall a. Monoid a => a
mempty
            modalCloseUrl :: Text
modalCloseUrl = DataController -> Text
forall controller. HasPath controller => controller -> Text
pathTo ShowTableRowsAction :: Text -> DataController
ShowTableRowsAction { Text
$sel:tableName:ShowDatabaseAction :: Text
tableName :: Text
tableName }
            modalTitle :: Text
modalTitle = Text
"Add Row"
            modal :: Modal
modal = Modal :: Html -> Maybe Html -> Text -> Text -> Modal
Modal { Html
$sel:modalContent:Modal :: Html
modalContent :: Html
modalContent, Maybe Html
$sel:modalFooter:Modal :: Maybe Html
modalFooter :: Maybe Html
modalFooter, Text
$sel:modalCloseUrl:Modal :: Text
modalCloseUrl :: Text
modalCloseUrl, Text
$sel:modalTitle:Modal :: Text
modalTitle :: Text
modalTitle }

            renderFormField :: ColumnDefinition -> Html
renderFormField ColumnDefinition
col = [hsx|
                    <div class="form-group">
                        <label class="row-form">{get #columnName col}</label>
                        <span style="float:right;">
                            <a class="text-muted row-form">{get #columnType col}</a>
                        </span>

                        <div class="d-flex">
                            {renderInputMethod col}
                        </div>
                    </div>|]

            onClick :: Text -> a -> Text -> Text
onClick Text
tableName a
fieldName Text
id = Text
"window.location.assign(" Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text -> Text
forall a. Show a => a -> Text
tshow (DataController -> Text
forall controller. HasPath controller => controller -> Text
pathTo (Text -> Text -> Text -> DataController
ToggleBooleanFieldAction Text
tableName (a -> Text
forall a b. ConvertibleStrings a b => a -> b
cs a
fieldName) Text
id)) Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
")"
            renderInputMethod :: ColumnDefinition -> Html 
            renderInputMethod :: ColumnDefinition -> (?context::ControllerContext) => Html
renderInputMethod ColumnDefinition
col | (Proxy "columnType" -> ColumnDefinition -> Text
forall model (name :: Symbol) value.
(KnownSymbol name, HasField name model value) =>
Proxy name -> model -> value
get IsLabel "columnType" (Proxy "columnType")
Proxy "columnType"
#columnType ColumnDefinition
col) Text -> Text -> Bool
forall a. Eq a => a -> a -> Bool
== Text
"boolean" = [hsx|
                            {isBooleanParam True col}
                            <input
                                id={get #columnName col <> "-alt"}
                                type="text"
                                name={get #columnName col <> "-inactive"}
                                class="form-control text-monospace text-secondary d-none"
                                />
                            <div class="form-control" id={get #columnName col <> "-boxcontainer"}>
                                <input
                                    id={get #columnName col <> "-input"}
                                    type="checkbox"
                                    name={get #columnName col}
                                    checked={get #columnDefault col == Just "true"}
                                    />
                            </div>
                            <input
                                id={get #columnName col <> "-hidden"}
                                type="hidden"
                                name={get #columnName col}
                                value={inputValue False}
                                />
                            <div class="input-group-append">
                                <button class="btn dropdown-toggle" type="button" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false"></button>
                                <div class="dropdown-menu dropdown-menu-right custom-menu menu-for-column shadow backdrop-blur">
                                    <a class="dropdown-item" data-value="DEFAULT" data-issql="True" onclick={fillField col "DEFAULT" "true"}>DEFAULT</a>
                                    <a class="dropdown-item" data-value="NULL" data-issql="True" onclick={fillField col "NULL" "true"}>NULL</a>
                                    <a class="dropdown-item">
                                        <input
                                            id={get #columnName col <> "-sqlbox"}
                                            type="checkbox"
                                            name={get #columnName col <> "_"}
                                            checked={isSqlFunction (getColDefaultValue col)}
                                            class="mr-1"
                                            onclick={"sqlModeCheckbox('" <> get #columnName col <> "', this, true)"}
                                            />
                                        <label class="form-check-label" for={get #columnName col <> "-sqlbox"}> Parse as SQL</label>
                                    </a>
                                    <input
                                        type="hidden"
                                        name={get #columnName col <> "_"}
                                        value={inputValue False}
                                        />
                                </div>
                            </div>
                                |]
            renderInputMethod ColumnDefinition
col = [hsx|
                                {isBooleanParam False col}
                                {if isForeignKeyColumn
                                        then select
                                        else theInput
                                }
                                <button class="btn dropdown-toggle" type="button" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false"></button>
                                <div class="dropdown-menu dropdown-menu-right custom-menu menu-for-column shadow backdrop-blur">
                                    <a class="dropdown-item" data-value="DEFAULT" data-issql="True" onclick={fillField col "DEFAULT" "false"}>DEFAULT</a>
                                    <a class="dropdown-item" data-value="NULL" data-issql="True" onclick={fillField col "NULL" "false"}>NULL</a>
                                    <a class="dropdown-item">
                                        <input
                                            id={get #columnName col <> "-sqlbox"}
                                            type="checkbox"
                                            name={get #columnName col <> "_"}
                                            checked={isSqlFunction (getColDefaultValue col)}
                                            class="mr-1"
                                            onclick={"sqlModeCheckbox('" <> get #columnName col <> "', this, false)"}
                                            />
                                        <label class="form-check-label" for={get #columnName col <> "-sqlbox"}> Parse as SQL</label>
                                    </a>
                                    <input
                                        type="hidden"
                                        name={get #columnName col <> "_"}
                                        value={inputValue False}
                                        />
                                </div>
                            |]
                                where
                                    isForeignKeyColumn :: Bool
                                    isForeignKeyColumn :: Bool
isForeignKeyColumn = Text
"_id" Text -> Text -> Bool
`Text.isSuffixOf` (Proxy "columnName" -> ColumnDefinition -> Text
forall model (name :: Symbol) value.
(KnownSymbol name, HasField name model value) =>
Proxy name -> model -> value
get IsLabel "columnName" (Proxy "columnName")
Proxy "columnName"
#columnName ColumnDefinition
col)


                                    theInput :: Html
                                    theInput :: Html
theInput = [hsx|
                                        <input
                                            id={get #columnName col <> "-input"}
                                            type="text"
                                            name={get #columnName col}
                                            class={classes ["form-control", ("text-monospace", isSqlFunction (getColDefaultValue col)), ("is-foreign-key-column", isForeignKeyColumn)]}
                                            value={renderDefaultWithoutType (getColDefaultValue col)}
                                            oninput={"stopSqlModeOnInput('" <> get #columnName col <> "')"}
                                        />
                                    |]

                                    select :: Html
                                    select :: Html
select = [hsx|
                                        <select
                                            id={get #columnName col <> "-input"}
                                            name={get #columnName col}
                                            class={classes ["form-control", ("is-foreign-key-column", isForeignKeyColumn)]}
                                            value={renderDefaultWithoutType (getColDefaultValue col)}
                                            data-select-url={selectUrl}
                                        />
                                    |]

                                    selectUrl :: Text
                                    selectUrl :: Text
selectUrl = DataController -> Text
forall controller. HasPath controller => controller -> Text
pathTo AutocompleteForeignKeyColumnAction :: Text -> Text -> Maybe Text -> DataController
AutocompleteForeignKeyColumnAction { Text
$sel:tableName:ShowDatabaseAction :: Text
tableName :: Text
tableName, $sel:columnName:ShowDatabaseAction :: Text
columnName = Proxy "columnName" -> ColumnDefinition -> Text
forall model (name :: Symbol) value.
(KnownSymbol name, HasField name model value) =>
Proxy name -> model -> value
get IsLabel "columnName" (Proxy "columnName")
Proxy "columnName"
#columnName ColumnDefinition
col, $sel:term:ShowDatabaseAction :: Maybe Text
term = Maybe Text
"" }