mirror of
https://github.com/scinote-eln/scinote-web.git
synced 2025-01-06 07:33:04 +08:00
31f8bc19d5
* Add unique database index to checklist items position and update errors to render flash message when a checklist item validation fails [SCI-8841] * Improve protocol checklist item reordering [SCI-8841] * Add act as list to gemfile and use sequential updates for reodering with act_as_list [SCI-8841]
165 lines
5.6 KiB
Ruby
165 lines
5.6 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module StepElements
|
|
class ChecklistItemsController < ApplicationController
|
|
include ApplicationHelper
|
|
include StepsActions
|
|
|
|
before_action :load_vars
|
|
before_action :load_checklist_item, only: %i(update toggle destroy)
|
|
before_action :check_toggle_permissions, only: %i(toggle)
|
|
before_action :check_manage_permissions, only: %i(create update destroy)
|
|
|
|
def create
|
|
checklist_item = @checklist.checklist_items.build(checklist_item_params.merge!(created_by: current_user))
|
|
|
|
ActiveRecord::Base.transaction do
|
|
checklist_item.save!
|
|
log_activity(
|
|
"#{@step.protocol.in_module? ? :task : :protocol}_step_checklist_item_added",
|
|
{
|
|
checklist_item: checklist_item.text,
|
|
checklist_name: @checklist.name
|
|
}
|
|
)
|
|
checklist_item_annotation(@step, checklist_item)
|
|
end
|
|
|
|
render json: checklist_item, serializer: ChecklistItemSerializer, user: current_user
|
|
rescue ActiveRecord::RecordInvalid
|
|
render json: { errors: checklist_item.errors }, status: :unprocessable_entity
|
|
end
|
|
|
|
def update
|
|
old_text = @checklist_item.text
|
|
@checklist_item.assign_attributes(
|
|
checklist_item_params.except(:position, :id).merge(last_modified_by: current_user)
|
|
)
|
|
|
|
if @checklist_item.save!
|
|
log_activity(
|
|
"#{@step.protocol.in_module? ? :task : :protocol}_step_checklist_item_edited",
|
|
checklist_item: @checklist_item.text,
|
|
checklist_name: @checklist.name
|
|
)
|
|
checklist_item_annotation(@step, @checklist_item, old_text)
|
|
end
|
|
|
|
render json: @checklist_item, serializer: ChecklistItemSerializer, user: current_user
|
|
rescue ActiveRecord::RecordInvalid
|
|
render json: { errors: @checklist_item.errors }, status: :unprocessable_entity
|
|
end
|
|
|
|
def toggle
|
|
@checklist_item.assign_attributes(
|
|
checklist_toggle_item_params.merge(last_modified_by: current_user)
|
|
)
|
|
|
|
if @checklist_item.save!
|
|
completed_items = @checklist_item.checklist.checklist_items.where(checked: true).count
|
|
all_items = @checklist_item.checklist.checklist_items.count
|
|
text_activity = smart_annotation_parser(@checklist_item.text).gsub(/\s+/, ' ')
|
|
type_of = if @checklist_item.saved_change_to_attribute(:checked).last
|
|
:check_step_checklist_item
|
|
else
|
|
:uncheck_step_checklist_item
|
|
end
|
|
log_activity(type_of,
|
|
checkbox: text_activity,
|
|
num_completed: completed_items.to_s,
|
|
num_all: all_items.to_s)
|
|
end
|
|
|
|
render json: @checklist_item, serializer: ChecklistItemSerializer, user: current_user
|
|
rescue ActiveRecord::RecordInvalid
|
|
render json: @checklist_item, serializer: ChecklistItemSerializer,
|
|
user: current_user,
|
|
status: :unprocessable_entity
|
|
end
|
|
|
|
def destroy
|
|
if @checklist_item.destroy
|
|
log_activity(
|
|
"#{@step.protocol.in_module? ? :task : :protocol}_step_checklist_item_deleted",
|
|
checklist_item: @checklist_item.text,
|
|
checklist_name: @checklist.name
|
|
)
|
|
render json: @checklist_item, serializer: ChecklistItemSerializer, user: current_user
|
|
else
|
|
render json: @checklist_item, serializer: ChecklistItemSerializer,
|
|
user: current_user,
|
|
status: :unprocessable_entity
|
|
end
|
|
end
|
|
|
|
def reorder
|
|
checklist_item = @checklist.checklist_items.find(checklist_item_params[:id])
|
|
ActiveRecord::Base.transaction do
|
|
checklist_item.insert_at(checklist_item_params[:position])
|
|
@checklist.touch
|
|
end
|
|
render json: params[:checklist_item_positions], status: :ok
|
|
rescue ActiveRecord::RecordInvalid
|
|
render json: { errors: checklist_item.errors }, status: :conflict
|
|
end
|
|
|
|
private
|
|
|
|
def check_toggle_permissions
|
|
if ActiveModel::Type::Boolean.new.cast(checklist_toggle_item_params[:checked])
|
|
render_403 unless can_check_my_module_steps?(@step.protocol.my_module)
|
|
else
|
|
render_403 unless can_uncheck_my_module_steps?(@step.protocol.my_module)
|
|
end
|
|
end
|
|
|
|
def check_manage_permissions
|
|
render_403 unless can_manage_step?(@step)
|
|
end
|
|
|
|
def checklist_item_params
|
|
params.require(:attributes).permit(:text, :position, :id)
|
|
end
|
|
|
|
def checklist_toggle_item_params
|
|
params.require(:attributes).permit(:checked)
|
|
end
|
|
|
|
def load_vars
|
|
@step = Step.find_by(id: params[:step_id])
|
|
return render_404 unless @step
|
|
|
|
@protocol = @step.protocol
|
|
|
|
@checklist = @step.checklists.find_by(id: params[:checklist_id])
|
|
return render_404 unless @checklist
|
|
end
|
|
|
|
def load_checklist_item
|
|
@checklist_item = @checklist.checklist_items.find_by(id: params[:id])
|
|
return render_404 unless @checklist_item
|
|
end
|
|
|
|
def log_activity(type_of, message_items = {})
|
|
Activities::CreateActivityService.call(
|
|
activity_type: type_of,
|
|
owner: current_user,
|
|
subject: @step.protocol,
|
|
team: @step.protocol.team,
|
|
project: @step.protocol.in_module? ? @step.protocol.my_module.project : nil,
|
|
message_items: message_items.merge(step_message_items)
|
|
)
|
|
end
|
|
|
|
def step_message_items
|
|
items = {
|
|
step: @step.id,
|
|
step_position: { id: @step.id, value_for: 'position_plus_one' }
|
|
}
|
|
|
|
items[:my_module] = @step.protocol.my_module.id if @step.protocol.in_module?
|
|
|
|
items
|
|
end
|
|
end
|
|
end
|