class StepsController < ApplicationController include ActionView::Helpers::TextHelper include ApplicationHelper before_action :load_vars, only: [:edit, :update, :destroy, :show] before_action :load_vars_nested, only: [:new, :create] before_action :convert_table_contents_to_utf8, only: [:create, :update] before_action :check_view_permissions, only: [:show] before_action :check_create_permissions, only: [:new, :create] before_action :check_edit_permissions, only: [:edit, :update] before_action :check_destroy_permissions, only: [:destroy] before_action :update_checklist_item_positions, only: [:create, :update] def new @step = Step.new respond_to do |format| format.json do render json: { html: render_to_string(partial: 'new.html.erb') } end end end def create @step = Step.new(step_params) @step.completed = false @step.position = @protocol.number_of_steps @step.protocol = @protocol @step.user = current_user @step.last_modified_by = current_user # Update default checked state @step.checklists.each do |checklist| checklist.checklist_items.each do |checklist_item| checklist_item.checked = false end end respond_to do |format| if @step.save # Post process all assets @step.assets.each do |asset| asset.post_process_file(@protocol.team) end # Generate activity if @protocol.in_module? Activity.create( type_of: :create_step, user: current_user, project: @my_module.experiment.project, my_module: @my_module, message: t( "activities.create_step", user: current_user.full_name, step: @step.position + 1, step_name: @step.name ) ) else # TODO: Activity for team if step # created in protocol management?? end # Update protocol timestamp update_protocol_ts(@step) format.json do render json: { html: render_to_string( partial: 'steps/step.html.erb', locals: { step: @step } ) }, status: :ok end else # On error, delete the newly added files from S3, as they were # uploaded on client-side (in case of client-side hacking of # asset's signature response) Asset.destroy_all(new_assets) format.json { render json: { html: render_to_string(partial: 'new.html.erb') }, status: :bad_request } end end end def show respond_to do |format| format.json do render json: { html: render_to_string( partial: 'steps/step.html.erb', locals: { step: @step } ) }, status: :ok end end end def edit respond_to do |format| format.json do render json: { html: render_to_string(partial: 'edit.html.erb') }, status: :ok end end end def update respond_to do |format| previous_size = @step.space_taken step_params_all = step_params # process only destroy update on step references. This prevents # skipping deleting reference in case update validation fails. # NOTE - step_params_all variable is updated destroy_attributes(step_params_all) @step.assign_attributes(step_params_all) @step.last_modified_by = current_user if @step.save @step.reload # Release team's space taken team = @protocol.team team.release_space(previous_size) team.save # Post process step assets @step.assets.each do |asset| asset.post_process_file(team) end # Generate activity if @protocol.in_module? Activity.create( type_of: :edit_step, user: current_user, project: @my_module.experiment.project, my_module: @my_module, message: t( "activities.edit_step", user: current_user.full_name, step: @step.position + 1, step_name: @step.name ) ) else # TODO: Activity for team if step # updated in protocol management?? end # Update protocol timestamp update_protocol_ts(@step) format.json { render json: { html: render_to_string({ partial: 'step.html.erb', locals: { step: @step } }) } } else format.json { render json: @step.errors.to_json, status: :bad_request } end end end def destroy # Update position on other steps of this module @protocol.steps.where("position > ?", @step.position).each do |step| step.position = step.position - 1 step.save end # Calculate space taken by this step team = @protocol.team previous_size = @step.space_taken # Destroy the step @step.destroy(current_user) # Release space taken by the step team.release_space(previous_size) team.save # Update protocol timestamp update_protocol_ts(@step) flash[:success] = t( "protocols.steps.destroy.success_flash", step: (@step.position + 1).to_s) if @protocol.in_module? redirect_to protocols_my_module_path(@step.my_module) else redirect_to edit_protocol_path(@protocol) end end # Responds to checkbox toggling in steps view def checklistitem_state chkItem = ChecklistItem.find_by_id(params["checklistitem_id"]) respond_to do |format| if chkItem checked = params[:checked] == "true" protocol = chkItem.checklist.step.protocol authorized = ((checked and can_check_checkbox(protocol)) or (!checked and can_uncheck_checkbox(protocol))) if authorized changed = chkItem.checked != checked chkItem.checked = checked if chkItem.save format.json { render json: {}, status: :accepted } # Create activity if changed str = checked ? "activities.check_step_checklist_item" : "activities.uncheck_step_checklist_item" completed_items = chkItem.checklist.checklist_items.where(checked: true).count all_items = chkItem.checklist.checklist_items.count message = t( str, user: current_user.full_name, checkbox: smart_annotation_parser(simple_format(chkItem.text)), step: chkItem.checklist.step.position + 1, step_name: chkItem.checklist.step.name, completed: completed_items, all: all_items ) # This should always hold true (only in module can # check items be checked, but still check just in case) if protocol.in_module? Activity.create( user: current_user, project: protocol.my_module.experiment.project, my_module: protocol.my_module, message: message, type_of: checked ? :check_step_checklist_item : :uncheck_step_checklist_item ) end end else format.json { render json: {}, status: :unprocessable_entity } end else format.json { render json: {}, status: :unauthorized } end else format.json { render json: {}, status: :not_found } end end end # Complete/uncomplete step def toggle_step_state step = Step.find_by_id(params[:id]) respond_to do |format| if step completed = params[:completed] == "true" protocol = step.protocol authorized = ((completed and can_complete_step_in_protocol(protocol)) or (!completed and can_uncomplete_step_in_protocol(protocol))) if authorized changed = step.completed != completed step.completed = completed # Update completed_on if changed step.completed_on = completed ? Time.current : nil end if step.save # Create activity if changed completed_steps = protocol.steps.where(completed: true).count all_steps = protocol.steps.count str = completed ? "activities.complete_step" : "activities.uncomplete_step" message = t( str, user: current_user.full_name, step: step.position + 1, step_name: step.name, completed: completed_steps, all: all_steps ) # Toggling step state can only occur in # module protocols, so my_module is always # not nil; nonetheless, check if my_module is present if protocol.in_module? Activity.create( user: current_user, project: protocol.my_module.experiment.project, my_module: protocol.my_module, message: message, type_of: completed ? :complete_step : :uncomplete_step ) end end # Create localized title for complete/uncomplete button localized_title = !completed ? t("protocols.steps.options.complete_title") : t("protocols.steps.options.uncomplete_title") format.json { render json: {new_title: localized_title}, status: :accepted } else format.json { render json: {}, status: :unprocessable_entity } end else format.json { render json: {}, status: :unauthorized } end else format.json { render json: {}, status: :not_found } end end end def move_up step = Step.find_by_id(params[:id]) respond_to do |format| if step if can_reorder_step_in_protocol(step.protocol) if step.position > 0 step_down = step.protocol.steps.where(position: step.position - 1).first step.position -= 1 step.save if step_down step_down.position += 1 step_down.save # Update protocol timestamp update_protocol_ts(step) format.json { render json: { move_direction: "up", step_up_position: step.position, step_down_position: step_down.position }, status: :ok } else format.json { render json: {}, status: :forbidden } end else format.json { render json: {}, status: :forbidden } end else format.json { render json: {}, status: :forbidden } end else format.json { render json: {}, status: :not_found } end end end def move_down step = Step.find_by_id(params[:id]) respond_to do |format| if step if can_reorder_step_in_protocol(step.protocol) if step.position < step.protocol.steps.count - 1 step_up = step.protocol.steps.where(position: step.position + 1).first step.position += 1 step.save if step_up step_up.position -= 1 step_up.save # Update protocol timestamp update_protocol_ts(step) format.json { render json: { move_direction: "down", step_up_position: step_up.position, step_down_position: step.position }, status: :ok } else format.json { render json: {}, status: :forbidden } end else format.json { render json: {}, status: :forbidden } end else format.json { render json: {}, status: :forbidden } end else format.json { render json: {}, status: :not_found } end end end private def update_checklist_item_positions if params["step"].present? && params["step"]["checklists_attributes"].present? params["step"]["checklists_attributes"].values.each do |cla| if cla["checklist_items_attributes"].present? cla["checklist_items_attributes"].each do |idx, item| item["position"] = idx end end end end end # This function is used for partial update of step references and # it's useful when you want to execute destroy action on attribute # collection separately from normal update action, for example if # you don't want that update validation interupt destroy action. # In case of step model you can delete checkboxes, assets or tables. def destroy_attributes(params) update_params = {} delete_step_tables(params) extract_destroy_params(params, update_params) @step.update_attributes(update_params) unless update_params.empty? end # Delete the step table def delete_step_tables(params) return unless params[:tables_attributes].present? params[:tables_attributes].each do |table| next unless table.second['_destroy'] table_to_destroy = Table.find_by(id: table.second['id']) table_to_destroy.report_elements.destroy_all end end # Checks if hash contains destroy parameter '_destroy' and returns # boolean value. def has_destroy_params(params) for key, values in params do if values.respond_to?(:each) for pos, attrs in params[key] do if attrs[:_destroy] == '1' return true end end end end false end # Extracts part of hash that contains destroy parameters. It deletes # values that contains destroy parameters from original variable and # puts them into update_params variable. def extract_destroy_params(params, update_params) for key, values in params do if values.respond_to?(:each) update_params[key] = {} unless update_params[key] attr_params = update_params[key] for pos, attrs in params[key] do if attrs[:_destroy] == '1' if attrs[:id].present? attr_params[pos] = { id: attrs[:id], _destroy: '1' } end params[key].delete(pos) elsif has_destroy_params(params[key][pos]) attr_params[pos] = { id: attrs[:id] } extract_destroy_params(params[key][pos], attr_params[pos]) end end end end end def load_vars @step = Step.find_by_id(params[:id]) @protocol = @step.protocol unless @protocol render_404 end if @protocol.in_module? @my_module = @protocol.my_module end end def load_vars_nested @protocol = Protocol.find_by_id(params[:protocol_id]) unless @protocol render_404 end if @protocol.in_module? @my_module = @protocol.my_module end end def convert_table_contents_to_utf8 if params.include? :step and params[:step].include? :tables_attributes then params[:step][:tables_attributes].each do |k,v| params[:step][:tables_attributes][k][:contents] = v[:contents].encode(Encoding::UTF_8).force_encoding(Encoding::UTF_8) end end end def update_protocol_ts(step) if step.present? && step.protocol.present? step.protocol.update(updated_at: Time.now) end end def check_view_permissions unless can_view_steps_in_protocol(@protocol) render_403 end end def check_create_permissions unless can_create_step_in_protocol(@protocol) render_403 end end def check_edit_permissions unless can_edit_step_in_protocol(@protocol) render_403 end end def check_destroy_permissions unless can_delete_step_in_protocol(@protocol) render_403 end end def step_params params.require(:step).permit( :name, :description, checklists_attributes: [ :id, :name, :_destroy, checklist_items_attributes: [ :id, :text, :position, :_destroy ] ], assets_attributes: [ :id, :file, :_destroy ], tables_attributes: [ :id, :name, :contents, :_destroy ] ) end end