Role CRUD LiveViews closes #325 #326

Merged
moritz merged 28 commits from feature/325_role_view into main 2026-01-08 16:21:42 +01:00
2 changed files with 45 additions and 4 deletions
Showing only changes of commit 61c98d1b88 - Show all commits

View file

@ -55,8 +55,9 @@
</div> </div>
<%= if can?(@current_user, :update, Mv.Authorization.Role) do %> <%= if can?(@current_user, :update, Mv.Authorization.Role) do %>
<.link navigate={~p"/admin/roles/#{role}/edit"} class="btn btn-ghost btn-xs"> <.link navigate={~p"/admin/roles/#{role}/edit"} class="btn btn-ghost btn-sm">
<.icon name="hero-pencil" class="size-4" /> <.icon name="hero-pencil" class="size-4" />
{gettext("Edit")}
</.link> </.link>
<% end %> <% end %>
</:action> </:action>
@ -66,10 +67,10 @@
<.link <.link
phx-click={JS.push("delete", value: %{id: role.id}) |> hide("#row-#{role.id}")} phx-click={JS.push("delete", value: %{id: role.id}) |> hide("#row-#{role.id}")}
data-confirm={gettext("Are you sure?")} data-confirm={gettext("Are you sure?")}
class="btn btn-ghost btn-xs text-error" class="btn btn-ghost btn-sm text-error"
aria-label={gettext("Delete role")}
> >
<.icon name="hero-trash" class="size-4" /> <.icon name="hero-trash" class="size-4" />
{gettext("Delete")}
</.link> </.link>
<% else %> <% else %>
<div <div
@ -78,11 +79,12 @@
data-tip={gettext("System roles cannot be deleted")} data-tip={gettext("System roles cannot be deleted")}
> >
<button <button
class="btn btn-ghost btn-xs text-error opacity-50 cursor-not-allowed" class="btn btn-ghost btn-sm text-error opacity-50 cursor-not-allowed"
disabled={true} disabled={true}
aria-label={gettext("Cannot delete system role")} aria-label={gettext("Cannot delete system role")}
> >
<.icon name="hero-trash" class="size-4" /> <.icon name="hero-trash" class="size-4" />
{gettext("Delete")}
</button> </button>
</div> </div>
<% end %> <% end %>

View file

@ -39,6 +39,29 @@ defmodule MvWeb.RoleLive.Show do
|> assign(:role, role)} |> assign(:role, role)}
end end
@impl true
def handle_event("delete", %{"id" => id}, socket) do
{:ok, role} = Mv.Authorization.get_role(id)
case Mv.Authorization.destroy_role(role) do
:ok ->
{:noreply,
socket
|> put_flash(:info, gettext("Role deleted successfully."))
|> push_navigate(to: ~p"/admin/roles")}
{:error, error} ->
error_message = format_error(error)
{:noreply,
put_flash(
socket,
:error,
gettext("Failed to delete role: %{error}", error: error_message)
)}
end
end
@impl true @impl true
def render(assigns) do def render(assigns) do
~H""" ~H"""
@ -57,6 +80,15 @@ defmodule MvWeb.RoleLive.Show do
<.icon name="hero-pencil-square" /> {gettext("Edit Role")} <.icon name="hero-pencil-square" /> {gettext("Edit Role")}
</.button> </.button>
<% end %> <% end %>
<%= if can?(@current_user, :destroy, Mv.Authorization.Role) and not @role.is_system_role do %>
<.link
phx-click={JS.push("delete", value: %{id: @role.id})}
data-confirm={gettext("Are you sure?")}
class="btn btn-error"
>
<.icon name="hero-trash" /> {gettext("Delete Role")}
</.link>
<% end %>
</:actions> </:actions>
</.header> </.header>
@ -86,6 +118,13 @@ defmodule MvWeb.RoleLive.Show do
""" """
end end
defp format_error(%Ash.Error.Invalid{} = error) do
Enum.map_join(error.errors, ", ", fn e -> e.message end)
end
defp format_error(error) when is_binary(error), do: error
defp format_error(_error), do: gettext("An error occurred")
defp permission_set_badge_class("own_data"), do: "badge badge-neutral badge-sm" defp permission_set_badge_class("own_data"), do: "badge badge-neutral badge-sm"
defp permission_set_badge_class("read_only"), do: "badge badge-info badge-sm" defp permission_set_badge_class("read_only"), do: "badge badge-info badge-sm"
defp permission_set_badge_class("normal_user"), do: "badge badge-success badge-sm" defp permission_set_badge_class("normal_user"), do: "badge badge-success badge-sm"