defmodule Shift73kWeb.ShiftAssignLive.Index do use Shift73kWeb, :live_view import Shift73k, only: [get_app_time_zone: 0] alias Shift73k.Repo alias Shift73k.Shifts alias Shift73k.Shifts.Templates alias Shift73k.Shifts.Templates.ShiftTemplate @custom_shift %ShiftTemplate{subject: "Custom shift", id: "custom-shift"} @custom_shift_opt {@custom_shift.subject, @custom_shift.id} @impl true def mount(_params, session, socket) do socket |> assign_defaults(session) |> assign(:page_title, "Schedule Shifts") |> assign(:custom_shift, @custom_shift) |> assign(:show_template_btn_active, false) |> assign(:show_template_details, false) |> assign(:selected_days, []) |> assign(:delete_days_shifts, nil) |> live_okreply() end @impl true def handle_params(_params, _url, socket) do socket |> init_shift_templates() |> init_shift_template() |> show_details_if_custom() |> assign_shift_length() |> assign_shift_template_changeset() |> assign_modal_close_handlers() |> init_today(Date.utc_today()) |> init_day_names() |> update_calendar() |> live_noreply() end defp assign_modal_close_handlers(socket) do to = Routes.shift_assign_index_path(socket, :index) assign(socket, modal_return_to: to, modal_close_action: :return) end defp get_shift_template("custom-shift"), do: @custom_shift defp get_shift_template(template_id), do: Templates.get_shift_template(template_id) defp assign_shift_length(%{assigns: %{shift_template: shift}} = socket) do assign(socket, :shift_length, format_shift_length(shift)) end defp assign_known_shifts(%{assigns: %{current_user: user}} = socket) do date_range = socket.assigns.date_range known_shifts = Shifts.list_shifts_by_user_in_date_range(user.id, date_range) assign(socket, :known_shifts, known_shifts) end defp init_today(socket, today) do assign(socket, current_date: today, cursor_date: cursor_date(today)) end defp cursor_date(%Date{} = date) do date |> Date.beginning_of_month() |> Date.add(4) end defp assign_shift_template_changeset(%{assigns: %{shift_template: shift}} = socket) do cs = Templates.change_shift_template(shift) assign(socket, :shift_template_changeset, cs) end defp init_shift_template(socket) do first_list_id = socket.assigns.shift_templates |> hd() |> elem(1) fave_id = socket.assigns.current_user.fave_shift_template_id assign_shift_template(socket, fave_id || first_list_id) end defp assign_shift_template(socket, template_id) do assign(socket, :shift_template, get_shift_template(template_id)) end defp init_shift_templates(%{assigns: %{current_user: user}} = socket) do shift_templates = Templates.list_shift_templates_by_user(user.id) |> Stream.map(fn t -> shift_template_option(t, user.fave_shift_template_id) end) |> Enum.concat([@custom_shift_opt]) assign(socket, :shift_templates, shift_templates) end defp shift_template_option(template, fave_id) do label = template.subject <> " (" <> format_shift_time(template.time_start) <> "—" <> format_shift_time(template.time_end) <> ")" label = case fave_id == template.id do true -> label <> " ★" false -> label end {label, template.id} end defp init_day_names(%{assigns: %{current_user: user, current_date: today}} = socket) do week_start = Date.beginning_of_week(today, user.week_start_at) day_names = week_start |> Date.range(Date.add(week_start, 6)) |> Enum.map(&Calendar.strftime(&1, "%a")) assign(socket, :day_names, day_names) end defp date_range(cursor_date, week_start_at) do last = cursor_date |> Date.end_of_month() |> Date.end_of_week(week_start_at) cursor_date |> Date.beginning_of_month() |> Date.beginning_of_week(week_start_at) |> Date.range(last) end defp assign_date_range(%{assigns: %{current_user: user}} = socket) do date_range = date_range(socket.assigns.cursor_date, user.week_start_at) assign(socket, :date_range, date_range) end defp assign_week_rows(%{assigns: %{date_range: date_range}} = socket) do assign(socket, :week_rows, Enum.chunk_every(date_range, 7)) end def day_color(day, current_date, cursor_date, selected_days) do cond do Enum.member?(selected_days, Date.to_string(day)) -> cond do Date.compare(day, current_date) == :eq -> "bg-triangle-info text-white" day.month != cursor_date.month -> "bg-triangle-light text-gray" true -> "bg-triangle-white" end Date.compare(day, current_date) == :eq -> "bg-info text-white" day.month != cursor_date.month -> "bg-light text-gray" true -> "" end end defp prep_template_params(params, current_user) do params |> Map.put("time_start", Time.from_iso8601!("T#{params["time_start"]}:00")) |> Map.put("time_end", Time.from_iso8601!("T#{params["time_end"]}:00")) |> Map.put("user_id", current_user.id) end defp show_details_if_custom(socket) do if socket.assigns.shift_template.id != @custom_shift.id || socket.assigns.show_template_details do socket else socket |> assign(:show_template_btn_active, true) |> push_event("toggle-template-details", %{targetId: "#templateDetailsCol"}) end end defp update_calendar(socket) do socket |> assign_date_range() |> assign_week_rows() |> assign_known_shifts() end defp new_nav_cursor("now", _cursor_date), do: Date.utc_today() defp new_nav_cursor(nav, cursor_date) do cursor_date |> Date.add((nav == "prev" && -30) || 30) |> cursor_date() end @impl true def handle_event("validate-shift-template", %{"shift_template" => params}, socket) do params = prep_template_params(params, socket.assigns.current_user) shift_template = socket.assigns.shift_template cs = shift_template |> Templates.change_shift_template(params) |> Map.put(:action, :validate) socket |> assign(:shift_template_changeset, cs) |> assign(:shift_template, Map.merge(shift_template, cs.changes)) |> assign_shift_length() |> live_noreply() end @impl true def handle_event( "change-selected-template", %{"template_select" => %{"template" => template_id}}, socket ) do socket |> assign_shift_template(template_id) |> show_details_if_custom() |> assign_shift_length() |> assign_shift_template_changeset() |> live_noreply() end @impl true def handle_event("month-nav", %{"month" => nav}, socket) do new_cursor = new_nav_cursor(nav, socket.assigns.cursor_date) socket |> assign(:cursor_date, new_cursor) |> update_calendar() |> live_noreply() end @impl true def handle_event("toggle-template-details", %{"target_id" => target_id}, socket) do socket |> assign(:show_template_btn_active, !socket.assigns.show_template_btn_active) |> push_event("toggle-template-details", %{targetId: target_id}) |> live_noreply() end @impl true def handle_event("collapse-shown", %{"target_id" => _target_id}, socket) do {:noreply, assign(socket, :show_template_details, true)} end @impl true def handle_event("collapse-hidden", %{"target_id" => _target_id}, socket) do {:noreply, assign(socket, :show_template_details, false)} end @impl true def handle_event("select-day", %{"day" => day}, socket) do selected_days = update_selected_days(socket.assigns.selected_days, day) {:noreply, assign(socket, :selected_days, selected_days)} end @impl true def handle_event("delete-days-shifts", _params, socket) do socket |> assign(:modal_close_action, :delete_days_shifts) |> assign(:delete_days_shifts, socket.assigns.selected_days) |> live_noreply() end @impl true def handle_event("clear-days", _params, socket) do {:noreply, assign(socket, :selected_days, [])} end @impl true def handle_event("save-days", _params, socket) do # 1. collect attrs from loaded shift template shift_data = ShiftTemplate.attrs(socket.assigns.shift_template) # 2. fashion list of shift attrs and insert socket.assigns.selected_days |> Stream.map(&Date.from_iso8601!/1) |> Stream.map(&Map.put(shift_data, :date, &1)) |> Enum.map(&Repo.timestamp/1) |> Shifts.create_multiple() |> handle_create_multiple_result(socket) |> assign(:selected_days, []) |> assign_known_shifts() |> live_noreply() end @impl true def handle_info({:put_flash_message, {flash_type, msg}}, socket) do socket |> put_flash(flash_type, msg) |> live_noreply() end @impl true def handle_info({:clear_selected_days, _}, socket) do socket |> assign(:selected_days, []) |> live_noreply() end @impl true def handle_info({:close_modal, _}, %{assigns: %{modal_close_action: :return}} = socket) do socket |> copy_flash() |> push_patch(to: socket.assigns.modal_return_to) |> live_noreply() end @impl true def handle_info({:close_modal, _}, %{assigns: %{modal_close_action: assign_key}} = socket) do socket |> assign(assign_key, nil) |> assign_modal_close_handlers() |> assign_known_shifts() |> live_noreply() end defp handle_create_multiple_result(result, socket) do day_count = length(socket.assigns.selected_days) {status, msg} = case result do {:error, errmsg} -> {:error, "Ope, problem error inserting shifts, page the dev! Message: #{errmsg}"} {n, _} -> s = (n > 1 && "s") || "" if n == day_count do {:success, "Successfully assigned shift to #{n} day#{s}"} else {:warning, "Some error, only #{n} day#{s} inserted but #{day_count} were selected"} end end put_flash(socket, status, msg) end def shifts_to_show(day_shifts) do if length(day_shifts) == 1 || length(day_shifts) > 2, do: Enum.take(day_shifts, 1), else: day_shifts end defp update_selected_days(selected_days, day) do case Enum.member?(selected_days, day) do false -> [day | selected_days] true -> Enum.reject(selected_days, fn d -> d == day end) end end end