2017-06-30 21:20:27 +08:00
|
|
|
class ProtocolsDatatable < CustomDatatable
|
2016-07-21 19:11:15 +08:00
|
|
|
# Needed for sanitize_sql_like method
|
|
|
|
include ActiveRecord::Sanitization::ClassMethods
|
2017-01-04 22:04:12 +08:00
|
|
|
include InputSanitizeHelper
|
2016-07-21 19:11:15 +08:00
|
|
|
|
2018-02-19 21:47:36 +08:00
|
|
|
def_delegator :@view, :can_read_protocol_in_repository?
|
2018-02-16 01:46:29 +08:00
|
|
|
def_delegator :@view, :can_manage_protocol_in_repository?
|
2016-07-21 19:11:15 +08:00
|
|
|
def_delegator :@view, :edit_protocol_path
|
2018-02-19 21:47:36 +08:00
|
|
|
def_delegator :@view, :can_restore_protocol_in_repository?
|
2018-01-05 22:15:50 +08:00
|
|
|
def_delegator :@view, :can_clone_protocol_in_repository?
|
2016-07-21 19:11:15 +08:00
|
|
|
def_delegator :@view, :clone_protocol_path
|
|
|
|
def_delegator :@view, :linked_children_protocol_path
|
2022-05-27 18:49:10 +08:00
|
|
|
def_delegator :@view, :protocol_path
|
2016-07-21 19:11:15 +08:00
|
|
|
|
2017-01-25 16:48:49 +08:00
|
|
|
def initialize(view, team, type, user)
|
2016-07-21 19:11:15 +08:00
|
|
|
super(view)
|
2017-01-25 16:48:49 +08:00
|
|
|
@team = team
|
2016-07-21 19:11:15 +08:00
|
|
|
# :public, :private or :archive
|
|
|
|
@type = type
|
|
|
|
@user = user
|
|
|
|
end
|
|
|
|
|
|
|
|
def sortable_columns
|
|
|
|
@sortable_columns ||= [
|
|
|
|
"Protocol.name",
|
|
|
|
"protocol_keywords_str",
|
|
|
|
"Protocol.nr_of_linked_children",
|
|
|
|
"full_username_str",
|
|
|
|
timestamp_db_column,
|
|
|
|
"Protocol.updated_at"
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
def searchable_columns
|
|
|
|
@searchable_columns ||= [
|
|
|
|
"Protocol.name",
|
|
|
|
timestamp_db_column,
|
|
|
|
"Protocol.updated_at"
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
# This hack is needed to display a correct amount of
|
|
|
|
# searched entries (needed for pagination).
|
|
|
|
# This is needed because of usage of GROUP operator in SQL.
|
|
|
|
# See https://github.com/antillas21/ajax-datatables-rails/issues/112
|
|
|
|
def as_json(options = {})
|
|
|
|
{
|
2017-06-30 21:20:27 +08:00
|
|
|
draw: dt_params[:draw].to_i,
|
|
|
|
recordsTotal: get_raw_records.length,
|
|
|
|
recordsFiltered: filter_records(get_raw_records).length,
|
|
|
|
data: data
|
2016-07-21 19:11:15 +08:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
# A hack that overrides the new_search_contition method default behavior of the ajax-datatables-rails gem
|
|
|
|
# now the method checks if the column is the created_at or updated_at and generate a custom SQL to parse
|
|
|
|
# it back to the caller method
|
|
|
|
def new_search_condition(column, value)
|
|
|
|
model, column = column.split('.')
|
|
|
|
model = model.constantize
|
|
|
|
case column
|
|
|
|
when 'published_on'
|
|
|
|
casted_column = ::Arel::Nodes::NamedFunction.new('CAST',
|
|
|
|
[ Arel.sql("to_char( protocols.created_at, '#{ formated_date }' ) AS VARCHAR") ] )
|
|
|
|
when 'updated_at'
|
|
|
|
casted_column = ::Arel::Nodes::NamedFunction.new('CAST',
|
|
|
|
[ Arel.sql("to_char( protocols.updated_at, '#{ formated_date }' ) AS VARCHAR") ] )
|
|
|
|
else
|
|
|
|
casted_column = ::Arel::Nodes::NamedFunction.new('CAST',
|
|
|
|
[model.arel_table[column.to_sym].as(typecast)])
|
|
|
|
end
|
|
|
|
casted_column.matches("%#{value}%")
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
# Returns json of current protocols (already paginated)
|
|
|
|
def data
|
|
|
|
result_data = []
|
|
|
|
records.each do |record|
|
|
|
|
protocol = Protocol.find(record.id)
|
|
|
|
result_data << {
|
2017-01-04 22:04:12 +08:00
|
|
|
'DT_RowId': record.id,
|
2018-02-16 01:46:29 +08:00
|
|
|
'DT_CanEdit': can_manage_protocol_in_repository?(protocol),
|
|
|
|
'DT_EditUrl': if can_manage_protocol_in_repository?(protocol)
|
2017-01-04 22:04:12 +08:00
|
|
|
edit_protocol_path(protocol,
|
2017-01-25 16:48:49 +08:00
|
|
|
team: @team,
|
2017-01-04 22:04:12 +08:00
|
|
|
type: @type)
|
|
|
|
end,
|
2018-01-05 22:15:50 +08:00
|
|
|
'DT_CanClone': can_clone_protocol_in_repository?(protocol),
|
|
|
|
'DT_CloneUrl': if can_clone_protocol_in_repository?(protocol)
|
2017-01-04 22:04:12 +08:00
|
|
|
clone_protocol_path(protocol,
|
2017-01-25 16:48:49 +08:00
|
|
|
team: @team,
|
2017-01-04 22:04:12 +08:00
|
|
|
type: @type)
|
|
|
|
end,
|
2018-02-19 21:47:36 +08:00
|
|
|
'DT_CanMakePrivate': can_manage_protocol_in_repository?(protocol),
|
|
|
|
'DT_CanPublish': can_manage_protocol_in_repository?(protocol),
|
|
|
|
'DT_CanArchive': can_manage_protocol_in_repository?(protocol),
|
|
|
|
'DT_CanRestore': can_restore_protocol_in_repository?(protocol),
|
2017-12-06 02:51:44 +08:00
|
|
|
'DT_CanExport': can_read_protocol_in_repository?(protocol),
|
2017-01-04 22:04:12 +08:00
|
|
|
'1': if protocol.in_repository_archived?
|
2017-01-11 22:50:11 +08:00
|
|
|
escape_input(record.name)
|
2017-01-04 22:04:12 +08:00
|
|
|
else
|
|
|
|
name_html(record)
|
|
|
|
end,
|
|
|
|
'2': keywords_html(record),
|
|
|
|
'3': modules_html(record),
|
2017-01-11 22:50:11 +08:00
|
|
|
'4': escape_input(record.full_username_str),
|
2017-01-04 22:04:12 +08:00
|
|
|
'5': timestamp_column_html(record),
|
|
|
|
'6': I18n.l(record.updated_at, format: :full)
|
2016-07-21 19:11:15 +08:00
|
|
|
}
|
|
|
|
end
|
|
|
|
result_data
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_raw_records_base
|
|
|
|
records =
|
|
|
|
Protocol
|
2017-01-25 16:48:49 +08:00
|
|
|
.where(team: @team)
|
2016-07-21 19:11:15 +08:00
|
|
|
.joins('LEFT OUTER JOIN "protocol_protocol_keywords" ON "protocol_protocol_keywords"."protocol_id" = "protocols"."id"')
|
|
|
|
.joins('LEFT OUTER JOIN "protocol_keywords" ON "protocol_protocol_keywords"."protocol_keyword_id" = "protocol_keywords"."id"')
|
|
|
|
|
|
|
|
if @type == :public
|
|
|
|
records =
|
|
|
|
records
|
2017-01-27 18:01:44 +08:00
|
|
|
.joins('LEFT OUTER JOIN users ON users.id = protocols.added_by_id')
|
2017-01-25 16:48:49 +08:00
|
|
|
.where('protocols.protocol_type = ?',
|
|
|
|
Protocol.protocol_types[:in_repository_public])
|
2016-07-21 19:11:15 +08:00
|
|
|
elsif @type == :private
|
|
|
|
records =
|
|
|
|
records
|
2017-01-25 16:48:49 +08:00
|
|
|
.joins('LEFT OUTER JOIN users ON users.id = protocols.added_by_id')
|
|
|
|
.where('protocols.protocol_type = ?',
|
|
|
|
Protocol.protocol_types[:in_repository_private])
|
2016-07-21 19:11:15 +08:00
|
|
|
.where(added_by: @user)
|
|
|
|
else
|
|
|
|
records =
|
|
|
|
records
|
2017-01-25 16:48:49 +08:00
|
|
|
.joins('LEFT OUTER JOIN users ON users.id = protocols.archived_by_id')
|
|
|
|
.where('protocols.protocol_type = ?',
|
|
|
|
Protocol.protocol_types[:in_repository_archived])
|
2016-07-21 19:11:15 +08:00
|
|
|
.where(added_by: @user)
|
|
|
|
end
|
|
|
|
|
|
|
|
records.group('"protocols"."id"')
|
|
|
|
end
|
|
|
|
|
|
|
|
# Query database for records (this will be later paginated and filtered)
|
|
|
|
# after that "data" function will return json
|
|
|
|
def get_raw_records
|
|
|
|
get_raw_records_base
|
|
|
|
.select(
|
|
|
|
'"protocols"."id"',
|
|
|
|
'"protocols"."name"',
|
|
|
|
'"protocols"."protocol_type"',
|
|
|
|
'string_agg("protocol_keywords"."name", \', \') AS "protocol_keywords_str"',
|
|
|
|
'"protocols"."nr_of_linked_children"',
|
|
|
|
'max("users"."full_name") AS "full_username_str"', # "Hack" to get single username
|
|
|
|
'"protocols"."created_at"',
|
|
|
|
'"protocols"."updated_at"',
|
|
|
|
'"protocols"."published_on"',
|
|
|
|
'"protocols"."archived_on"'
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Various helper methods
|
|
|
|
|
|
|
|
def timestamp_db_column
|
|
|
|
if @type == :public
|
|
|
|
"Protocol.published_on"
|
|
|
|
elsif @type == :private
|
|
|
|
"Protocol.created_at"
|
|
|
|
else
|
|
|
|
"Protocol.archived_on"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-12-13 17:39:18 +08:00
|
|
|
def name_html(record)
|
2022-05-27 18:49:10 +08:00
|
|
|
"<a href='#{protocol_path(record)}'>#{escape_input(record.name)}</a>"
|
2016-12-13 17:39:18 +08:00
|
|
|
end
|
|
|
|
|
2016-07-21 19:11:15 +08:00
|
|
|
def keywords_html(record)
|
2021-07-23 17:56:28 +08:00
|
|
|
if !record.protocol_keywords_str || record.protocol_keywords_str.blank?
|
2016-07-21 19:11:15 +08:00
|
|
|
"<i>#{I18n.t("protocols.no_keywords")}</i>"
|
|
|
|
else
|
|
|
|
kws = record.protocol_keywords_str.split(", ")
|
|
|
|
res = []
|
|
|
|
kws.sort_by{ |word| word.downcase }.each do |kw|
|
2017-01-26 22:52:46 +08:00
|
|
|
sanitized_kw = sanitize_input(kw)
|
|
|
|
res << "<a href='#' data-action='filter' " \
|
|
|
|
"data-param='#{sanitized_kw}'>#{sanitized_kw}</a>"
|
2016-07-21 19:11:15 +08:00
|
|
|
end
|
2017-01-26 22:52:46 +08:00
|
|
|
res.join(', ')
|
2016-07-21 19:11:15 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def modules_html(record)
|
2020-04-28 18:30:43 +08:00
|
|
|
"<a href='#' data-action='load-linked-children'" \
|
|
|
|
"data-url='#{linked_children_protocol_path(record)}'>" \
|
2018-08-21 18:08:14 +08:00
|
|
|
"#{record.nr_of_linked_children}" \
|
|
|
|
"</a>"
|
2016-07-21 19:11:15 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def timestamp_column_html(record)
|
|
|
|
if @type == :public
|
|
|
|
I18n.l(record.published_on, format: :full)
|
|
|
|
elsif @type == :private
|
|
|
|
I18n.l(record.created_at, format: :full)
|
|
|
|
else
|
|
|
|
I18n.l(record.archived_on, format: :full)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# OVERRIDE - This is only called when filtering results;
|
|
|
|
# when using GROUP BY function, SQL cannot perform a WHERE
|
|
|
|
# clause on aggregated columns (protocol keywords & users' full_name), but
|
|
|
|
# since we want those 2 columns to be searchable/filterable, we do an "inner"
|
|
|
|
# query where we select only protocol IDs which are filtered by those 2 columns
|
|
|
|
# using HAVING keyword (which is the correct way to filter aggregated columns).
|
|
|
|
# Another OR is then appended to the WHERE clause, checking if protocol is inside
|
|
|
|
# this list of IDs.
|
|
|
|
def build_conditions_for(query)
|
|
|
|
# Inner query to retrieve list of protocol IDs where concatenated
|
|
|
|
# protocol keywords string, or user's full_name contains searched query
|
2017-06-30 21:20:27 +08:00
|
|
|
search_val = dt_params[:search][:value]
|
2016-07-21 19:11:15 +08:00
|
|
|
records_having = get_raw_records_base.having(
|
|
|
|
::Arel::Nodes::NamedFunction.new(
|
|
|
|
'CAST',
|
|
|
|
[::Arel::Nodes::SqlLiteral.new("string_agg(\"protocol_keywords\".\"name\", ' ') AS #{typecast}")]
|
|
|
|
).matches("%#{sanitize_sql_like(search_val)}%").to_sql +
|
|
|
|
" OR " +
|
|
|
|
::Arel::Nodes::NamedFunction.new(
|
|
|
|
'CAST',
|
|
|
|
[::Arel::Nodes::SqlLiteral.new("max(\"users\".\"full_name\") AS #{typecast}")]
|
|
|
|
).matches("%#{sanitize_sql_like(search_val)}%").to_sql
|
|
|
|
).select(:id)
|
|
|
|
|
|
|
|
# Call parent function
|
|
|
|
criteria = super(query)
|
|
|
|
|
|
|
|
# Aight, now append another or
|
|
|
|
criteria = criteria.or(Protocol.arel_table[:id].in(records_having.arel))
|
|
|
|
criteria
|
|
|
|
end
|
|
|
|
end
|