defmodule LivebookWeb.FormComponents do use Phoenix.Component import LivebookWeb.CoreComponents alias Phoenix.LiveView.JS @doc """ Renders a text input with label and error messages. """ attr :id, :any, default: nil attr :name, :any attr :label, :string, default: nil attr :value, :any attr :errors, :list, default: [] attr :field, Phoenix.HTML.FormField, doc: "a form field struct retrieved from the form" attr :class, :string, default: nil attr :rest, :global, include: ~w(autocomplete readonly disabled) def text_field(assigns) do assigns = assigns_from_field(assigns) ~H""" <.field_wrapper id={@id} name={@name} label={@label} errors={@errors}> """ end @doc """ Renders a textarea input with label and error messages. """ attr :id, :any, default: nil attr :name, :any attr :label, :string, default: nil attr :value, :any attr :errors, :list, default: [] attr :field, Phoenix.HTML.FormField, doc: "a form field struct retrieved from the form" attr :resizable, :boolean, default: false attr :rest, :global, include: ~w(autocomplete readonly disabled rows cols) def textarea_field(assigns) do assigns = assigns_from_field(assigns) ~H""" <.field_wrapper id={@id} name={@name} label={@label} errors={@errors}> """ end @doc """ Renders a hidden input. """ attr :id, :any, default: nil attr :name, :any attr :value, :any attr :field, Phoenix.HTML.FormField, doc: "a form field struct retrieved from the form" attr :rest, :global, include: ~w(autocomplete readonly disabled) def hidden_field(assigns) do assigns = assigns_from_field(assigns) ~H""" """ end @doc """ Renders a password input with label and error messages. """ attr :id, :any, default: nil attr :name, :any attr :label, :string, default: nil attr :value, :any attr :errors, :list, default: [] attr :field, Phoenix.HTML.FormField, doc: "a form field struct retrieved from the form" attr :class, :string, default: nil attr :rest, :global, include: ~w(autocomplete readonly disabled) def password_field(assigns) do assigns = assigns_from_field(assigns) ~H""" <.field_wrapper id={@id} name={@name} label={@label} errors={@errors}> <.with_password_toggle id={@id <> "-toggle"}> """ end @doc """ Renders a hex color input with label and error messages. """ attr :id, :any, default: nil attr :name, :any attr :label, :string, default: nil attr :value, :any attr :errors, :list, default: [] attr :field, Phoenix.HTML.FormField, doc: "a form field struct retrieved from the form" attr :randomize, JS, default: %JS{} attr :rest, :global def hex_color_field(assigns) do assigns = assigns_from_field(assigns) ~H""" <.field_wrapper id={@id} name={@name} label={@label} errors={@errors}>
""" end @doc """ Renders a switch input with label and error messages. """ attr :id, :any, default: nil attr :name, :any attr :label, :string, default: nil attr :value, :any attr :errors, :list, default: [] attr :field, Phoenix.HTML.FormField, doc: "a form field struct retrieved from the form" attr :disabled, :boolean, default: false attr :checked_value, :string, default: "true" attr :unchecked_value, :string, default: "false" attr :tooltip, :string, default: nil attr :rest, :global def switch_field(assigns) do assigns = assigns_from_field(assigns) ~H"""
<%= @label %>
<.error :for={msg <- @errors}><%= msg %>
""" end @doc """ Renders checkbox input with label and error messages. """ attr :id, :any, default: nil attr :name, :any attr :label, :string, default: nil attr :value, :any attr :errors, :list, default: [] attr :field, Phoenix.HTML.FormField, doc: "a form field struct retrieved from the form" attr :disabled, :boolean, default: false attr :checked_value, :string, default: "true" attr :unchecked_value, :string, default: "false" attr :rest, :global def checkbox_field(assigns) do assigns = assigns_from_field(assigns) ~H"""
<.error :for={msg <- @errors}><%= msg %>
""" end @doc """ Renders radio inputs with label and error messages. """ attr :id, :any, default: nil attr :name, :any attr :label, :string, default: nil attr :value, :any attr :errors, :list, default: [] attr :field, Phoenix.HTML.FormField, doc: "a form field struct retrieved from the form" attr :options, :list, default: [], doc: "a list of `{value, description}` tuples" attr :rest, :global def radio_field(assigns) do assigns = assigns_from_field(assigns) ~H"""
<.error :for={msg <- @errors}><%= msg %>
""" end @doc """ Renders emoji input with label and error messages. """ attr :id, :any, default: nil attr :name, :any attr :label, :string, default: nil attr :value, :any attr :errors, :list, default: [] attr :field, Phoenix.HTML.FormField, doc: "a form field struct retrieved from the form" attr :rest, :global def emoji_field(assigns) do assigns = assigns_from_field(assigns) ~H""" <.field_wrapper id={@id} name={@name} label={@label} errors={@errors}>
<%= @value %>
""" end @doc """ Renders select input with label and error messages. """ attr :id, :any, default: nil attr :name, :any attr :label, :string, default: nil attr :value, :any attr :errors, :list, default: [] attr :field, Phoenix.HTML.FormField, doc: "a form field struct retrieved from the form" attr :options, :list, default: [] attr :prompt, :string, default: nil attr :rest, :global def select_field(assigns) do assigns = assigns_from_field(assigns) ~H""" <.field_wrapper id={@id} name={@name} label={@label} errors={@errors}> """ end defp assigns_from_field(%{field: %Phoenix.HTML.FormField{} = field} = assigns) do assigns |> assign(field: nil, id: assigns.id || field.id) |> assign(:errors, Enum.map(field.errors, &translate_error/1)) |> assign_new(:name, fn -> field.name end) |> assign_new(:value, fn -> field.value end) end defp assigns_from_field(assigns), do: assigns @doc """ Translates an error message using gettext. """ def translate_error({msg, opts}) do # Because the error messages we show in our forms and APIs # are defined inside Ecto, we need to translate them dynamically. Enum.reduce(opts, msg, fn {key, value}, acc -> String.replace(acc, "%{#{key}}", fn _ -> to_string(value) end) end) end attr :id, :any, required: true attr :name, :any, required: true attr :label, :string, required: true attr :errors, :list, required: true slot :inner_block, required: true defp field_wrapper(assigns) do ~H"""
<.label :if={@label} for={@id}><%= @label %> <%= render_slot(@inner_block) %> <.error :for={msg <- @errors}><%= msg %>
""" end @doc """ Renders a label. """ attr :for, :string, default: nil slot :inner_block, required: true def label(assigns) do ~H""" """ end @doc """ Generates a generic error message. """ slot :inner_block, required: true def error(assigns) do ~H""" """ end @doc """ Renders a wrapper around password input with an added visibility toggle button. The toggle switches the input's type between `password` and `text`. ## Examples <.with_password_toggle id="secret-password-toggle"> """ attr :id, :string, required: true slot :inner_block, required: true def with_password_toggle(assigns) do ~H"""
<%= render_slot(@inner_block) %>
""" end end