kanbanworkflowstimelinescrumrubyroadmapproject-planningproject-managementopenprojectangularissue-trackerifcgantt-chartganttbug-trackerboardsbcf
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
272 lines
8.2 KiB
272 lines
8.2 KiB
#-- copyright
|
|
# OpenProject is a project management system.
|
|
# Copyright (C) 2012-2015 the OpenProject Foundation (OPF)
|
|
#
|
|
# This program is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU General Public License version 3.
|
|
#
|
|
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
|
|
# Copyright (C) 2006-2013 Jean-Philippe Lang
|
|
# Copyright (C) 2010-2013 the ChiliProject Team
|
|
#
|
|
# This program is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU General Public License
|
|
# as published by the Free Software Foundation; either version 2
|
|
# of the License, or (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
#
|
|
# See doc/COPYRIGHT.rdoc for more details.
|
|
#++
|
|
|
|
module Api::Experimental::Concerns::ColumnData
|
|
def get_columns_for_json(columns)
|
|
columns.map do |column|
|
|
{ name: column.name,
|
|
title: column.caption,
|
|
sortable: column.sortable?,
|
|
groupable: column.groupable?,
|
|
custom_field: column.is_a?(QueryCustomFieldColumn) &&
|
|
column.custom_field.as_json(only: [:id, :field_format]),
|
|
meta_data: get_column_meta(column)
|
|
}
|
|
end
|
|
end
|
|
|
|
private
|
|
|
|
def get_column_meta(column)
|
|
# This is where we want to add column specific behaviour to instruct the
|
|
# front end how to deal with it. Needs to be things like user link, project
|
|
# link, datetime
|
|
{
|
|
data_type: column_data_type(column),
|
|
link: link_meta(column)
|
|
}
|
|
end
|
|
|
|
def link_meta(column)
|
|
link_meta = static_link_meta[column.name]
|
|
|
|
if link_meta
|
|
link_meta
|
|
elsif column.respond_to?(:custom_field)
|
|
linked_custom_field_meta(column)
|
|
else
|
|
{ display: false }
|
|
end
|
|
end
|
|
|
|
def static_link_meta
|
|
{
|
|
id: { display: true, model_type: 'work_package' },
|
|
subject: { display: true, model_type: 'work_package' },
|
|
type: { display: false },
|
|
status: { display: false },
|
|
priority: { display: false },
|
|
parent: { display: true, model_type: 'work_package' },
|
|
assigned_to: { display: true, model_type: 'user' },
|
|
responsible: { display: true, model_type: 'user' },
|
|
author: { display: true, model_type: 'user' },
|
|
project: { display: true, model_type: 'project' }
|
|
}
|
|
end
|
|
|
|
def linked_custom_field_meta(column)
|
|
case column.custom_field.field_format
|
|
when 'user', 'version'
|
|
{ display: true, model_type: column.custom_field.field_format }
|
|
else
|
|
{ display: false }
|
|
end
|
|
end
|
|
|
|
def column_data_type(column)
|
|
if column.is_a?(QueryCustomFieldColumn)
|
|
column.custom_field.field_format
|
|
elsif column.class.to_s =~ /CurrencyQueryColumn/
|
|
'currency'
|
|
elsif c = WorkPackage.columns_hash[column.name.to_s]
|
|
c.type.to_s
|
|
elsif WorkPackage.columns_hash[column.name.to_s + '_id']
|
|
'object'
|
|
else
|
|
'default'
|
|
end
|
|
end
|
|
|
|
def valid_columns(columns)
|
|
column_names, custom_field_ids = separate_columns_by_custom_fields(columns)
|
|
|
|
valid_column_names = Query.available_columns.map { |c| c.name.to_s } & column_names
|
|
|
|
existing_cf_ids = existing_custom_field_ids(custom_field_ids)
|
|
|
|
valid_cf_column_names = columns.select { |name|
|
|
id = custom_field_id_in(name)
|
|
existing_cf_ids.include?(id)
|
|
}
|
|
|
|
# keep order of provided columns
|
|
columns & (valid_column_names + valid_cf_column_names)
|
|
end
|
|
|
|
def separate_columns_by_custom_fields(columns)
|
|
cf_columns, non_cf_columns = columns.partition { |name| custom_field_id_in(name) }
|
|
|
|
cf_columns_id = cf_columns.map { |name| custom_field_id_in(name) }
|
|
|
|
[non_cf_columns, cf_columns_id]
|
|
end
|
|
|
|
def existing_custom_field_ids(ids)
|
|
if ids.empty?
|
|
[]
|
|
else
|
|
WorkPackageCustomField.where(id: ids).pluck(:id).map(&:to_s)
|
|
end
|
|
end
|
|
|
|
def columns_total_sums(column_names, work_packages)
|
|
column_names.map do |column_name|
|
|
column_sum(column_name, work_packages)
|
|
end
|
|
end
|
|
|
|
def column_sum(column_name, work_packages)
|
|
if column_should_be_summed_up?(column_name)
|
|
column_data = fetch_column_data(column_name, work_packages, false)
|
|
|
|
column_data.map { |c| c.nil? ? 0 : c }
|
|
.compact
|
|
.sum
|
|
end
|
|
end
|
|
|
|
def columns_group_sums(column_names, work_packages, group_by)
|
|
# NOTE RS: This is basically the grouped_sums method from sums.rb but we
|
|
# have no query to play with here
|
|
return unless group_by
|
|
|
|
if custom_field_id_in(group_by)
|
|
sum_columns(column_names, work_packages) do |wp|
|
|
wp.custom_values.detect { |cv| cv.custom_field_id == custom_field_id_in(group_by).to_i }
|
|
end
|
|
else
|
|
sum_columns(column_names, work_packages) do |wp|
|
|
wp.send(group_by)
|
|
end
|
|
end
|
|
end
|
|
|
|
def sum_columns(column_names, work_packages)
|
|
column_names.map do |column_name|
|
|
work_packages.map { |wp| yield wp }
|
|
.uniq
|
|
.inject({}) do |group_sums, current_group|
|
|
work_packages_in_current_group = work_packages.select { |wp|
|
|
(yield wp) == current_group
|
|
}
|
|
|
|
group_sums.merge current_group => column_sum(column_name, work_packages_in_current_group)
|
|
end
|
|
end
|
|
end
|
|
|
|
# FIXME: Deprecated. Should be removable once the experimental WP controller is gone
|
|
def includes_for_columns(column_names)
|
|
column_names = Array(column_names)
|
|
includes = (WorkPackage.reflections.keys.map(&:to_sym) & column_names.map(&:to_sym))
|
|
|
|
if column_names.any? { |c| custom_field_id_in(c) }
|
|
includes << { custom_values: :custom_field }
|
|
end
|
|
|
|
includes
|
|
end
|
|
|
|
def fetch_columns_data(column_names, work_packages)
|
|
column_names, custom_field_column_ids = separate_columns_by_custom_fields(column_names)
|
|
|
|
columns = column_names.map { |column_name|
|
|
fetch_non_custom_field_column_data(column_name, work_packages)
|
|
}
|
|
|
|
columns += custom_field_column_ids.map { |cf_id|
|
|
fetch_custom_field_column_data(cf_id, work_packages)
|
|
}
|
|
|
|
columns
|
|
end
|
|
|
|
def fetch_column_data(column_name, work_packages, display = true)
|
|
if custom_field_id = custom_field_id_in(column_name)
|
|
fetch_custom_field_column_data(custom_field_id, work_packages, display)
|
|
else
|
|
fetch_non_custom_field_column_data(column_name, work_packages)
|
|
end
|
|
end
|
|
|
|
def fetch_custom_field_column_data(custom_field_id, work_packages, display = true)
|
|
custom_field_data = work_packages.map { |wp|
|
|
wp.custom_values_display_data(custom_field_id)
|
|
}
|
|
|
|
if display
|
|
custom_field_data.flatten
|
|
else
|
|
custom_field_data.flatten.map { |d| d.nil? ? nil : d[:value] }
|
|
end
|
|
end
|
|
|
|
def fetch_non_custom_field_column_data(column_name, work_packages)
|
|
work_packages.map do |work_package|
|
|
# Note: Doing as_json here because if we just take the
|
|
# value.attributes then we can't get any methods later. Name and
|
|
# subject are the default properties that the front end currently
|
|
# looks for to summarize an object.
|
|
value = work_package.send(column_name)
|
|
|
|
if value.is_a?(ActiveRecord::Base)
|
|
value.as_json(only: 'id', methods: [:name, :subject])
|
|
else
|
|
value
|
|
end
|
|
end
|
|
end
|
|
|
|
def column_should_be_summed_up?(column_name)
|
|
# see ::Query::Sums mix in
|
|
column_is_numeric?(column_name) &&
|
|
Setting.work_package_list_summable_columns.include?(column_name.to_s)
|
|
end
|
|
|
|
def column_is_numeric?(column_name)
|
|
# TODO RS: We want to leave out ids even though they are numeric
|
|
[:int, :integer, :float].include? column_type(column_name)
|
|
end
|
|
|
|
def custom_field_id_in(name)
|
|
groups = name.to_s.scan(/cf_(\d+)/).flatten
|
|
|
|
if groups
|
|
groups[0]
|
|
end
|
|
end
|
|
|
|
def column_type(column_name)
|
|
if id = custom_field_id_in(column_name)
|
|
CustomField.find(id).field_format.to_sym
|
|
else
|
|
column = WorkPackage.columns_hash[column_name]
|
|
column.nil? ? :none : column.type
|
|
end
|
|
end
|
|
end
|
|
|