scinote-web/app/controllers/canvas_controller.rb
2016-08-23 11:08:10 +02:00

320 lines
7.9 KiB
Ruby

class CanvasController < ApplicationController
before_action :load_vars
before_action :check_view_canvas, only: [:edit, :full_zoom, :medium_zoom, :small_zoom]
before_action :check_edit_canvas, only: [:edit, :update]
def edit
render partial: 'canvas/edit',
locals: { experiment: @experiment, my_modules: @my_modules },
:content_type => 'text/html'
end
def full_zoom
render partial: 'canvas/full_zoom',
locals: { experiment: @experiment, my_modules: @my_modules },
:content_type => 'text/html'
end
def medium_zoom
render partial: 'canvas/medium_zoom',
locals: { experiment: @experiment, my_modules: @my_modules },
:content_type => 'text/html'
end
def small_zoom
render partial: 'canvas/small_zoom',
locals: { experiment: @experiment, my_modules: @my_modules },
:content_type => 'text/html'
end
def update
error = false
# Make sure that remove parameter is valid
to_archive = []
if can_archive_modules(@experiment) and
update_params[:remove].present? then
to_archive = update_params[:remove].split(",")
unless to_archive.all? { |id| is_int? id }
error = true
else
to_archive.collect! { |id| id.to_i }
end
end
if error then
render_403 and return
end
# Make sure connections parameter is valid
connections = []
if can_edit_connections(@experiment) and
update_params[:connections].present? then
conns = update_params[:connections].split(",")
unless conns.length % 2 == 0 and
conns.all? { |c| c.is_a? String } then
error = true
else
conns.each_slice(2).each do |c|
connections << [c[0], c[1]]
end
end
end
if error then
render_403 and return
end
# Make sure positions parameter is valid
positions = Hash.new
if can_reposition_modules(@experiment) and
update_params[:positions].present? then
poss = update_params[:positions].split(";")
center = ""
(poss.collect { |pos| pos.split(",") }).each_with_index do |pos, index|
unless (pos.length == 3 and
pos[0].is_a? String and
is_int? pos[1] and
is_int? pos[2])
error = true
break
end
if index == 0
center = pos
x = 0
y = 0
else
x = pos[1].to_i - center[1].to_i
y = pos[2].to_i - center[2].to_i
end
# Multiple modules cannot have same position
if positions.any? { |k,v| v[:x] == x and v[:y] == y} then
error = true
break
end
positions[pos[0]] = { x: x, y: y }
end
end
if error then
render_403 and return
end
# Make sure that to_add is an array of strings,
# as well as that positions for newly added modules exist
to_add = []
if can_create_modules(@experiment) and
update_params[:add].present? and
update_params["add-names"].present? then
ids = update_params[:add].split(",")
names = update_params["add-names"].split("|")
unless ids.length == names.length and
ids.all? { |id| id.is_a? String and positions.include? id } and
names.all? { |name| name.is_a? String }
error = true
else
ids.each_with_index do |id, i|
to_add << {
id: id,
name: names[i],
x: positions[id][:x],
y: positions[id][:y]
}
end
end
end
if error then
render_403 and return
end
# Make sure rename parameter is valid
to_rename = Hash.new
if can_edit_modules(@experiment) and
update_params[:rename].present? then
begin
to_rename = JSON.parse(update_params[:rename])
# Okay, JSON parsed!
unless (
to_rename.is_a? Hash and
to_rename.keys.all? { |k| k.is_a? String } and
to_rename.values.all? { |k| k.is_a? String }
)
error = true
end
rescue
error = true
end
end
if error then
render_403 and return
end
# Make sure move parameter is valid
to_move = {}
if can_move_modules(@experiment) && update_params[:move].present?
begin
to_move = JSON.parse(update_params[:move])
# Okay, JSON parsed!
unless (
to_move.is_a? Hash and
to_move.keys.all? { |k| k.is_a? String } &&
to_move.values.all? { |k| k.is_a? String }
)
error = true
end
rescue
error = true
end
end
# Distinguish between moving modules/module_groups
to_move_groups = {}
to_move.each do |key, value|
if key =~ /.*,.*/
to_move_groups[key.split(',')] = value
to_move.delete(key)
end
end
render_403 and return if error
# Make sure that to_clone is an array of pairs,
# as well as that all IDs exist
to_clone = Hash.new
if can_clone_modules(@experiment) and
update_params[:cloned].present? then
clones = update_params[:cloned].split(";")
(clones.collect { |v| v.split(",") }).each do |val|
unless (val.length == 2 and
is_int? val[0] and
val[1].is_a? String and
to_add.any? { |m| m[:id] == val[1] })
error = true
break
else
to_clone[val[1]] = val[0]
end
end
end
if error then
render_403 and return
end
module_groups = Hash.new
if can_edit_module_groups(@experiment) and
update_params["module-groups"].present? then
begin
module_groups = JSON.parse(update_params["module-groups"])
# Okay, JSON parsed!
unless (
module_groups.is_a? Hash and
module_groups.keys.all? { |k| k.is_a? String } and
module_groups.values.all? { |k| k.is_a? String }
)
error = true
end
rescue
error = true
end
end
if error then
render_403 and return
end
# Call the "master" function to do all the updating for us
unless @experiment.update_canvas(
to_archive,
to_add,
to_rename,
to_move,
to_move_groups,
to_clone,
connections,
positions,
current_user,
module_groups
)
render_403 and return
end
#Save activities that modules were archived
to_archive.each do |module_id|
my_module = MyModule.find_by_id(module_id)
unless my_module.blank?
Activity.create(
type_of: :archive_module,
project: my_module.experiment.project,
my_module: my_module,
user: current_user,
message: t(
'activities.archive_module',
user: current_user.full_name,
module: my_module.name
)
)
end
end
# Create workflow image
@experiment.delay.generate_workflow_img
flash[:success] = t(
"experiments.canvas.update.success_flash")
redirect_to canvas_experiment_path(@experiment)
end
private
def update_params
params.permit(
:id,
:connections,
:positions,
:add,
"add-names",
:rename,
:move,
:cloned,
:remove,
"module-groups"
)
end
def load_vars
@experiment = Experiment.find_by_id(params[:id])
unless @experiment
respond_to do |format|
format.html { render_404 and return }
format.any(:xml, :json, :js) { render(json: { redirect_url: not_found_url }, status: :not_found) and return }
end
end
@my_modules = @experiment.active_modules
end
def check_edit_canvas
unless can_edit_canvas(@experiment)
render_403 and return
end
end
def check_view_canvas
unless can_view_experiment(@experiment)
render_403 and return
end
end
# Check if given value is "integer" string (e.g. "15")
def is_int?(val)
/\A[-+]?\d+\z/ === val
end
end