From ac537518db446a511844a34bbdee69105d84938f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Lu=C3=ADs=20Arteiro?= Date: Fri, 11 Nov 2022 13:11:52 +0000 Subject: [PATCH] fix: Editing timers while running works properly. #195 The timer can be updated while it is running. Validation now works properly. --- lib/app/timer.ex | 23 ++++++- lib/app_web/live/app_live.ex | 102 ++++++++++++++++++++++++---- lib/app_web/live/app_live.html.heex | 1 - 3 files changed, 109 insertions(+), 17 deletions(-) diff --git a/lib/app/timer.ex b/lib/app/timer.ex index 31af3ecb..462257d9 100644 --- a/lib/app/timer.ex +++ b/lib/app/timer.ex @@ -81,7 +81,7 @@ defmodule App.Timer do end @doc """ - Lists all the timer from a given item.id. + Lists all the timers from a given item.id. ## Examples @@ -93,6 +93,27 @@ defmodule App.Timer do |> Repo.all() end + @doc """ + Lists all the timers changesets from a given item.id. + This is useful for form validation, as it returns the timers in a changeset form, in which you can add errors. + + ## Examples + + iex> list_timers(1) + [ #Ecto.Changeset, valid?: true> ] + """ + def list_timers_changesets(item_id) do + from(v in Timer, where: [item_id: ^item_id], order_by: [asc: :id]) + |> Repo.all() + |> Enum.map(fn t -> + Timer.changeset(t, %{ + id: t.id, + start: t.start, + stop: t.stop, + item_id: t.item_id + }) end) + end + @doc """ `stop_timer_for_item_id/1` stops a timer for the given item_id if there is one. Fails silently if there is no timer for the given item_id. diff --git a/lib/app_web/live/app_live.ex b/lib/app_web/live/app_live.ex index 004a2e3a..35728ae2 100644 --- a/lib/app_web/live/app_live.ex +++ b/lib/app_web/live/app_live.ex @@ -77,7 +77,7 @@ defmodule AppWeb.AppLive do start: NaiveDateTime.utc_now() }) - AppWeb.Endpoint.broadcast(@topic, "update", :start) + AppWeb.Endpoint.broadcast(@topic, "update", {:start, item.id}) {:noreply, socket} end @@ -86,7 +86,7 @@ defmodule AppWeb.AppLive do timer_id = Map.get(data, "timerid") {:ok, _timer} = Timer.stop(%{id: timer_id}) - AppWeb.Endpoint.broadcast(@topic, "update", :stop) + AppWeb.Endpoint.broadcast(@topic, "update", {:stop, Map.get(data, "id")}) {:noreply, socket} end @@ -94,17 +94,7 @@ defmodule AppWeb.AppLive do def handle_event("edit-item", data, socket) do item_id = String.to_integer(data["id"]) - timers_list = Timer.list_timers(item_id) - - timers_list_changeset = - Enum.map(timers_list, fn t -> - Timer.changeset(t, %{ - id: t.id, - start: t.start, - stop: t.stop, - item_id: t.item_id - }) - end) + timers_list_changeset = Timer.list_timers_changesets(item_id) {:noreply, assign(socket, editing: item_id, editing_timers: timers_list_changeset)} @@ -130,6 +120,66 @@ defmodule AppWeb.AppLive do end @impl true + def handle_event( + "update-item-timer", + %{ + "timer_id" => id, + "index" => index, + "timer_start" => timer_start, + "timer_stop" => timer_stop + }, + socket + ) when timer_stop == "" do + + timer_changeset_list = socket.assigns.editing_timers + index = String.to_integer(index) + changeset_obj = Enum.at(timer_changeset_list, index) + + try do + start = + App.DateTimeParser.parse!(timer_start, "%Y-%m-%dT%H:%M:%S") + |> DateTime.to_naive() + + other_timers_list = + List.delete_at(socket.assigns.editing_timers, index) + + max_end = other_timers_list |> Enum.map(fn chs -> chs.data.stop end) |> Enum.max() + + case NaiveDateTime.compare(start, max_end) do + :gt -> + Timer.update_timer(%{id: id, start: start, stop: nil}) + {:noreply, assign(socket, editing: nil, editing_timers: [])} + _ -> + updated_changeset_timers_list = + error_timer_changeset( + timer_changeset_list, + changeset_obj, + index, + :id, + "When editing an ongoing timer, make sure it's after all the others." + ) + + {:noreply, + assign(socket, editing_timers: updated_changeset_timers_list)} + + end + + rescue + _e -> + updated_changeset_timers_list = + error_timer_changeset( + timer_changeset_list, + changeset_obj, + index, + :id, + "Date format invalid on either start or stop." + ) + + {:noreply, + assign(socket, editing_timers: updated_changeset_timers_list)} + end + end + def handle_event( "update-item-timer", %{ @@ -270,11 +320,33 @@ defmodule AppWeb.AppLive do end @impl true - def handle_info(%Broadcast{event: "update", payload: _message}, socket) do + def handle_info(%Broadcast{event: "update", payload: payload}, socket) do + person_id = get_person_id(socket.assigns) items = Item.items_with_timers(person_id) - {:noreply, assign(socket, items: items)} + isEditingItem = socket.assigns.editing + + # If the item is being edited, we update the timer list of the item being edited. + if isEditingItem do + case payload do + {:start, item_id} -> + timers_list_changeset = Timer.list_timers_changesets(item_id) + {:noreply, assign(socket, items: items, editing: item_id, editing_timers: timers_list_changeset)} + + + {:stop, item_id} -> + timers_list_changeset = Timer.list_timers_changesets(item_id) + {:noreply, assign(socket, items: items, editing: item_id, editing_timers: timers_list_changeset)} + + _ -> + {:noreply, assign(socket, items: items)} + end + + # If not, just update the item list. + else + {:noreply, assign(socket, items: items)} + end end # only show certain UI elements (buttons) if there are items: diff --git a/lib/app_web/live/app_live.html.heex b/lib/app_web/live/app_live.html.heex index 316ec07e..db127c54 100644 --- a/lib/app_web/live/app_live.html.heex +++ b/lib/app_web/live/app_live.html.heex @@ -256,7 +256,6 @@