Make stuff sortable

This commit is contained in:
david 2024-11-11 04:04:13 +01:00
parent 50e853098b
commit ee5dbcf33e
21 changed files with 161 additions and 28 deletions

View file

@ -49,7 +49,12 @@ class ElementsController < ApplicationController
# PATCH/PUT /elements/1 # PATCH/PUT /elements/1
def update def update
if @element.update(element_params) if @element.update(element_params)
redirect_to @element, notice: "Element was successfully updated.", status: :see_other respond_to do |format|
format.turbo_stream
format.html do
redirect_to @element, notice: "Element was successfully updated.", status: :see_other
end
end
else else
render :edit, status: :unprocessable_entity render :edit, status: :unprocessable_entity
end end
@ -82,6 +87,6 @@ class ElementsController < ApplicationController
# Only allow a list of trusted parameters through. # Only allow a list of trusted parameters through.
def element_params def element_params
params.require(:element).permit(:page_id, :title, :description) params.require(:element).permit(:page_id, :title, :description, :position)
end end
end end

View file

@ -35,7 +35,12 @@ class PagesController < ApplicationController
# PATCH/PUT /pages/1 # PATCH/PUT /pages/1
def update def update
if @page.update(page_params) if @page.update(page_params)
redirect_to @page, notice: "Page was successfully updated.", status: :see_other respond_to do |format|
format.turbo_stream
format.html do
redirect_to @page, notice: "Page was successfully updated.", status: :see_other
end
end
else else
render :edit, status: :unprocessable_entity render :edit, status: :unprocessable_entity
end end

View file

@ -115,7 +115,7 @@ class SuccessCriteriaController < ApplicationController
# Only allow a list of trusted parameters through. # Only allow a list of trusted parameters through.
def success_criterion_params def success_criterion_params
params.require(:success_criterion).permit(:element_id, :title, :quick_criterion, :quick_fail, :quick_fix, :priority, :level, :result, :test_comment, :check_id) params.require(:success_criterion).permit(:element_id, :title, :quick_criterion, :quick_fail, :quick_fix, :priority, :level, :result, :test_comment, :check_id, :position)
end end
def set_element def set_element

View file

@ -34,6 +34,12 @@ application.register("rich-text-link-targets", RichTextLinkTargetsController)
import SetThemeController from "./set_theme_controller" import SetThemeController from "./set_theme_controller"
application.register("set-theme", SetThemeController) application.register("set-theme", SetThemeController)
import SortElementsController from "./sort_elements_controller"
application.register("sort-elements", SortElementsController)
import SortableController from "./sortable_controller"
application.register("sortable", SortableController)
import ThemeSwitcherController from "./theme_switcher_controller" import ThemeSwitcherController from "./theme_switcher_controller"
application.register("theme-switcher", ThemeSwitcherController) application.register("theme-switcher", ThemeSwitcherController)

View file

@ -0,0 +1,7 @@
import { Controller } from "@hotwired/stimulus"
// Connects to data-controller="sort-elements"
export default class extends Controller {
connect() {
}
}

View file

@ -0,0 +1,55 @@
import { Controller } from "@hotwired/stimulus"
import Sortable from "sortablejs"
import { put } from "@rails/request.js";
// Connects to data-controller="sortable"
export default class extends Controller {
linkedElement = null
connect() {
this.element.style.cursor = "grab"
console.log("dataset", this.element.dataset)
if (this.element.dataset["linkedElementId"]) {
this.linkedElement = document.getElementById(this.element.dataset["linkedElementId"])
console.log("Has a linked element", this.linkedElement)
}
new Sortable(this.element, {
group: this.groupValue,
onEnd: this.onEndFactory(this.linkedElement),
})
}
onEndFactory(linkedElement) {
return function (event) {
const position = event.newIndex + 1
const url = event.item.dataset["sortableUrl"]
const formName = event.item.dataset["formName"]
const positionAttribute = event.item.dataset["positionAttribute"]
let body = {}
body[formName] = {}
body[formName][positionAttribute] = position
console.log("event", event, "url", url)
// Expect backend to update list items via turbo if necessary
put(url, {
body: JSON.stringify(body),
contentType: "application/json",
headers: {
"Accept": "text/vnd.turbo-stream.html, text/html, application/xhtml+xml"
}
})
console.log(linkedElement)
if (linkedElement) {
console.log("move linked", linkedElement)
let children = linkedElement.children
let child = children[event.oldIndex]
let newAfter = children[event.newIndex]
if (event.oldIndex < event.newIndex) {
newAfter = children[event.newIndex + 1]
}
console.log("move ", child, "before", newAfter)
child.parentNode.insertBefore(child, newAfter)
}
}
}
}

