Make stuff sortable
This commit is contained in:
parent
50e853098b
commit
ee5dbcf33e
21 changed files with 161 additions and 28 deletions
|
|
@ -49,7 +49,12 @@ class ElementsController < ApplicationController
|
|||
# PATCH/PUT /elements/1
|
||||
def update
|
||||
if @element.update(element_params)
|
||||
respond_to do |format|
|
||||
format.turbo_stream
|
||||
format.html do
|
||||
redirect_to @element, notice: "Element was successfully updated.", status: :see_other
|
||||
end
|
||||
end
|
||||
else
|
||||
render :edit, status: :unprocessable_entity
|
||||
end
|
||||
|
|
@ -82,6 +87,6 @@ class ElementsController < ApplicationController
|
|||
|
||||
# Only allow a list of trusted parameters through.
|
||||
def element_params
|
||||
params.require(:element).permit(:page_id, :title, :description)
|
||||
params.require(:element).permit(:page_id, :title, :description, :position)
|
||||
end
|
||||
end
|
||||
|
|
|
|||
|
|
@ -35,7 +35,12 @@ class PagesController < ApplicationController
|
|||
# PATCH/PUT /pages/1
|
||||
def update
|
||||
if @page.update(page_params)
|
||||
respond_to do |format|
|
||||
format.turbo_stream
|
||||
format.html do
|
||||
redirect_to @page, notice: "Page was successfully updated.", status: :see_other
|
||||
end
|
||||
end
|
||||
else
|
||||
render :edit, status: :unprocessable_entity
|
||||
end
|
||||
|
|
|
|||
|
|
@ -115,7 +115,7 @@ class SuccessCriteriaController < ApplicationController
|
|||
|
||||
# Only allow a list of trusted parameters through.
|
||||
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
|
||||
|
||||
def set_element
|
||||
|
|
|
|||
|
|
@ -34,6 +34,12 @@ application.register("rich-text-link-targets", RichTextLinkTargetsController)
|
|||
import SetThemeController from "./set_theme_controller"
|
||||
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"
|
||||
application.register("theme-switcher", ThemeSwitcherController)
|
||||
|
||||
|
|
|
|||
7
app/javascript/controllers/sort_elements_controller.js
Normal file
7
app/javascript/controllers/sort_elements_controller.js
Normal file
|
|
@ -0,0 +1,7 @@
|
|||
import { Controller } from "@hotwired/stimulus"
|
||||
|
||||
// Connects to data-controller="sort-elements"
|
||||
export default class extends Controller {
|
||||
connect() {
|
||||
}
|
||||
}
|
||||
55
app/javascript/controllers/sortable_controller.js
Normal file
55
app/javascript/controllers/sortable_controller.js
Normal 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)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -7,6 +7,7 @@ class ChecklistEntry < ApplicationRecord
|
|||
before_validation :set_position
|
||||
before_update :update_positions, if: :position_changed?
|
||||
|
||||
private
|
||||
def set_position
|
||||
self.position ||= (checklist.checklist_entries.pluck(:position).max || 0) + 1
|
||||
end
|
||||
|
|
|
|||
|
|
@ -8,7 +8,8 @@ class Element < ApplicationRecord
|
|||
|
||||
delegate :report, to: :page
|
||||
|
||||
after_validation :set_position
|
||||
before_validation :set_position
|
||||
before_update :update_positions, if: :position_changed?
|
||||
|
||||
# Calculate actual conformity 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)
|
||||
end
|
||||
|
||||
def number
|
||||
"#{page.position}.#{position}"
|
||||
end
|
||||
|
||||
private
|
||||
def set_position
|
||||
Rails.logger.debug("element: position #{position}")
|
||||
self.position ||= (page.elements.pluck(:position).max || 0) + 1
|
||||
end
|
||||
|
||||
def number
|
||||
"#{page.position}.#{position}"
|
||||
def update_positions
|
||||
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
|
||||
|
|
|
|||
|
|
@ -1,13 +1,22 @@
|
|||
class Page < ApplicationRecord
|
||||
belongs_to :report, touch: true
|
||||
has_many :elements, dependent: :destroy
|
||||
has_many :elements, -> { order(:position) }, dependent: :destroy
|
||||
|
||||
has_rich_text :comment
|
||||
|
||||
before_validation :set_position
|
||||
before_update :update_positions, if: :position_changed?
|
||||
|
||||
|
||||
private
|
||||
def set_position
|
||||
self.position ||= (report.pages.pluck(:position).max || 0) + 1
|
||||
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
|
||||
|
|
|
|||
|
|
@ -1,7 +1,7 @@
|
|||
# frozen_string_literal: true
|
||||
|
||||
class Report < ApplicationRecord
|
||||
has_many :pages, dependent: :destroy
|
||||
has_many :pages, -> { order(:position) }, dependent: :destroy
|
||||
has_many :elements, through: :pages, dependent: :destroy
|
||||
has_rich_text :comment
|
||||
|
||||
|
|
|
|||
|
|
@ -15,6 +15,7 @@ class SuccessCriterion < ApplicationRecord
|
|||
enum :level, %i[A AA AAA]
|
||||
|
||||
before_save :set_position
|
||||
before_update :update_positions, if: :position_changed?
|
||||
|
||||
def level_value
|
||||
return nil unless level
|
||||
|
|
@ -26,9 +27,20 @@ class SuccessCriterion < ApplicationRecord
|
|||
"HEADER"
|
||||
end
|
||||
|
||||
def number
|
||||
[ page.position, element.position, position ].join(".")
|
||||
end
|
||||
|
||||
private
|
||||
def set_position
|
||||
self.position ||= (element.success_criteria.pluck(:position).max || 0) + 1
|
||||
Rails.logger.debug("set position: "+position.to_s)
|
||||
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
|
||||
|
|
|
|||
|
|
@ -26,7 +26,7 @@
|
|||
</p>
|
||||
<% 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| %>
|
||||
<%= render sc %>
|
||||
<% end %>
|
||||
|
|
|
|||
|
|
@ -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
|
||||
=< link_to("##{dom_id(element)}", data: { "turbo": false }) do
|
||||
i.bi.bi-boxes.me-1
|
||||
=< "#{element.number} #{element.title}"
|
||||
span id=dom_id(element, :page_nav_title)
|
||||
= "#{element.number} #{element.title}"
|
||||
- else
|
||||
=< 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
|
||||
span id=dom_id(element, :page_nav_title)
|
||||
=< "#{element.number} #{element.title}"
|
||||
7
app/views/elements/update.turbo_stream.slim
Normal file
7
app/views/elements/update.turbo_stream.slim
Normal 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}"
|
||||
|
|
@ -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="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>
|
||||
<div id="element_list" data-controller="details-list">
|
||||
<% page.elements.each do |element| %>
|
||||
<%= render element %>
|
||||
<% end %>
|
||||
|
|
|
|||
7
app/views/pages/update.turbo_stream.slim
Normal file
7
app/views/pages/update.turbo_stream.slim
Normal 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}"
|
||||
|
|
@ -9,19 +9,20 @@
|
|||
- if @page_nav_mode != :comment || current_page.nil?
|
||||
- if report.pages.any?
|
||||
nav.mt-3 id=dom_id(report, :page_nav_spy)
|
||||
ul
|
||||
ul data={ controller: :sortable }
|
||||
- report.pages.each do |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=""
|
||||
summary class=(is_current ? "active" : nil)
|
||||
.content
|
||||
i.bi.me-1 class="bi-file-earmark-check#{is_current ? "" : "" }"
|
||||
- if is_current
|
||||
=< "#{page.position} #{page.path}"
|
||||
span id=dom_id(page, :title) =< "#{page.position} #{page.path}"
|
||||
- else
|
||||
=< link_to("#{page.position} #{page.path}", report_path(report, page_id: page.id), data: { "turbo-frame": :_top })
|
||||
ul id=dom_id(page, :page_nav_elements)
|
||||
=< link_to(report_path(report, page_id: page.id), data: { "turbo-frame": :_top }) do
|
||||
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|
|
||||
= render partial: "elements/page_nav_row", locals: { element: element, current_page: current_page == element.page }
|
||||
/li
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
/ = turbo_frame_tag(dom_id(success_criterion, :frame)) do
|
||||
- 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/body", locals: {success_criterion: success_criterion }
|
||||
|
|
@ -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, :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)
|
||||
7
db/migrate/20241111011637_remove_position_indices.rb
Normal file
7
db/migrate/20241111011637_remove_position_indices.rb
Normal 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
5
db/schema.rb
generated
|
|
@ -10,7 +10,7 @@
|
|||
#
|
||||
# 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|
|
||||
t.string "name", null: false
|
||||
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.integer "page_id", 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"
|
||||
end
|
||||
|
||||
|
|
@ -145,7 +144,6 @@ ActiveRecord::Schema[8.0].define(version: 2024_11_08_180654) do
|
|||
t.datetime "created_at", null: false
|
||||
t.datetime "updated_at", null: false
|
||||
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"
|
||||
end
|
||||
|
||||
|
|
@ -192,7 +190,6 @@ ActiveRecord::Schema[8.0].define(version: 2024_11_08_180654) do
|
|||
t.integer "priority"
|
||||
t.integer "position", null: false
|
||||
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"
|
||||
end
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue