defmodule MvWeb.UserLive.Form do @moduledoc """ LiveView form for creating and editing users. ## Features - Create new users with email - Edit existing user details - Optional password setting (checkbox to toggle) - Link/unlink member accounts - Email synchronization with linked members ## Form Fields **Required:** - email **Optional:** - password (for password authentication strategy) - linked member (select from existing members) ## Password Management - New users: Can optionally set password with confirmation - Existing users: Can change password (no confirmation required, admin action) - Checkbox toggles password section visibility ## Member Linking Users can be linked to existing member accounts. When linked, emails are synchronized bidirectionally with User.email as the source of truth. ## Events - `validate` - Real-time form validation - `save` - Submit form (create or update user) - `toggle_password_section` - Show/hide password fields """ use MvWeb, :live_view @impl true def render(assigns) do ~H""" <.header> {@page_title} <:subtitle>{gettext("Use this form to manage user records in your database.")} <.form for={@form} id="user-form" phx-change="validate" phx-submit="save"> <.input field={@form[:email]} label={gettext("Email")} required type="email" />
<%= if @show_password_fields do %>
<.input field={@form[:password]} label={gettext("Password")} type="password" required autocomplete="new-password" /> <%= if !@user do %> <.input field={@form[:password_confirmation]} label={gettext("Confirm Password")} type="password" required autocomplete="new-password" /> <% end %>

{gettext("Password requirements")}:

  • {gettext("At least 8 characters")}
  • {gettext("Include both letters and numbers")}
  • {gettext("Consider using special characters")}
<%= if @user do %>

{gettext("Admin Note")}: {gettext( "As an administrator, you can directly set a new password for this user using the same secure Ash Authentication system." )}

<% end %>
<% else %> <%= if @user do %>

{gettext("Note")}: {gettext( "Check 'Change Password' above to set a new password for this user." )}

<% else %>

{gettext("Note")}: {gettext( "User will be created without a password. Check 'Set Password' to add one." )}

<% end %> <% end %>
<.button phx-disable-with={gettext("Saving...")} variant="primary"> {gettext("Save User")} <.button navigate={return_path(@return_to, @user)}>{gettext("Cancel")}
""" end @impl true def mount(params, _session, socket) do user = case params["id"] do nil -> nil id -> Ash.get!(Mv.Accounts.User, id, domain: Mv.Accounts) end action = if is_nil(user), do: gettext("New"), else: gettext("Edit") page_title = action <> " " <> gettext("User") {:ok, socket |> assign(:return_to, return_to(params["return_to"])) |> assign(user: user) |> assign(:page_title, page_title) |> assign(:show_password_fields, false) |> assign_form()} end defp return_to("show"), do: "show" defp return_to(_), do: "index" @impl true def handle_event("toggle_password_section", _params, socket) do show_password_fields = !socket.assigns.show_password_fields socket = socket |> assign(:show_password_fields, show_password_fields) |> assign_form() {:noreply, socket} end def handle_event("validate", %{"user" => user_params}, socket) do {:noreply, assign(socket, form: AshPhoenix.Form.validate(socket.assigns.form, user_params))} end def handle_event("save", %{"user" => user_params}, socket) do case AshPhoenix.Form.submit(socket.assigns.form, params: user_params) do {:ok, user} -> notify_parent({:saved, user}) socket = socket |> put_flash(:info, "User #{socket.assigns.form.source.type}d successfully") |> push_navigate(to: return_path(socket.assigns.return_to, user)) {:noreply, socket} {:error, form} -> {:noreply, assign(socket, form: form)} end end defp notify_parent(msg), do: send(self(), {__MODULE__, msg}) defp assign_form(%{assigns: %{user: user, show_password_fields: show_password_fields}} = socket) do form = if user do # For existing users, use admin password action if password fields are shown action = if show_password_fields, do: :admin_set_password, else: :update_user AshPhoenix.Form.for_update(user, action, domain: Mv.Accounts, as: "user") else # For new users, use password registration if password fields are shown action = if show_password_fields, do: :register_with_password, else: :create_user AshPhoenix.Form.for_create(Mv.Accounts.User, action, domain: Mv.Accounts, as: "user" ) end assign(socket, form: to_form(form)) end defp return_path("index", _user), do: ~p"/users" defp return_path("show", user), do: ~p"/users/#{user.id}" end