additionals/app/helpers/additionals_queries_helper.rb
2024-08-27 09:30:58 +02:00

360 lines
14 KiB
Ruby

# frozen_string_literal: true
module AdditionalsQueriesHelper
def additionals_limit_for_pager
params[:search].present? ? Additionals.max_live_search_results : per_page_option
end
def render_live_search_info(entries:, count: nil)
return if count.nil? ||
count <= Additionals.max_live_search_results ||
entries.count < Additionals.max_live_search_results
tag.p class: 'icon icon-warning' do
tag.em class: 'info' do
l :info_live_search_result_restriction, value: Additionals.max_live_search_results
end
end
end
def additionals_query_session_key(object_type)
:"#{object_type}_query"
end
def additionals_retrieve_query(object_type, user_filter: nil, search_string: nil)
session_key = additionals_query_session_key object_type
query_class = Object.const_get :"#{object_type.camelcase}Query"
if params[:query_id].present?
additionals_load_query_id(query_class,
session_key,
params[:query_id],
object_type,
user_filter:,
search_string:)
elsif api_request? ||
params[:set_filter] ||
session[session_key].nil? ||
session[session_key][:project_id] != (@project ? @project.id : nil)
# Give it a name, required to be valid
@query = query_class.new name: '_'
@query.project = @project
@query.user_filter = user_filter if user_filter
@query.search_string = search_string if search_string
@query.build_from_params params
session[session_key] = { project_id: @query.project_id }
# session has a limit to 4k, we have to use a cache for it for larger data
Rails.cache.write(additionals_query_cache_key(object_type),
filters: @query.filters,
group_by: @query.group_by,
column_names: @query.column_names,
totalable_names: @query.totalable_names,
sort_criteria: params[:sort].presence || @query.sort_criteria.to_a)
else
# retrieve from session
@query = query_class.find_by id: session[session_key][:id] if session[session_key][:id]
session_data = Rails.cache.read additionals_query_cache_key(object_type)
@query ||= query_class.new(name: '_',
filters: session_data.nil? ? nil : session_data[:filters],
group_by: session_data.nil? ? nil : session_data[:group_by],
column_names: session_data.nil? ? nil : session_data[:column_names],
totalable_names: session_data.nil? ? nil : session_data[:totalable_names],
sort_criteria: params[:sort].presence || (session_data.nil? ? nil : session_data[:sort_criteria]))
@query.project = @project
@query.user_filter = user_filter if user_filter
@query.display_type = params[:display_type] if params[:display_type]
if params[:sort].present?
@query.sort_criteria = params[:sort]
# we have to write cache for sort order
Rails.cache.write(additionals_query_cache_key(object_type),
filters: @query.filters,
group_by: @query.group_by,
column_names: @query.column_names,
totalable_names: @query.totalable_names,
sort_criteria: params[:sort])
elsif session_data.present?
@query.sort_criteria = session_data[:sort_criteria]
end
end
end
def additionals_load_query_id(query_class, session_key, query_id, object_type, user_filter: nil, search_string: nil)
scope = query_class.where project_id: nil
scope = scope.or query_class.where(project_id: @project) if @project
@query = scope.find query_id
raise ::Unauthorized unless @query.visible?
@query.project = @project
@query.user_filter = user_filter if user_filter
@query.search_string = search_string if search_string
session[session_key] = { id: @query.id, project_id: @query.project_id }
@query.sort_criteria = params[:sort] if params[:sort].present?
# we have to write cache for sort order
Rails.cache.write(additionals_query_cache_key(object_type),
filters: @query.filters,
group_by: @query.group_by,
column_names: @query.column_names,
totalable_names: @query.totalable_names,
sort_criteria: @query.sort_criteria)
end
def additionals_query_cache_key(object_type)
project_id = @project ? @project.id : 0
"#{object_type}_query_data_#{session.id}_#{project_id}"
end
def render_grouped_users_with_select2(users, search_term: nil, with_me: true, with_ano: false, me_value: 'me')
@users = { active: [], groups: [], registered: [], locked: [] }
users = users.like search_term if search_term.present?
sorted_users = users.select("users.*, #{User.table_name}.last_login_on IS NULL AS select_order")
.order("select_order ASC, #{User.table_name}.last_login_on DESC")
.limit(AdditionalsConf.select2_init_entries)
.to_a
.sort_by(&:name)
with_users = false
sorted_users.each do |user|
case user.type
when 'User'
case user.status
when Principal::STATUS_ACTIVE
@users[:active] << { id: user.id, name: user.name, obj: user }
with_users = true
when Principal::STATUS_REGISTERED
@users[:registered] << { id: user.id, name: user.name, obj: user }
with_users = true
when Principal::STATUS_LOCKED
@users[:locked] << { id: user.id, name: user.name, obj: user }
with_users = true
end
when 'Group'
@users[:groups] << { id: user.id, name: user.name, obj: user }
with_users = true
end
end
# TODO: this should be false without search results?
# with_me = false unless with_users
# Additionals.debug "with_me: #{with_me}"
# Additionals.debug "active: #{@users[:active].pluck :id}"
# Additionals.debug "locked: #{@users[:locked].pluck :id}"
# Additionals.debug "groups: #{@users[:groups].pluck :id}"
render layout: false,
format: :json,
partial: 'auto_completes/grouped_users',
locals: { with_me: with_me && (search_term.blank? || l(:label_me).downcase.include?(search_term.downcase)),
with_ano: with_ano && (search_term.blank? || l(:label_user_anonymous).downcase.include?(search_term.downcase)),
me_value:,
sep_required: false }
end
def additionals_query_to_xlsx(query, no_id_link: false)
require 'write_xlsx'
options = { no_id_link:,
filename: StringIO.new(+'') }
export_to_xlsx query.entries, query.columns, options
options[:filename].string
end
def additionals_result_to_xlsx(items, columns, options)
raise 'option filename is mission' if options[:filename].blank?
require 'write_xlsx'
export_to_xlsx items, columns, options
end
def export_to_xlsx(items, columns, options)
workbook = WriteXLSX.new options[:filename]
worksheet = workbook.add_worksheet
# Freeze header row and # column.
freeze_row = options[:freeze_first_row].nil? || options[:freeze_first_row] ? 1 : 0
freeze_column = options[:freeze_first_column].nil? || options[:freeze_first_column] ? 1 : 0
worksheet.freeze_panes freeze_row, freeze_column
options[:columns_width] = if options[:xlsx_write_header_row].present?
send options[:xlsx_write_header_row], workbook, worksheet, columns
else
xlsx_write_header_row workbook, worksheet, columns
end
options[:columns_width] = if options[:xlsx_write_item_rows].present?
send options[:xlsx_write_item_rows], workbook, worksheet, columns, items, options
else
xlsx_write_item_rows workbook, worksheet, columns, items, options
end
columns.size.times do |index|
worksheet.set_column index, index, options[:columns_width][index]
end
workbook.close
end
def xlsx_write_header_row(workbook, worksheet, columns)
columns_width = []
columns.each_with_index do |c, index|
value = if c.is_a? String
c
else
c.caption.to_s
end
worksheet.write 0, index, value, workbook.add_format(xlsx_cell_format(:header))
columns_width << xlsx_get_column_width(value)
end
columns_width
end
def xlsx_write_item_rows(workbook, worksheet, columns, items, options)
hyperlink_format = workbook.add_format xlsx_cell_format(:link)
even_text_format = workbook.add_format xlsx_cell_format(:text, '', 0)
even_text_format.set_num_format 0x31
odd_text_format = workbook.add_format xlsx_cell_format(:text, '', 1)
odd_text_format.set_num_format 0x31
items.each_with_index do |line, line_index|
columns.each_with_index do |c, column_index|
value = csv_content(c, line).dup
if c.name == :id # ID
if options[:no_id_link]
# id without link
worksheet.write(line_index + 1,
column_index,
value,
workbook.add_format(xlsx_cell_format(:cell, value, line_index)))
else
link = url_for controller: line.class.name.underscore.pluralize, action: 'show', id: line.id
worksheet.write line_index + 1, column_index, link, hyperlink_format, value
end
elsif xlsx_hyperlink_cell? value
worksheet.write line_index + 1, column_index, value[0..254], hyperlink_format, value
elsif !c.inline?
# block column can be multiline strings
value.gsub! "\r\n", "\n"
text_format = line_index.even? ? even_text_format : odd_text_format
worksheet.write_rich_string line_index + 1, column_index, value, text_format
else
worksheet.write(line_index + 1,
column_index,
value,
workbook.add_format(xlsx_cell_format(:cell, value, line_index)))
end
width = xlsx_get_column_width value
options[:columns_width][column_index] = width if options[:columns_width][column_index] < width
end
end
options[:columns_width]
end
def xlsx_get_column_width(value)
value_str = value.to_s
# 1.1: margin
width = (value_str.length + value_str.chars.count { |e| !e.ascii_only? }) * 1.1 + 1
# 30: max width
[width, 30].min
end
def xlsx_cell_format(type, value = 0, index = 0)
format = { border: 1, text_wrap: 1, valign: 'top' }
case type
when :header
format[:bold] = 1
format[:color] = 'white'
format[:bg_color] = 'gray'
when :link
format[:color] = 'blue'
format[:underline] = 1
format[:bg_color] = 'silver' unless index.even?
else
format[:bg_color] = 'silver' unless index.even?
format[:color] = 'red' if value.is_a?(Numeric) && value.negative?
end
format
end
def xlsx_hyperlink_cell?(token)
return false if token.blank? || !token.is_a?(String)
# Match http, https or ftp URL
%r{\A[fh]tt?ps?://}.match?(token) ||
# Match mailto:
token.start_with?('mailto:') ||
# Match internal or external sheet link
/\A(?:in|ex)ternal:/.match?(token)
end
def set_flash_from_bulk_save(entries, unsaved_ids, name_plural:)
if unsaved_ids.empty?
flash[:notice] = l :notice_successful_update unless entries.empty?
else
flash[:error] = l :notice_failed_to_save_entity,
name_plural:,
count: unsaved_ids.size,
total: entries.size,
ids: "##{unsaved_ids.join ', #'}"
end
end
# Returns the query definition as hidden field tags
# columns in ignored_column_names are skipped (names as symbols)
# TODO: this is a temporary fix and should be removed
# after https://www.redmine.org/issues/29830 is in Redmine core.
def query_as_hidden_field_tags(query, with_block_columns: false)
tags = hidden_field_tag 'set_filter', '1', id: nil
if query.filters.present?
query.filters.each do |field, filter|
tags << hidden_field_tag('f[]', field, id: nil)
tags << hidden_field_tag("op[#{field}]", filter[:operator], id: nil)
filter[:values].each do |value|
tags << hidden_field_tag("v[#{field}][]", value, id: nil)
end
end
else
tags << hidden_field_tag('f[]', '', id: nil)
end
ignored_block_columns = with_block_columns ? [] : query.block_columns.map(&:name)
query.columns.each do |column|
next if ignored_block_columns.include? column.name
tags << hidden_field_tag('c[]', column.name, id: nil)
end
if query.totalable_names.present?
query.totalable_names.each do |name|
tags << hidden_field_tag('t[]', name, id: nil)
end
end
tags << hidden_field_tag('group_by', query.group_by, id: nil) if query.group_by.present?
tags << hidden_field_tag('sort', query.sort_criteria.to_param, id: nil) if query.sort_criteria.present?
tags
end
def build_search_query_term(params)
(params[:q] || params[:term]).to_s.strip
end
def link_to_nonzero(value, path)
value.zero? ? value : link_to(value, path)
end
# NOTE: copy of Redmine 5.1
# TODO: this method can be removed, if Redmine 5.0 is not supported anymore
def filename_for_export(query, default_name)
query_name = params[:query_name].presence || query.name
query_name = default_name if query_name == '_' || query_name.blank?
# Convert file names using the same rules as Wiki titles
filename_for_content_disposition(Wiki.titleize(query_name).downcase)
end
end