defmodule CanneryWeb.RangeLive.Index do @moduledoc """ Main page for range day mode, where `AmmoGroup`s can be used up. """ use CanneryWeb, :live_view import CanneryWeb.Components.AmmoGroupCard alias Cannery.{ActivityLog, ActivityLog.ShotGroup, Ammo, Repo} alias CanneryWeb.Endpoint alias Phoenix.LiveView.Socket @impl true def mount(_params, _session, socket), do: {:ok, socket |> display_shot_groups()} @impl true def handle_params(params, _url, %{assigns: %{live_action: live_action}} = socket) do {:noreply, apply_action(socket, live_action, params)} end defp apply_action( %{assigns: %{current_user: current_user}} = socket, :add_shot_group, %{"id" => id} ) do socket |> assign(:page_title, gettext("Record Shots")) |> assign(:ammo_group, Ammo.get_ammo_group!(id, current_user)) end defp apply_action(%{assigns: %{current_user: current_user}} = socket, :edit, %{"id" => id}) do socket |> assign(:page_title, gettext("Edit Shot Records")) |> assign(:shot_group, ActivityLog.get_shot_group!(id, current_user)) end defp apply_action(socket, :new, _params) do socket |> assign(:page_title, gettext("New Shot Records")) |> assign(:shot_group, %ShotGroup{}) end defp apply_action(socket, :index, _params) do socket |> assign(:page_title, gettext("Shot Records")) |> assign(:shot_group, nil) end @impl true def handle_event("delete", %{"id" => id}, %{assigns: %{current_user: current_user}} = socket) do {:ok, _} = ActivityLog.get_shot_group!(id, current_user) |> ActivityLog.delete_shot_group(current_user) prompt = dgettext("prompts", "Shot records deleted succesfully") {:noreply, socket |> put_flash(:info, prompt) |> display_shot_groups()} end def handle_event( "toggle_staged", %{"ammo_group_id" => ammo_group_id}, %{assigns: %{current_user: current_user}} = socket ) do ammo_group = Ammo.get_ammo_group!(ammo_group_id, current_user) {:ok, _ammo_group} = ammo_group |> Ammo.update_ammo_group(%{"staged" => !ammo_group.staged}, current_user) prompt = dgettext("prompts", "Ammo unstaged succesfully") {:noreply, socket |> put_flash(:info, prompt) |> display_shot_groups()} end @spec display_shot_groups(Socket.t()) :: Socket.t() defp display_shot_groups(%{assigns: %{current_user: current_user}} = socket) do shot_groups = ActivityLog.list_shot_groups(current_user) |> Repo.preload(ammo_group: :ammo_type) ammo_groups = Ammo.list_staged_ammo_groups(current_user) columns = [ %{label: gettext("Ammo"), key: :name}, %{label: gettext("Rounds shot"), key: :count}, %{label: gettext("Notes"), key: :notes}, %{label: gettext("Date"), key: :date}, %{label: nil, key: :actions, sortable: false} ] rows = shot_groups |> Enum.map(fn shot_group -> shot_group |> get_row_data_for_shot_group(columns) end) chart_data = shot_groups |> get_chart_data_for_shot_group() socket |> assign( ammo_groups: ammo_groups, columns: columns, rows: rows, chart_data: chart_data, shot_groups: shot_groups ) end @spec get_chart_data_for_shot_group([ShotGroup.t()]) :: [map()] defp get_chart_data_for_shot_group(shot_groups) do shot_groups |> Repo.preload(ammo_group: :ammo_type) |> Enum.group_by(fn %{date: date} -> date end, fn %{count: count} -> count end) |> Enum.map(fn {date, rounds} -> sum = Enum.sum(rounds) %{ date: date, count: sum, label: gettext("Rounds shot: %{count}", count: sum) } end) |> Enum.sort_by(fn %{date: date} -> date end) end @spec get_row_data_for_shot_group(ShotGroup.t(), [map()]) :: map() defp get_row_data_for_shot_group(%{date: date} = shot_group, columns) do shot_group = shot_group |> Repo.preload(ammo_group: :ammo_type) assigns = %{shot_group: shot_group} columns |> Map.new(fn %{key: key} -> value = case key do :name -> {shot_group.ammo_group.ammo_type.name, ~H""" <.link navigate={Routes.ammo_group_show_path(Endpoint, :show, @shot_group.ammo_group)} class="link" > <%= @shot_group.ammo_group.ammo_type.name %> """} :date -> date |> display_date() :actions -> ~H"""