defmodule MvWeb.UserLive.Form do 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" />

{gettext("Member Assignment")}

<%= if @member_assignment_mode == "assign_existing" do %>
<.input field={@form[:member_id]} label={gettext("Select Member")} type="select" options={@available_members} prompt={gettext("Choose a member...")} />
<% end %>
<%= 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") # Load available members that have no user assigned {:ok, available_members} = Mv.Membership.list_members() available_members_with_user = Ash.load!(available_members, :user) available_member_options = available_members_with_user |> Enum.filter(fn member -> is_nil(member.user) end) |> Enum.map(fn member -> {"#{member.first_name} #{member.last_name} (#{member.email})", member.id} end) {:ok, socket |> assign(:return_to, return_to(params["return_to"])) |> assign(user: user) |> assign(:page_title, page_title) |> assign(:show_password_fields, false) |> assign(:member_assignment_mode, "create_new") |> assign(:available_members, available_member_options) |> 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("set_member_mode", %{"mode" => mode}, socket) do socket = socket |> assign(:member_assignment_mode, mode) |> 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, as: "user", actor: socket.assigns.current_user, domain: Mv.Accounts ) else # For new users, use password registration if password fields are shown action = if show_password_fields, do: :register_with_password, else: :create_user # Only include member_id if assign_existing mode is selected AND not using password action accept = if socket.assigns.member_assignment_mode == "assign_existing" and not show_password_fields do [:email, :member_id] else [:email] end AshPhoenix.Form.for_create(Mv.Accounts.User, action, as: "user", actor: socket.assigns.current_user, domain: Mv.Accounts, accept: accept ) 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