View file

@ -7,6 +7,7 @@ class ChecklistEntry < ApplicationRecord
before_validation :set_position before_validation :set_position
before_update :update_positions, if: :position_changed? before_update :update_positions, if: :position_changed?
private
def set_position def set_position
self.position ||= (checklist.checklist_entries.pluck(:position).max || 0) + 1 self.position ||= (checklist.checklist_entries.pluck(:position).max || 0) + 1
end end

View file

@ -8,7 +8,8 @@ class Element < ApplicationRecord
delegate :report, to: :page delegate :report, to: :page
after_validation :set_position before_validation :set_position
before_update :update_positions, if: :position_changed?
# Calculate actual conformity level: # Calculate actual conformity level:
# - if a success_criterion has result :failed -> the confirmity_level # - if a success_criterion has result :failed -> the confirmity_level
@ -33,12 +34,20 @@ class Element < ApplicationRecord
@max_level ||= success_criteria.reject(&:not_applicable?).max(&:level) @max_level ||= success_criteria.reject(&:not_applicable?).max(&:level)
end end
def number
"#{page.position}.#{position}"
end
private
def set_position def set_position
Rails.logger.debug("element: position #{position}") Rails.logger.debug("element: position #{position}")
self.position ||= (page.elements.pluck(:position).max || 0) + 1 self.position ||= (page.elements.pluck(:position).max || 0) + 1
end end
def number def update_positions
"#{page.position}.#{position}" if position_was
page.elements.where("position > ?", position_was).update_all("position = position - 1")
end
page.elements.where(position: position..).update_all("position = position + 1")
end end
end end

View file

@ -1,13 +1,22 @@
class Page < ApplicationRecord class Page < ApplicationRecord
belongs_to :report, touch: true belongs_to :report, touch: true
has_many :elements, dependent: :destroy has_many :elements, -> { order(:position) }, dependent: :destroy
has_rich_text :comment has_rich_text :comment
before_validation :set_position before_validation :set_position
before_update :update_positions, if: :position_changed?
private private
def set_position def set_position
self.position ||= (report.pages.pluck(:position).max || 0) + 1 self.position ||= (report.pages.pluck(:position).max || 0) + 1
end end
def update_positions
if position_was
report.pages.where("position > ?", position_was).update_all("position = position - 1")
end
report.pages.where(position: position..).update_all("position = position + 1")
end
end end

View file

@ -1,7 +1,7 @@
# frozen_string_literal: true # frozen_string_literal: true
class Report < ApplicationRecord class Report < ApplicationRecord
has_many :pages, dependent: :destroy has_many :pages, -> { order(:position) }, dependent: :destroy
has_many :elements, through: :pages, dependent: :destroy has_many :elements, through: :pages, dependent: :destroy
has_rich_text :comment has_rich_text :comment

View file

@ -15,6 +15,7 @@ class SuccessCriterion < ApplicationRecord
enum :level, %i[A AA AAA] enum :level, %i[A AA AAA]
before_save :set_position before_save :set_position
before_update :update_positions, if: :position_changed?
def level_value def level_value
return nil unless level return nil unless level
@ -26,9 +27,20 @@ class SuccessCriterion < ApplicationRecord
"HEADER" "HEADER"
end end
def number
[ page.position, element.position, position ].join(".")
end
private private
def set_position def set_position
self.position ||= (element.success_criteria.pluck(:position).max || 0) + 1 self.position ||= (element.success_criteria.pluck(:position).max || 0) + 1
Rails.logger.debug("set position: "+position.to_s) Rails.logger.debug("set position: "+position.to_s)
end end
def update_positions
if position_was
element.success_criteria.where("position > ?", position_was).update_all("position = position - 1")
end
element.success_criteria.where(position: position..).update_all("position = position + 1")
end
end end

View file

@ -26,7 +26,7 @@
</p> </p>
<% end %> <% end %>
<div id="<%= dom_id(element, :success_criteria_list) %>" class="mb-3"> <div id="<%= dom_id(element, :success_criteria_list) %>" class="mb-3" data-controller="sortable" data-form-name="success_criterion" data-position-attribute= "position">
<% element.success_criteria.each do |sc| %> <% element.success_criteria.each do |sc| %>
<%= render sc %> <%= render sc %>
<% end %> <% end %>

View file

