Skip to content

Commit c978859

Browse files
committed
Merge branch 'release/14.1' into dev
2 parents 4e5b6f6 + 0541e03 commit c978859

File tree

6 files changed

+84
-13
lines changed

6 files changed

+84
-13
lines changed

docs/system-admin-guide/manage-work-packages/work-package-settings/README.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -17,15 +17,15 @@ You can adjust the following:
1717

1818
3. **Use current date as start date for new work packages**. This way the current date will always be set as a start date if your create new work packages. Also, if you copy projects, the new work packages will get the current date as start date.
1919

20-
4. **Progress calculation** lets you pick between two modes for how the **%&nsbp;Complete** field is calculated for work packages.
20+
4. **Progress calculation** lets you pick between two modes for how the **% Complete** field is calculated for work packages.
2121
- **Work-based**: % Complete is automatically calculated based on Work and Remaining work values for that work package, both of which are then necessary to have a value for % Complete.
2222
- **Status-based**: you will have to define fixed % Complete values for each [work package status](../work-package-status), which will update automatically when team members update the status of their work packages.
2323

2424
5. **Default highlighting mode** (Enterprise add-on) defines which should be the default [attribute highlighting](../../../user-guide/work-packages/work-package-table-configuration/#attribute-highlighting-enterprise-add-on) mode, e.g. to highlight the following criteria in the work package table. This setting is only available for Enterprise on-premises and Enterprise cloud users.
2525

2626
![default highlighting mode](openproject_system_guide_default_highlighting_mode.png)
2727

28-
6. Customize the appearance of the work package tables to **define which work package attributes are displayed in the work package tables by default**.
28+
6. Customize the appearance of the work package lists to **define which work package attributes are displayed in the work package lists by default and in what order**.
2929

3030
Do not forget to save your changes with the blue **Save** button at the bottom.
3131

docs/user-guide/work-packages/work-packages-faq/README.md

+1-6
Original file line numberDiff line numberDiff line change
@@ -45,12 +45,7 @@ You can set the assignee filter in the work package table to "Assignee and belon
4545

4646
### How can I track the progress of my work package?
4747

48-
You can track the progress either manually by changing the progress
49-
bar in the work package details yourself. Or you can track it
50-
automatically by assigning the progress in % to each status of
51-
a work package. Please find the guide on how to do the automatic
52-
tracking (in bullet point 5)
53-
[here](../../../system-admin-guide/manage-work-packages/work-package-settings).
48+
Progress of a work package is demonstrated by the value of **% Complete**. This is calculated either based on the status of the work package or on the values of *Work* and *Remaining work*. Please read more on progress reporting [here](../../time-and-costs/progress-tracking/#progress-reporting-modes).
5449

5550
### How can I track the progress of work packages with children?
5651

lib/api/v3/projects/project_eager_loading_wrapper.rb

+15-3
Original file line numberDiff line numberDiff line change
@@ -37,12 +37,15 @@ class ProjectEagerLoadingWrapper < API::V3::Utilities::EagerLoading::EagerLoadin
3737

3838
class << self
3939
def wrap(projects)
40-
custom_fields = projects.first.available_custom_fields if projects.present?
41-
ancestors = ancestor_projects(projects) if projects.present?
40+
if projects.present?
41+
custom_fields_by_project_id = custom_fields_from_projects
42+
43+
ancestors = ancestor_projects(projects)
44+
end
4245

4346
super
4447
.each do |project|
45-
project.available_custom_fields = custom_fields
48+
project.available_custom_fields = custom_fields_by_project_id[project.id]
4649
project.ancestors_from_root = ancestors.select { |a| a.is_ancestor_of?(project) }.sort_by(&:lft)
4750
end
4851
end
@@ -59,6 +62,15 @@ def ancestor_project_select(project)
5962

6063
projects_table[:lft].lt(project.lft).and(projects_table[:rgt].gt(project.rgt))
6164
end
65+
66+
def custom_fields_from_projects
67+
ProjectCustomFieldProjectMapping
68+
.eager_load(:project_custom_field)
69+
.merge(ProjectCustomField.visible)
70+
.each_with_object(Hash.new { |h, k| h[k] = [] }) do |mapping, acc|
71+
acc[mapping.project_id] << mapping.project_custom_field
72+
end
73+
end
6274
end
6375
end
6476
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
#-- copyright
2+
# OpenProject is an open source project management software.
3+
# Copyright (C) 2012-2024 the OpenProject GmbH
4+
#
5+
# This program is free software; you can redistribute it and/or
6+
# modify it under the terms of the GNU General Public License version 3.
7+
#
8+
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
9+
# Copyright (C) 2006-2013 Jean-Philippe Lang
10+
# Copyright (C) 2010-2013 the ChiliProject Team
11+
#
12+
# This program is free software; you can redistribute it and/or
13+
# modify it under the terms of the GNU General Public License
14+
# as published by the Free Software Foundation; either version 2
15+
# of the License, or (at your option) any later version.
16+
#
17+
# This program is distributed in the hope that it will be useful,
18+
# but WITHOUT ANY WARRANTY; without even the implied warranty of
19+
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20+
# GNU General Public License for more details.
21+
#
22+
# You should have received a copy of the GNU General Public License
23+
# along with this program; if not, write to the Free Software
24+
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
25+
#
26+
# See COPYRIGHT and LICENSE files for more details.
27+
#++
28+
29+
require "spec_helper"
30+
31+
RSpec.describe API::V3::Projects::ProjectEagerLoadingWrapper do
32+
shared_let(:projects) { create_list(:project, 3) }
33+
34+
describe ".wrap" do
35+
subject(:loaded_projects) { described_class.wrap(projects) }
36+
37+
it "returns wrapped projects with relations eager loaded" do
38+
expect(loaded_projects.size).to eq(projects.size)
39+
association_names = %i[@available_custom_fields @ancestors_from_root]
40+
loaded_projects.each do |loaded_project|
41+
association_names.each do |association|
42+
expect(loaded_project.__getobj__.instance_variables).to include(association)
43+
end
44+
end
45+
end
46+
47+
context "with available custom fields" do
48+
let!(:text_project_custom_field) do
49+
create :text_project_custom_field, projects: [projects.second, projects.third]
50+
end
51+
52+
let!(:string_project_custom_field) do
53+
create :string_project_custom_field, projects: [projects.third]
54+
end
55+
56+
it "returns available custom fields for each project separately" do
57+
expect(loaded_projects.first.available_custom_fields).to eq([])
58+
expect(loaded_projects.second.available_custom_fields).to eq([text_project_custom_field])
59+
expect(loaded_projects.third.available_custom_fields)
60+
.to eq([text_project_custom_field, string_project_custom_field])
61+
end
62+
end
63+
end
64+
end

spec/requests/api/v3/versions/available_projects_resource_spec.rb

+2-2
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@
2929
require "spec_helper"
3030
require "rack/test"
3131

32-
RSpec.describe "API v3 members available projects resource" do
32+
RSpec.describe "API v3 versions available projects resource" do
3333
include Rack::Test::Methods
3434
include API::V3::Utilities::PathHelper
3535

@@ -78,7 +78,7 @@
7878

7979
subject(:response) { last_response }
8080

81-
describe "GET api/v3/members/available_projects" do
81+
describe "GET api/v3/versions/available_projects" do
8282
let(:projects) { [manage_project, view_project, unauthorized_project] }
8383
let(:path) { api_v3_paths.versions_available_projects }
8484

0 commit comments

Comments
 (0)