defmodule CanneryWeb.Components.MovePackComponent do @moduledoc """ Livecomponent that can move a pack to another container """ use CanneryWeb, :live_component alias Cannery.{Accounts.User, Ammo, Ammo.Pack, Containers, Containers.Container} alias Ecto.Changeset alias Phoenix.LiveView.Socket @impl true @spec update( %{ required(:current_user) => User.t(), required(:pack) => Pack.t(), optional(any()) => any() }, Socket.t() ) :: {:ok, Socket.t()} def update( %{pack: %{container_id: container_id} = pack, current_user: current_user} = assigns, socket ) do changeset = pack |> Pack.update_changeset(%{}, current_user) containers = Containers.list_containers(current_user) |> Enum.reject(fn %{id: id} -> id == container_id end) socket = socket |> assign(assigns) |> assign(changeset: changeset, containers: containers) {:ok, socket} end @impl true def handle_event( "move", %{"container_id" => container_id}, %{assigns: %{pack: pack, current_user: current_user, return_to: return_to}} = socket ) do %{name: container_name} = Containers.get_container!(container_id, current_user) socket = pack |> Ammo.update_pack(%{"container_id" => container_id}, current_user) |> case do {:ok, _pack} -> prompt = dgettext("prompts", "Ammo moved to %{name} successfully", name: container_name) socket |> put_flash(:info, prompt) |> push_navigate(to: return_to) {:error, %Changeset{} = changeset} -> socket |> assign(changeset: changeset) end {:noreply, socket} end @impl true def render(%{containers: containers} = assigns) do columns = [ %{label: gettext("Container"), key: :name}, %{label: gettext("Type"), key: :type}, %{label: gettext("Location"), key: :location}, %{label: gettext("Actions"), key: :actions, sortable: false} ] rows = containers |> get_rows_for_containers(assigns, columns) assigns = assigns |> Map.merge(%{columns: columns, rows: rows}) ~H""" <div class="w-full flex flex-col space-y-8 justify-center items-center"> <h2 class="mb-8 text-center title text-xl text-primary-600"> {dgettext("actions", "Move ammo")} </h2> <%= if @containers |> Enum.empty?() do %> <h2 class="title text-xl text-primary-600"> {gettext("No other containers")} {display_emoji("😔")} </h2> <.link navigate={~p"/containers/new"} class="btn btn-primary"> {dgettext("actions", "Add another container!")} </.link> <% else %> <.live_component module={CanneryWeb.Components.TableComponent} id="move-pack-table" columns={@columns} rows={@rows} /> <% end %> </div> """ end @spec get_rows_for_containers([Container.t()], map(), [map()]) :: [map()] defp get_rows_for_containers(containers, assigns, columns) do containers |> Enum.map(fn container -> columns |> Map.new(fn %{key: key} -> {key, get_row_value_by_key(key, container, assigns)} end) end) end @spec get_row_value_by_key(atom(), Container.t(), map()) :: any() defp get_row_value_by_key(:actions, container, assigns) do assigns = assigns |> Map.put(:container, container) ~H""" <div class="px-4 py-2 space-x-4 flex justify-center items-center"> <button type="button" class="btn btn-primary" phx-click="move" phx-target={@myself} phx-value-container_id={@container.id} > {dgettext("actions", "Select")} </button> </div> """ end defp get_row_value_by_key(key, container, _assigns), do: container |> Map.get(key) end