@ -1,9 +1,11 @@
li id=dom_id(element, :page_nav_row) li id=dom_id(element, :page_nav_row) data={ "sortable-url": element_path(element), "form-name": "element", "position-attribute": "position" }
- if current_page - if current_page
=< link_to("##{dom_id(element)}", data: { "turbo": false }) do =< link_to("##{dom_id(element)}", data: { "turbo": false }) do
i.bi.bi-boxes.me-1 i.bi.bi-boxes.me-1
=< "#{element.number} #{element.title}" span id=dom_id(element, :page_nav_title)
= "#{element.number} #{element.title}"
- else - else
=< link_to(report_path(element.report, page_id: element.page.id, anchor: dom_id(element)), data: { "turbo": false }) do =< link_to(report_path(element.report, page_id: element.page.id, anchor: dom_id(element)), data: { "turbo": false }) do
i.bi.bi-boxes.me-1 i.bi.bi-boxes.me-1
=< "#{element.number} #{element.title}" span id=dom_id(element, :page_nav_title)
=< "#{element.number} #{element.title}"

View file

@ -0,0 +1,7 @@
- @element.page.elements.each do |element|
= turbo_stream.update dom_id(element, :page_nav_title), "#{element.number} #{element.title}"
= turbo_stream.update dom_id(element, :title), "#{element.page.position}.#{element.position} #{element.title}"
/ - element.success_criteria.each do |sc|
/ = turbo_stream.update dom_id(sc, :title), "#{sc.page.position}.#{sc.element.position}.#{sc.position} #{sc.title}"
- element.success_criteria.each do |sc|
= turbo_stream.update dom_id(sc, :position), "#{sc.page.position}.#{sc.element.position}.#{sc.position}"

View file

@ -1,9 +1,9 @@
<div id="<%= dom_id page %>" class="mb-3" data-bs-scrollspy-target="#<%= dom_id(page.report, :page_nav_spy) %>" data-controller="bs-scrollspy"> <div id="<%= dom_id page %>" class="mb-3" data-bs-scrollspy-target="#<%= dom_id(page.report, :page_nav_spy) %>" data-controller="bs-scrollspy">
<div class="text-end">
<a href="#" data-action="click->details-list#closeAll" data-controller="hotkey" data-hotkey="z">Alle zu [z]</a>
<a href="#" data-action="click->details-list#openAll" data-controller="hotkey" data-hotkey="a">Alle auf [a]</a>
</div>
<div id="element_list" data-controller="details-list"> <div id="element_list" data-controller="details-list">
<div class="text-end">
<a href="#" data-action="click->details-list#closeAll" data-controller="hotkey" data-hotkey="z">Alle zu [z]</a>
<a href="#" data-action="click->details-list#openAll" data-controller="hotkey" data-hotkey="a">Alle auf [a]</a>
</div>
<% page.elements.each do |element| %> <% page.elements.each do |element| %>
<%= render element %> <%= render element %>
<% end %> <% end %>

View file

@ -0,0 +1,7 @@
- @page.report.pages.each do |page|
= turbo_stream.update dom_id(page, :title), "#{page.position} #{page.path}"
- page.elements.each do |element|
= turbo_stream.update dom_id(element, :title), "#{element.page.position}.#{element.position} #{element.title}"
= turbo_stream.replace dom_id(element, :page_nav_row), partial: "elements/page_nav_row", locals: { element: element, current_page: element.page == @page}
- element.success_criteria.each do |sc|
= turbo_stream.update dom_id(sc, :position), "#{sc.page.position}.#{sc.element.position}.#{sc.position}"

View file

@ -9,19 +9,20 @@
- if @page_nav_mode != :comment || current_page.nil? - if @page_nav_mode != :comment || current_page.nil?
- if report.pages.any? - if report.pages.any?
nav.mt-3 id=dom_id(report, :page_nav_spy) nav.mt-3 id=dom_id(report, :page_nav_spy)
ul ul data={ controller: :sortable }
- report.pages.each do |page| - report.pages.each do |page|
- is_current = current_page == page - is_current = current_page == page
li li data={ "sortable-url": page_path(page), "form-name": "page", "position-attribute": "position" }
details.tree open=current_page_displayed(page) class="" details.tree open=current_page_displayed(page) class=""
summary class=(is_current ? "active" : nil) summary class=(is_current ? "active" : nil)
.content .content
i.bi.me-1 class="bi-file-earmark-check#{is_current ? "" : "" }" i.bi.me-1 class="bi-file-earmark-check#{is_current ? "" : "" }"
- if is_current - if is_current
=< "#{page.position} #{page.path}" span id=dom_id(page, :title) =< "#{page.position} #{page.path}"
- else - else
=< link_to("#{page.position} #{page.path}", report_path(report, page_id: page.id), data: { "turbo-frame": :_top }) =< link_to(report_path(report, page_id: page.id), data: { "turbo-frame": :_top }) do
ul id=dom_id(page, :page_nav_elements) span id=dom_id(page, :title) =< "#{page.position} #{page.path}"
ul id=dom_id(page, :page_nav_elements) data={ controller: "sortable", "linked-element-id": "element_list" }
- page.elements.each do |element| - page.elements.each do |element|
= render partial: "elements/page_nav_row", locals: { element: element, current_page: current_page == element.page } = render partial: "elements/page_nav_row", locals: { element: element, current_page: current_page == element.page }
/li /li

