mirror of
https://github.com/scinote-eln/scinote-web.git
synced 2025-01-15 20:11:22 +08:00
658 lines
22 KiB
Ruby
658 lines
22 KiB
Ruby
class MyModulesController < ApplicationController
|
|
include TeamsHelper
|
|
include ActionView::Helpers::TextHelper
|
|
include InputSanitizeHelper
|
|
include Rails.application.routes.url_helpers
|
|
include ActionView::Helpers::UrlHelper
|
|
include ApplicationHelper
|
|
include MyModulesHelper
|
|
include Breadcrumbs
|
|
|
|
before_action :load_vars, except: %i(index restore_group create new save_table_state
|
|
inventory_assigning_my_module_filter actions_toolbar)
|
|
before_action :load_experiment, only: %i(create new index)
|
|
before_action :check_create_permissions, only: %i(new create)
|
|
before_action :check_archive_permissions, only: %i(update)
|
|
before_action :check_manage_permissions, only: %i(
|
|
description due_date update_description update_protocol_description update_protocol
|
|
)
|
|
before_action :check_read_permissions, except: %i(create new update update_description
|
|
inventory_assigning_my_module_filter
|
|
update_protocol_description restore_group
|
|
save_table_state actions_toolbar index)
|
|
before_action :check_update_state_permissions, only: :update_state
|
|
before_action :set_inline_name_editing, only: %i(protocols activities archive index)
|
|
before_action :load_experiment_my_modules, only: %i(protocols activities archive)
|
|
before_action :set_breadcrumbs_items, only: %i(protocols activities archive index)
|
|
before_action :set_navigator, only: %i(protocols activities archive index)
|
|
|
|
layout 'fluid'.freeze
|
|
|
|
def index
|
|
respond_to do |format|
|
|
format.json do
|
|
my_modules = Lists::MyModulesService.new(@experiment.my_modules.readable_by_user(current_user),
|
|
params.merge(experiment: @experiment),
|
|
user: current_user).call
|
|
render json: my_modules, each_serializer: Lists::MyModuleSerializer, user: current_user,
|
|
meta: pagination_dict(my_modules)
|
|
end
|
|
format.html do
|
|
save_view_type('table')
|
|
render 'my_modules/index'
|
|
end
|
|
end
|
|
end
|
|
|
|
def new
|
|
@my_module = @experiment.my_modules.new
|
|
assigned_users = User.where(id: @experiment.user_assignments.select(:user_id))
|
|
|
|
render json: {
|
|
html: render_to_string(
|
|
partial: 'my_modules/modals/new_modal', locals: { view_mode: params[:view_mode],
|
|
users: assigned_users }
|
|
)
|
|
}
|
|
end
|
|
|
|
def create
|
|
@my_module = @experiment.my_modules.new(my_module_params)
|
|
new_pos = @my_module.get_new_position
|
|
@my_module.assign_attributes(
|
|
created_by: current_user,
|
|
last_modified_by: current_user,
|
|
x: new_pos[:x],
|
|
y: new_pos[:y]
|
|
)
|
|
@my_module.transaction do
|
|
if my_module_tags_params[:tag_ids].present?
|
|
@my_module.tags << @experiment.project.tags.where(id: my_module_tags_params[:tag_ids])
|
|
end
|
|
if my_module_designated_users_params[:user_ids].present? && can_designate_users_to_new_task?(@experiment)
|
|
@my_module.designated_users << @experiment.users.where(id: my_module_designated_users_params[:user_ids])
|
|
elsif !can_designate_users_to_new_task?(@experiment)
|
|
@my_module.designated_users << current_user
|
|
end
|
|
@my_module.save!
|
|
Activities::CreateActivityService.call(
|
|
activity_type: :create_module,
|
|
owner: current_user,
|
|
team: @my_module.experiment.project.team,
|
|
project: @my_module.experiment.project,
|
|
subject: @my_module,
|
|
message_items: { my_module: @my_module.id }
|
|
)
|
|
log_user_designation_activity
|
|
redirect_to canvas_experiment_path(@experiment) if params[:my_module][:view_mode] == 'canvas'
|
|
rescue ActiveRecord::RecordInvalid
|
|
render json: @my_module.errors, status: :unprocessable_entity
|
|
raise ActiveRecord::Rollback
|
|
end
|
|
end
|
|
|
|
def show
|
|
respond_to do |format|
|
|
format.html do
|
|
render json: {
|
|
html: render_to_string(partial: 'show')
|
|
}
|
|
end
|
|
format.json do
|
|
render json: @my_module, serializer: Lists::MyModuleSerializer, controller: self, user: current_user
|
|
end
|
|
end
|
|
end
|
|
|
|
# Description modal window in full-zoom canvas
|
|
def description
|
|
render json: {
|
|
html: render_to_string(partial: 'description'),
|
|
title: t('my_modules.description.title', module: escape_input(@my_module.name))
|
|
}
|
|
end
|
|
|
|
def save_table_state
|
|
current_user.settings.update(visible_my_module_table_columns: params[:columns])
|
|
current_user.save!
|
|
end
|
|
|
|
def status_state
|
|
if @my_module.last_transition_error && @my_module.last_transition_error['type'] == 'repository_snapshot'
|
|
flash[:repository_snapshot_error] = @my_module.last_transition_error
|
|
end
|
|
|
|
render json: { status_changing: @my_module.status_changing? }
|
|
end
|
|
|
|
def canvas_dropdown_menu
|
|
@experiment_managable = can_manage_experiment?(@experiment)
|
|
@group_my_modules = @my_module.my_module_group&.my_modules&.preload(user_assignments: %i(user user_role))
|
|
render json: {
|
|
html: render_to_string({ partial: 'canvas/edit/my_module_dropdown_menu',
|
|
locals: { my_module: @my_module } })
|
|
}
|
|
end
|
|
|
|
def activities
|
|
params[:subjects] = {
|
|
MyModule: [@my_module.id]
|
|
}
|
|
@activity_types = Activity.activity_types_list
|
|
|
|
activities = ActivitiesService.load_activities(current_user, current_team, activity_filters)
|
|
|
|
@grouped_activities = activities.group_by do |activity|
|
|
Time.zone.at(activity.created_at).to_date.to_s
|
|
end
|
|
|
|
@next_page = activities.next_page
|
|
|
|
respond_to do |format|
|
|
format.json do
|
|
render json: {
|
|
activities_html: render_to_string(
|
|
partial: 'global_activities/activity_list',
|
|
formats: :html
|
|
),
|
|
next_page: @next_page
|
|
}
|
|
end
|
|
format.html
|
|
end
|
|
end
|
|
|
|
# Different controller for showing activities inside tab
|
|
def activities_tab
|
|
@activities = @my_module.last_activities(1, @per_page)
|
|
|
|
render json: {
|
|
html: render_to_string(partial: 'activities', formats: :html)
|
|
}
|
|
end
|
|
|
|
# Due date modal window in full-zoom canvas
|
|
def due_date
|
|
render json: {
|
|
html: render_to_string(partial: 'due_date', formats: :html),
|
|
title: t('.due_date.title', module: escape_input(@my_module.name))
|
|
}
|
|
end
|
|
|
|
def update
|
|
@my_module.assign_attributes(my_module_params)
|
|
@my_module.last_modified_by = current_user
|
|
name_changed = @my_module.name_changed?
|
|
description_changed = @my_module.description_changed?
|
|
start_date_changes = @my_module.changes[:started_on]
|
|
due_date_changes = @my_module.changes[:due_date]
|
|
|
|
render_403 && return if @my_module.completed_on_changed? && !can_complete_my_module?(@my_module)
|
|
|
|
render_403 && return if description_changed && !can_update_my_module_description?(@my_module)
|
|
|
|
render_403 && return if start_date_changes.present? && !can_update_my_module_start_date?(@my_module)
|
|
|
|
render_403 && return if due_date_changes.present? && !can_update_my_module_start_date?(@my_module)
|
|
|
|
if @my_module.archived_changed?(from: false, to: true)
|
|
saved = @my_module.archive(current_user)
|
|
else
|
|
render_403 && return unless can_manage_my_module?(@my_module)
|
|
|
|
saved = @my_module.save
|
|
if saved
|
|
if description_changed
|
|
log_activity(:change_module_description)
|
|
TinyMceAsset.update_images(@my_module, params[:tiny_mce_images], current_user)
|
|
end
|
|
|
|
log_activity(:rename_task) if name_changed
|
|
log_start_date_change_activity(start_date_changes) if start_date_changes.present?
|
|
log_due_date_change_activity(due_date_changes) if due_date_changes.present?
|
|
end
|
|
end
|
|
if saved
|
|
alerts = []
|
|
alerts << 'alert-green' if @my_module.completed?
|
|
unless @my_module.completed?
|
|
alerts << 'alert-red' if @my_module.is_overdue?
|
|
alerts << 'alert-yellow' if @my_module.is_one_day_prior?
|
|
end
|
|
render json: {
|
|
status: :ok,
|
|
start_date_label: render_to_string(
|
|
partial: 'my_modules/start_date_label',
|
|
formats: :html,
|
|
locals: { my_module: @my_module, start_date_editable: true }
|
|
),
|
|
due_date_label: render_to_string(
|
|
partial: 'my_modules/due_date_label',
|
|
formats: :html,
|
|
locals: { my_module: @my_module, due_date_editable: true }
|
|
),
|
|
card_due_date_label: render_to_string(
|
|
partial: 'my_modules/card_due_date_label',
|
|
formats: :html,
|
|
locals: { my_module: @my_module }
|
|
),
|
|
table_due_date_label: {
|
|
html: render_to_string(partial: 'experiments/table_due_date_label',
|
|
formats: :html,
|
|
locals: { my_module: @my_module, user: current_user }),
|
|
due_status: my_module_due_status(@my_module)
|
|
},
|
|
module_header_due_date: render_to_string(
|
|
partial: 'my_modules/module_header_due_date',
|
|
formats: :html,
|
|
locals: { my_module: @my_module }
|
|
),
|
|
description_label: render_to_string(
|
|
partial: 'my_modules/description_label',
|
|
formats: :html,
|
|
locals: { my_module: @my_module }
|
|
),
|
|
alerts: alerts
|
|
}
|
|
else
|
|
render json: @my_module.errors, status: :unprocessable_entity
|
|
end
|
|
end
|
|
|
|
def update_description
|
|
render_403 && return unless can_update_my_module_description?(@my_module)
|
|
|
|
old_description = @my_module.description
|
|
if @my_module.update(description: params.require(:my_module)[:description])
|
|
log_activity(:change_module_description)
|
|
TinyMceAsset.update_images(@my_module, params[:tiny_mce_images], current_user)
|
|
my_module_annotation_notification(old_description)
|
|
render json: {
|
|
html: custom_auto_link(
|
|
@my_module.tinymce_render(:description),
|
|
simple_format: false,
|
|
tags: %w(img),
|
|
team: current_team
|
|
)
|
|
}
|
|
else
|
|
render json: @my_module.errors, status: :unprocessable_entity
|
|
end
|
|
end
|
|
|
|
def update_protocol_description
|
|
protocol = @my_module.protocol
|
|
old_description = protocol.description
|
|
return render_404 unless protocol
|
|
|
|
if protocol.update(description: params.require(:protocol)[:description])
|
|
log_activity(:protocol_description_in_task_edited)
|
|
TinyMceAsset.update_images(protocol, params[:tiny_mce_images], current_user)
|
|
protocol_annotation_notification(old_description)
|
|
render json: {
|
|
html: custom_auto_link(
|
|
protocol.tinymce_render(:description),
|
|
simple_format: false,
|
|
tags: %w(img),
|
|
team: current_team
|
|
)
|
|
}
|
|
else
|
|
render json: protocol.errors, status: :unprocessable_entity
|
|
end
|
|
end
|
|
|
|
def protocols
|
|
@protocol = @my_module.protocol
|
|
@assigned_repositories = @my_module.readable_live_and_snapshot_repositories_list(current_user)
|
|
end
|
|
|
|
def protocol
|
|
render json: @my_module.protocol, serializer: ProtocolSerializer, user: current_user
|
|
end
|
|
|
|
def update_protocol
|
|
protocol = @my_module.protocol
|
|
old_description = protocol.description
|
|
|
|
ActiveRecord::Base.transaction do
|
|
protocol.update!(protocol_params)
|
|
log_activity(:protocol_name_in_task_edited) if protocol.saved_change_to_name?
|
|
log_activity(:protocol_description_in_task_edited) if protocol.saved_change_to_description?
|
|
TinyMceAsset.update_images(protocol, params[:tiny_mce_images], current_user)
|
|
protocol_annotation_notification(old_description)
|
|
end
|
|
|
|
render json: protocol, serializer: ProtocolSerializer, user: current_user
|
|
rescue ActiveRecord::RecordInvalid
|
|
render json: protocol.errors, status: :unprocessable_entity
|
|
end
|
|
|
|
def archive
|
|
@archived_results = @my_module.archived_results
|
|
end
|
|
|
|
def restore_group
|
|
experiment = Experiment.find(params[:id])
|
|
return render_403 unless can_read_experiment?(experiment)
|
|
|
|
my_modules = experiment.my_modules.archived.where(id: params[:my_modules_ids])
|
|
counter = 0
|
|
my_modules.each do |my_module|
|
|
next unless can_restore_my_module?(my_module)
|
|
|
|
my_module.transaction do
|
|
my_module.restore!(current_user)
|
|
log_activity(:restore_module, my_module)
|
|
counter += 1
|
|
rescue StandardError => e
|
|
Rails.logger.error e.message
|
|
raise ActiveRecord::Rollback
|
|
end
|
|
end
|
|
if counter == my_modules.size
|
|
message = t('my_modules.restore_group.success_flash_html', number: counter)
|
|
elsif counter.positive?
|
|
message = t('my_modules.restore_group.partial_success_flash_html', number: counter)
|
|
else
|
|
error = t('my_modules.restore_group.error_flash')
|
|
end
|
|
|
|
if params[:view] == 'table'
|
|
if message
|
|
render json: { message: message }
|
|
else
|
|
render json: { error: error }, status: :unprocessable_entity
|
|
end
|
|
else
|
|
flash[:notice] = message if message
|
|
flash[:error] = error if error
|
|
redirect_to module_archive_experiment_path(experiment)
|
|
end
|
|
end
|
|
|
|
def update_state
|
|
old_status_id = @my_module.my_module_status_id
|
|
if @my_module.update(my_module_status_id: update_status_params[:status_id])
|
|
log_activity(:change_status_on_task_flow, @my_module, my_module_status_old: old_status_id,
|
|
my_module_status_new: @my_module.my_module_status.id)
|
|
|
|
render json: { status: :changed }
|
|
else
|
|
render json: { errors: @my_module.errors.messages.values.flatten.join('\n') }, status: :unprocessable_entity
|
|
end
|
|
end
|
|
|
|
def permissions
|
|
if stale?(@my_module)
|
|
render json: {
|
|
editable: can_manage_my_module?(@my_module),
|
|
moveable: can_move_my_module?(@my_module),
|
|
archivable: can_archive_my_module?(@my_module),
|
|
restorable: can_restore_my_module?(@my_module)
|
|
}
|
|
end
|
|
end
|
|
|
|
def actions_dropdown
|
|
if stale?(@my_module)
|
|
render json: {
|
|
html: render_to_string(
|
|
partial: 'experiments/table_row_actions',
|
|
locals: { my_module: @my_module }
|
|
)
|
|
}
|
|
end
|
|
end
|
|
|
|
def actions_toolbar
|
|
render json: {
|
|
actions:
|
|
Toolbars::MyModulesService.new(
|
|
current_user,
|
|
my_module_ids: params[:items].present? ? JSON.parse(params[:items]).map { |i| i['id'] } : params[:items]
|
|
).actions
|
|
}
|
|
end
|
|
|
|
def provisioning_status
|
|
render json: { provisioning_status: @my_module.provisioning_status }
|
|
end
|
|
|
|
def inventory_assigning_my_module_filter
|
|
viewable_experiments = Experiment.viewable_by_user(current_user, current_team)
|
|
assignable_my_modules = MyModule.repository_row_assignable_by_user(current_user)
|
|
|
|
experiment = Experiment.viewable_by_user(current_user, current_team)
|
|
.joins(:my_modules)
|
|
.where(experiments: { id: viewable_experiments })
|
|
.where(my_modules: { id: assignable_my_modules })
|
|
.find_by(id: params[:experiment_id])
|
|
|
|
return render_404 if experiment.blank?
|
|
|
|
my_modules = experiment.my_modules
|
|
.where(my_modules: { id: assignable_my_modules })
|
|
.order(:name)
|
|
.pluck(:id, :name)
|
|
|
|
return render plain: [].to_json if my_modules.blank?
|
|
|
|
render json: my_modules
|
|
end
|
|
|
|
private
|
|
|
|
def load_vars
|
|
@my_module = MyModule.preload(user_assignments: %i(user user_role)).find_by(id: params[:id])
|
|
if @my_module
|
|
@experiment = @my_module.experiment
|
|
@project = @my_module.experiment.project if @experiment
|
|
else
|
|
render_404
|
|
end
|
|
end
|
|
|
|
def load_experiment
|
|
@experiment = Experiment.preload(user_assignments: %i(user user_role))
|
|
.find_by(id: params[:id] || params[:experiment_id])
|
|
|
|
render_404 unless @experiment
|
|
render_403 unless can_read_experiment?(@experiment)
|
|
end
|
|
|
|
def load_experiment_my_modules
|
|
@experiment_my_modules = if @my_module.experiment.archived_branch?
|
|
@my_module.experiment.my_modules.order(:name)
|
|
else
|
|
@my_module.experiment.my_modules.where(archived: @my_module.archived?).order(:name)
|
|
end
|
|
end
|
|
|
|
def check_create_permissions
|
|
render_403 && return unless can_create_experiment_tasks?(@experiment)
|
|
end
|
|
|
|
def check_manage_permissions
|
|
render_403 && return unless can_manage_my_module?(@my_module)
|
|
end
|
|
|
|
def check_archive_permissions
|
|
return render_403 if my_module_params[:archived] == 'true' && !can_archive_my_module?(@my_module)
|
|
end
|
|
|
|
def check_read_permissions
|
|
current_team_switch(@project.team) if current_team != @project.team
|
|
render_403 unless can_read_my_module?(@my_module)
|
|
end
|
|
|
|
def check_update_state_permissions
|
|
return render_403 unless can_update_my_module_status?(@my_module)
|
|
|
|
render_404 unless @my_module.my_module_status
|
|
end
|
|
|
|
def set_inline_name_editing
|
|
if action_name == 'index'
|
|
return unless can_manage_experiment?(@experiment)
|
|
|
|
@inline_editable_title_config = {
|
|
name: 'title',
|
|
params_group: 'experiment',
|
|
item_id: @experiment.id,
|
|
field_to_udpate: 'name',
|
|
path_to_update: experiment_path(@experiment)
|
|
}
|
|
return
|
|
end
|
|
|
|
return unless can_manage_my_module?(@my_module)
|
|
|
|
@inline_editable_title_config = {
|
|
name: 'title',
|
|
params_group: 'my_module',
|
|
item_id: @my_module.id,
|
|
field_to_udpate: 'name',
|
|
path_to_update: my_module_path(@my_module)
|
|
}
|
|
end
|
|
|
|
def my_module_params
|
|
permitted_params = params.require(:my_module).permit(:name, :description, :started_on, :due_date, :archived)
|
|
|
|
if permitted_params[:started_on].present?
|
|
permitted_params[:started_on] =
|
|
Time.zone.strptime(permitted_params[:started_on], '%Y/%m/%d %H:%M')
|
|
end
|
|
if permitted_params[:due_date].present?
|
|
permitted_params[:due_date] =
|
|
Time.zone.strptime(permitted_params[:due_date], '%Y/%m/%d %H:%M')
|
|
end
|
|
|
|
permitted_params
|
|
end
|
|
|
|
def my_module_tags_params
|
|
params.require(:my_module).permit(tag_ids: [])
|
|
end
|
|
|
|
def my_module_designated_users_params
|
|
params.require(:my_module).permit(user_ids: [])
|
|
end
|
|
|
|
def protocol_params
|
|
params.require(:protocol).permit(:name, :description)
|
|
end
|
|
|
|
def update_status_params
|
|
params.require(:my_module).permit(:status_id)
|
|
end
|
|
|
|
def log_start_date_change_activity(start_date_changes)
|
|
type_of = if start_date_changes[0].nil? # set started_on
|
|
message_items = { my_module_started_on: @my_module.started_on }
|
|
:set_task_start_date
|
|
elsif start_date_changes[1].nil? # remove started_on
|
|
message_items = { my_module_started_on: start_date_changes[0] }
|
|
:remove_task_start_date
|
|
else # change started_on
|
|
message_items = { my_module_started_on: @my_module.started_on }
|
|
:change_task_start_date
|
|
end
|
|
log_activity(type_of, @my_module, message_items)
|
|
end
|
|
|
|
def log_due_date_change_activity(due_date_changes)
|
|
type_of = if due_date_changes[0].nil? # set due_date
|
|
message_items = { my_module_duedate: @my_module.due_date }
|
|
:set_task_due_date
|
|
elsif due_date_changes[1].nil? # remove due_date
|
|
message_items = { my_module_duedate: due_date_changes[0] }
|
|
:remove_task_due_date
|
|
else # change due_date
|
|
message_items = { my_module_duedate: @my_module.due_date }
|
|
:change_task_due_date
|
|
end
|
|
log_activity(type_of, @my_module, message_items)
|
|
end
|
|
|
|
def log_user_designation_activity
|
|
users = User.where.not(id: current_user.id).where(id: params[:my_module][:user_ids])
|
|
|
|
users.each do |user|
|
|
log_activity(:designate_user_to_my_module, @my_module, { user_target: user.id })
|
|
end
|
|
end
|
|
|
|
def save_view_type(view_type)
|
|
view_state = @experiment.current_view_state(current_user)
|
|
view_state.state['my_modules']['view_type'] = view_type
|
|
view_state.save!
|
|
end
|
|
|
|
def log_activity(type_of, my_module = nil, message_items = {})
|
|
my_module ||= @my_module
|
|
message_items = { my_module: my_module.id }.merge(message_items)
|
|
|
|
Activities::CreateActivityService
|
|
.call(activity_type: type_of,
|
|
owner: current_user,
|
|
team: my_module.team,
|
|
project: my_module.project,
|
|
subject: my_module,
|
|
message_items: message_items)
|
|
end
|
|
|
|
def activity_filters
|
|
params.permit(
|
|
:page, :from_date, :to_date, types: [], users: [], subjects: {}
|
|
)
|
|
end
|
|
|
|
def my_module_annotation_notification(old_text = nil)
|
|
smart_annotation_notification(
|
|
old_text: old_text,
|
|
new_text: @my_module.description,
|
|
subject: @my_module,
|
|
title: t('notifications.my_module_description_annotation_title',
|
|
my_module: @my_module.name,
|
|
user: current_user.full_name),
|
|
message: t('notifications.my_module_description_annotation_message_html',
|
|
project: link_to(@my_module.experiment.project.name, project_url(@my_module.experiment.project)),
|
|
experiment: link_to(@my_module.experiment.name, my_modules_experiment_url(@my_module.experiment)),
|
|
my_module: link_to(@my_module.name, protocols_my_module_url(@my_module)))
|
|
)
|
|
end
|
|
|
|
def protocol_annotation_notification(old_text = nil)
|
|
smart_annotation_notification(
|
|
old_text: old_text,
|
|
new_text: @my_module.protocol.description,
|
|
subject: @my_module,
|
|
title: t('notifications.my_module_protocol_annotation_title',
|
|
my_module: @my_module.name,
|
|
user: current_user.full_name),
|
|
message: t('notifications.my_module_protocol_annotation_message_html',
|
|
project: link_to(@my_module.experiment.project.name, project_url(@my_module.experiment.project)),
|
|
experiment: link_to(@my_module.experiment.name, my_modules_experiment_url(@my_module.experiment)),
|
|
my_module: link_to(@my_module.name, protocols_my_module_url(@my_module)))
|
|
)
|
|
end
|
|
|
|
def set_navigator
|
|
if action_name == 'index'
|
|
@navigator = {
|
|
url: tree_navigator_experiment_path(@experiment),
|
|
archived: params[:view_mode] == 'archived',
|
|
id: @experiment.code
|
|
}
|
|
return
|
|
end
|
|
|
|
@navigator = {
|
|
url: tree_navigator_my_module_path(@my_module),
|
|
archived: @my_module.archived_branch? || params[:view_mode] == 'archived',
|
|
id: @my_module.code
|
|
}
|
|
end
|
|
end
|