From 7d5a8be2c6b898a673bde6b7602eb53b517d0263 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Lui=CC=81s=20Arteiro?= Date: Thu, 30 Mar 2023 14:22:29 +0100 Subject: [PATCH 01/94] feat: Adding position field and functions to change the position after adding a new item. #145 --- lib/app/item.ex | 55 ++++++++++++++++++- .../20220627162154_create_items.exs | 1 + 2 files changed, 54 insertions(+), 2 deletions(-) diff --git a/lib/app/item.ex b/lib/app/item.ex index 7f68cd13..cfff19e2 100644 --- a/lib/app/item.ex +++ b/lib/app/item.ex @@ -13,6 +13,7 @@ defmodule App.Item do field :person_id, :integer field :status, :integer field :text, :string + field :position, :integer has_many :timer, Timer many_to_many(:tags, Tag, join_through: ItemTag, on_replace: :delete) @@ -51,7 +52,11 @@ defmodule App.Item do """ def create_item(attrs) do - %Item{} + + ## Make room at beginning of list first. + reorder_list_after_adding_item(%Item{position: -1}) + + %Item{position: 0} |> changeset(attrs) |> PaperTrail.insert(originator: %{id: Map.get(attrs, :person_id, 0)}) end @@ -68,7 +73,11 @@ defmodule App.Item do {:error, %Ecto.Changeset{}} """ def create_item_with_tags(attrs) do - %Item{} + + ## Make room at beginning of list first. + reorder_list_after_adding_item(%Item{position: -1}) + + %Item{position: 0} |> changeset_with_tags(attrs) |> PaperTrail.insert(originator: %{id: Map.get(attrs, :person_id, 0)}) end @@ -186,6 +195,48 @@ defmodule App.Item do |> Repo.update() end + + @doc """ + Moves the item from a X position to Y position. + It does this by reordering the list. + + Please see method #1 of + https://betterprogramming.pub/the-best-way-to-update-a-drag-and-drop-sorting-list-through-database-schemas-31bed7371cd0 + """ + def move_item(item_id, to_position) do + item = get_item!(item_id) + placeholder = %Item{item | position: to_position} + + Repo.transaction(fn -> + reorder_list_after_removing_item(item) + reorder_list_after_adding_item(placeholder) + update_item(item, %{position: to_position}) + end) + end + + defp reorder_list_after_adding_item(%Item{position: position}) do + # Increments the positions above a given position. + # We are making space for the item to be added. + + from(i in Item, + where: i.position > ^position, + update: [inc: [position: 1]] + ) + |> Repo.update_all([]) + end + + defp reorder_list_after_removing_item(%Item{position: position}) do + # Decrements the positions above a given position. + # We are making all the positions above the given position decrement so they stay sequential. + + from(i in Item, + where: i.position > ^position, + update: [inc: [position: -1]] + ) + |> Repo.update_all([]) + end + + # 🐲 H E R E B E D R A G O N S! πŸ‰ # ⏳ Working with Time is all Dragons! πŸ™„ # πŸ‘©β€πŸ’» Feedback/Pairing/Refactoring Welcome! πŸ™ diff --git a/priv/repo/migrations/20220627162154_create_items.exs b/priv/repo/migrations/20220627162154_create_items.exs index af518729..904a2172 100644 --- a/priv/repo/migrations/20220627162154_create_items.exs +++ b/priv/repo/migrations/20220627162154_create_items.exs @@ -6,6 +6,7 @@ defmodule App.Repo.Migrations.CreateItems do add(:text, :string) add(:person_id, :integer) add(:status, :integer) + add(:position, :integer) timestamps() end From 9b1a7f5806326152654ba4e08c9e233cbff4dd5a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Lui=CC=81s=20Arteiro?= Date: Fri, 31 Mar 2023 10:40:45 +0100 Subject: [PATCH 02/94] feat: Trying to get the highlight to work on the client of Phoenix. #145 --- assets/css/app.css | 14 +++++ assets/js/app.js | 60 +++++++++++++++++++++ lib/app/item.ex | 3 +- lib/app_web/live/app_live.ex | 55 ++++++++++++++++++- lib/app_web/live/app_live.html.heex | 11 +++- lib/app_web/templates/layout/root.html.heex | 2 +- 6 files changed, 141 insertions(+), 4 deletions(-) diff --git a/assets/css/app.css b/assets/css/app.css index eca5b057..1b9aaae5 100644 --- a/assets/css/app.css +++ b/assets/css/app.css @@ -85,3 +85,17 @@ input[type=radio].has-error:not(.phx-no-feedback) { } [x-cloak] { display: none !important; } + + +/* For the drag and drop feature */ +.cursor-grab{ + cursor: grab; +} + +.cursor-grabbing{ + cursor: grabbing; +} + +.bg-yellow-300{ + background-color: rgb(253 224 71); +} \ No newline at end of file diff --git a/assets/js/app.js b/assets/js/app.js index 5a396ef5..d084810f 100644 --- a/assets/js/app.js +++ b/assets/js/app.js @@ -7,6 +7,33 @@ import topbar from "../vendor/topbar" let Hooks = {} +Hooks.Items = { + mounted() { + const hook = this + + this.el.addEventListener("highlight", e => { + hook.pushEventTo("#items", "highlight", {id: e.detail.id}) + }) + + this.el.addEventListener("remove-highlight", e => { + hook.pushEventTo("#items", "remove-highlight", {id: e.detail.id}) + }) + + this.el.addEventListener("dragoverItem", e => { + const currentItemId = e.detail.currentItemId + const selectedItemId = e.detail.selectedItemId + if( currentItemId != selectedItemId) { + hook.pushEventTo("#items", "dragoverItem", {currentItemId: currentItemId, selectedItemId: selectedItemId}) + } + }) + + this.el.addEventListener("update-indexes", e => { + const ids = [...document.querySelectorAll(".item")].map( i => i.dataset.id) + hook.pushEventTo("#items", "updateIndexes", {ids: ids}) + }) + } +} + let csrfToken = document.querySelector("meta[name='csrf-token']").getAttribute("content") let liveSocket = new LiveSocket("/live", Socket, { @@ -27,6 +54,39 @@ topbar.config({barColors: {0: "#29d"}, shadowColor: "rgba(0, 0, 0, .3)"}) window.addEventListener("phx:page-loading-start", info => topbar.show()) window.addEventListener("phx:page-loading-stop", info => topbar.hide()) +// Drag and drop highlight handlers +window.addEventListener("phx:highlight", (e) => { + document.querySelectorAll("[data-highlight]").forEach(el => { + if(el.id == e.detail.id) { + liveSocket.execJS(el, el.getAttribute("data-highlight")) + } + }) +}) + +window.addEventListener("phx:remove-highlight", (e) => { + document.querySelectorAll("[data-highlight]").forEach(el => { + if(el.id == e.detail.id) { + liveSocket.execJS(el, el.getAttribute("data-remove-highlight")) + } + }) +}) + +window.addEventListener("phx:dragover-item", (e) => { + const selectedItem = document.querySelector(`#${e.detail.selected_item_id}`) + const currentItem = document.querySelector(`#${e.detail.current_item_id}`) + + const items = document.querySelector('#items') + const listItems = [...document.querySelectorAll('.item')] + + if(listItems.indexOf(selectedItem) < listItems.indexOf(currentItem)){ + items.insertBefore(selectedItem, currentItem.nextSibling) + } + + if(listItems.indexOf(selectedItem) > listItems.indexOf(currentItem)){ + items.insertBefore(selectedItem, currentItem) + } +}) + // connect if there are any LiveViews on the page liveSocket.connect() diff --git a/lib/app/item.ex b/lib/app/item.ex index cfff19e2..9f2a0016 100644 --- a/lib/app/item.ex +++ b/lib/app/item.ex @@ -143,13 +143,14 @@ defmodule App.Item do """ def list_items do Item - |> order_by(desc: :inserted_at) + |> order_by(desc: :position) |> where([i], is_nil(i.status) or i.status != 6) |> Repo.all() end def list_person_items(person_id) do Item + |> order_by(desc: :position) |> where(person_id: ^person_id) |> Repo.all() |> Repo.preload(tags: from(t in Tag, order_by: t.text)) diff --git a/lib/app_web/live/app_live.ex b/lib/app_web/live/app_live.ex index e84ffe8d..77f20df7 100644 --- a/lib/app_web/live/app_live.ex +++ b/lib/app_web/live/app_live.ex @@ -3,9 +3,12 @@ defmodule AppWeb.AppLive do use AppWeb, :live_view use Timex alias App.{Item, Tag, Timer} + alias Phoenix.PubSub + alias Phoenix.Socket.Broadcast + alias Phoenix.LiveView.JS + # run authentication on mount on_mount(AppWeb.AuthController) - alias Phoenix.Socket.Broadcast @topic "live" @stats_topic "stats" @@ -220,6 +223,55 @@ defmodule AppWeb.AppLive do end end + @impl true + def handle_event("highlight", %{"id" => id}, socket) do + AppWeb.Endpoint.broadcast(@topic, "liveview_items", {:drag_item, id}) + {:noreply, socket} + end + + @impl true + def handle_event("remove-highlight", %{"id" => id}, socket) do + AppWeb.Endpoint.broadcast(@topic, "liveview_items", {:drop_item, id}) + {:noreply, socket} + end + + @impl true + def handle_event( + "dragoverItem", + %{"currentItemId" => current_item_id, "selectedItemId" => selected_item_id}, + socket + ) do + + AppWeb.Endpoint.broadcast(@topic, "liveview_items", {:dragover_item, {current_item_id, selected_item_id }}) + {:noreply, socket} + end + + + @impl true + def handle_info(%Broadcast{event: "liveview_items", payload: {:dragover_item, {current_item_id, selected_item_id}}}, socket) do + {:noreply, + push_event(socket, "dragover-item", %{ + current_item_id: current_item_id, + selected_item_id: selected_item_id + })} + end + + @impl true + def handle_info(%Broadcast{event: "liveview_items", payload: {:drag_item, item_id}}, socket) do + {:noreply, push_event(socket, "highlight", %{id: item_id})} + end + + @impl true + def handle_info(%Broadcast{event: "liveview_items", payload: {:drop_item, item_id}}, socket) do + {:noreply, push_event(socket, "remove-highlight", %{id: item_id})} + end + + @impl true + def handle_event("updateIndexes", %{"ids" => ids}, socket) do + #Tasks.update_items_index(ids) + {:noreply, socket} + end + @impl true def handle_info(%Broadcast{event: "update", payload: payload}, socket) do person_id = get_person_id(socket.assigns) @@ -268,6 +320,7 @@ defmodule AppWeb.AppLive do {:noreply, socket} end + # only show certain UI elements (buttons) if there are items: def has_items?(items), do: length(items) > 1 diff --git a/lib/app_web/live/app_live.html.heex b/lib/app_web/live/app_live.html.heex index 3ba31abd..90aab724 100644 --- a/lib/app_web/live/app_live.html.heex +++ b/lib/app_web/live/app_live.html.heex @@ -133,11 +133,20 @@ - @@ -418,7 +421,6 @@ <% else %> -