View file

@ -1,5 +1,5 @@
/ = turbo_frame_tag(dom_id(success_criterion, :frame)) do / = turbo_frame_tag(dom_id(success_criterion, :frame)) do
- expanded = false unless defined?(expanded) - expanded = false unless defined?(expanded)
details.success_criterion id="#{dom_id(success_criterion)}" class="#{success_criterion.result}" details.success_criterion id="#{dom_id(success_criterion)}" class="#{success_criterion.result}" data={ "sortable-url": success_criterion_path(success_criterion), "form-name": :success_criterion, "position-attribute": "position" }
== render partial: "success_criteria/header", locals: {success_criterion: success_criterion } == render partial: "success_criteria/header", locals: {success_criterion: success_criterion }
== render partial: "success_criteria/body", locals: {success_criterion: success_criterion } == render partial: "success_criteria/body", locals: {success_criterion: success_criterion }

View file

@ -1,2 +1,5 @@
= turbo_stream.replace dom_id(@success_criterion, :header), partial: "success_criteria/header", locals: {success_criterion: @success_criterion } = turbo_stream.replace dom_id(@success_criterion, :header), partial: "success_criteria/header", locals: {success_criterion: @success_criterion }
= turbo_stream.replace dom_id(@success_criterion, :body), partial: "success_criteria/body", locals: {success_criterion: @success_criterion } = turbo_stream.replace dom_id(@success_criterion, :body), partial: "success_criteria/body", locals: {success_criterion: @success_criterion }
- @success_criterion.element.success_criteria.each do |sc|
= turbo_stream.update(dom_id(sc, :position), sc.number)

View file

@ -0,0 +1,7 @@
class RemovePositionIndices < ActiveRecord::Migration[8.0]
def change
remove_index :success_criteria, column: [:element_id, :position]
remove_index :elements, column: [:page_id, :position]
remove_index :pages, column: [:report_id, :position]
end
end

5
db/schema.rb generated
View file

@ -10,7 +10,7 @@
# #
# It's strongly recommended that you check this file into your version control system. # It's strongly recommended that you check this file into your version control system.
ActiveRecord::Schema[8.0].define(version: 2024_11_08_180654) do ActiveRecord::Schema[8.0].define(version: 2024_11_11_011637) do
create_table "action_text_rich_texts", force: :cascade do |t| create_table "action_text_rich_texts", force: :cascade do |t|
t.string "name", null: false t.string "name", null: false
t.text "body" t.text "body"
@ -115,7 +115,6 @@ ActiveRecord::Schema[8.0].define(version: 2024_11_08_180654) do
t.datetime "updated_at", null: false t.datetime "updated_at", null: false
t.integer "page_id", null: false t.integer "page_id", null: false
t.integer "position", null: false t.integer "position", null: false
t.index ["page_id", "position"], name: "index_elements_on_page_id_and_position", unique: true
t.index ["page_id"], name: "index_elements_on_page_id" t.index ["page_id"], name: "index_elements_on_page_id"
end end
@ -145,7 +144,6 @@ ActiveRecord::Schema[8.0].define(version: 2024_11_08_180654) do
t.datetime "created_at", null: false t.datetime "created_at", null: false
t.datetime "updated_at", null: false t.datetime "updated_at", null: false
t.text "notes" t.text "notes"
t.index ["report_id", "position"], name: "index_pages_on_report_id_and_position", unique: true
t.index ["report_id"], name: "index_pages_on_report_id" t.index ["report_id"], name: "index_pages_on_report_id"
end end
@ -192,7 +190,6 @@ ActiveRecord::Schema[8.0].define(version: 2024_11_08_180654) do
t.integer "priority" t.integer "priority"
t.integer "position", null: false t.integer "position", null: false
t.index ["check_id"], name: "index_success_criteria_on_check_id" t.index ["check_id"], name: "index_success_criteria_on_check_id"
t.index ["element_id", "position"], name: "index_success_criteria_on_element_id_and_position", unique: true
t.index ["element_id"], name: "index_success_criteria_on_element_id" t.index ["element_id"], name: "index_success_criteria_on_element_id"
end end