Ignore non-working days when rescheduling an issue (#2161).
Weekly non-working days can be configured in application settings (set to saturday and sunday by default). git-svn-id: svn+ssh://rubyforge.org/var/svn/redmine/trunk@10747 e93f8b46-1217-0410-a6f0-8f06a7374b81
This commit is contained in:
parent
48d83884c3
commit
b71355f10b
|
@ -56,7 +56,7 @@ module SettingsHelper
|
|||
Setting.send(setting).include?(value),
|
||||
:id => nil
|
||||
) + text.to_s,
|
||||
:class => 'block'
|
||||
:class => (options[:inline] ? 'inline' : 'block')
|
||||
)
|
||||
end.join.html_safe
|
||||
end
|
||||
|
|
|
@ -17,6 +17,7 @@
|
|||
|
||||
class Issue < ActiveRecord::Base
|
||||
include Redmine::SafeAttributes
|
||||
include Redmine::Utils::DateCalculation
|
||||
|
||||
belongs_to :project
|
||||
belongs_to :tracker
|
||||
|
@ -863,6 +864,11 @@ class Issue < ActiveRecord::Base
|
|||
(start_date && due_date) ? due_date - start_date : 0
|
||||
end
|
||||
|
||||
# Returns the duration in working days
|
||||
def working_duration
|
||||
(start_date && due_date) ? working_days(start_date, due_date) : 0
|
||||
end
|
||||
|
||||
def soonest_start
|
||||
@soonest_start ||= (
|
||||
relations_to.collect{|relation| relation.successor_soonest_start} +
|
||||
|
@ -870,22 +876,33 @@ class Issue < ActiveRecord::Base
|
|||
).compact.max
|
||||
end
|
||||
|
||||
def reschedule_after(date)
|
||||
# Sets start_date on the given date or the next working day
|
||||
# and changes due_date to keep the same working duration.
|
||||
def reschedule_on(date)
|
||||
wd = working_duration
|
||||
date = next_working_date(date)
|
||||
self.start_date = date
|
||||
self.due_date = add_working_days(date, wd)
|
||||
end
|
||||
|
||||
# Reschedules the issue on the given date or the next working day and saves the record.
|
||||
# If the issue is a parent task, this is done by rescheduling its subtasks.
|
||||
def reschedule_on!(date)
|
||||
return if date.nil?
|
||||
if leaf?
|
||||
if start_date.nil? || start_date < date
|
||||
self.start_date, self.due_date = date, date + duration
|
||||
reschedule_on(date)
|
||||
begin
|
||||
save
|
||||
rescue ActiveRecord::StaleObjectError
|
||||
reload
|
||||
self.start_date, self.due_date = date, date + duration
|
||||
reschedule_on(date)
|
||||
save
|
||||
end
|
||||
end
|
||||
else
|
||||
leaves.each do |leaf|
|
||||
leaf.reschedule_after(date)
|
||||
leaf.reschedule_on!(date)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
|
@ -135,7 +135,7 @@ class IssueRelation < ActiveRecord::Base
|
|||
def set_issue_to_dates
|
||||
soonest_start = self.successor_soonest_start
|
||||
if soonest_start && issue_to
|
||||
issue_to.reschedule_after(soonest_start)
|
||||
issue_to.reschedule_on!(soonest_start)
|
||||
end
|
||||
end
|
||||
|
||||
|
|
|
@ -13,6 +13,8 @@
|
|||
|
||||
<p><%= setting_select :issue_done_ratio, Issue::DONE_RATIO_OPTIONS.collect {|i| [l("setting_issue_done_ratio_#{i}"), i]} %></p>
|
||||
|
||||
<p><%= setting_multiselect :non_working_week_days, (1..7).map {|d| [day_name(d), d]}, :inline => true %></p>
|
||||
|
||||
<p><%= setting_text_field :issues_export_limit, :size => 6 %></p>
|
||||
|
||||
<p><%= setting_text_field :gantt_items_limit, :size => 6 %></p>
|
||||
|
|
|
@ -399,6 +399,7 @@ en:
|
|||
setting_session_timeout: Session inactivity timeout
|
||||
setting_thumbnails_enabled: Display attachment thumbnails
|
||||
setting_thumbnails_size: Thumbnails size (in pixels)
|
||||
setting_non_working_week_days: Non-working days
|
||||
|
||||
permission_add_project: Create project
|
||||
permission_add_subprojects: Create subprojects
|
||||
|
|
|
@ -395,6 +395,7 @@ fr:
|
|||
setting_session_timeout: Durée maximale d'inactivité
|
||||
setting_thumbnails_enabled: Afficher les vignettes des images
|
||||
setting_thumbnails_size: Taille des vignettes (en pixels)
|
||||
setting_non_working_week_days: Jours non travaillés
|
||||
|
||||
permission_add_project: Créer un projet
|
||||
permission_add_subprojects: Créer des sous-projets
|
||||
|
|
|
@ -220,3 +220,8 @@ thumbnails_enabled:
|
|||
thumbnails_size:
|
||||
format: int
|
||||
default: 100
|
||||
non_working_week_days:
|
||||
serialized: true
|
||||
default:
|
||||
- 6
|
||||
- 7
|
||||
|
|
|
@ -51,5 +51,68 @@ module Redmine
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
module DateCalculation
|
||||
# Returns the number of working days between from and to
|
||||
def working_days(from, to)
|
||||
days = (to - from).to_i
|
||||
if days > 0
|
||||
weeks = days / 7
|
||||
result = weeks * (7 - non_working_week_days.size)
|
||||
days_left = days - weeks * 7
|
||||
start_cwday = from.cwday
|
||||
days_left.times do |i|
|
||||
unless non_working_week_days.include?(((start_cwday + i - 1) % 7) + 1)
|
||||
result += 1
|
||||
end
|
||||
end
|
||||
result
|
||||
else
|
||||
0
|
||||
end
|
||||
end
|
||||
|
||||
# Adds working days to the given date
|
||||
def add_working_days(date, working_days)
|
||||
if working_days > 0
|
||||
weeks = working_days / (7 - non_working_week_days.size)
|
||||
result = weeks * 7
|
||||
days_left = working_days - weeks * (7 - non_working_week_days.size)
|
||||
cwday = date.cwday
|
||||
while days_left > 0
|
||||
cwday += 1
|
||||
unless non_working_week_days.include?(((cwday - 1) % 7) + 1)
|
||||
days_left -= 1
|
||||
end
|
||||
result += 1
|
||||
end
|
||||
next_working_date(date + result)
|
||||
else
|
||||
date
|
||||
end
|
||||
end
|
||||
|
||||
# Returns the date of the first day on or after the given date that is a working day
|
||||
def next_working_date(date)
|
||||
cwday = date.cwday
|
||||
days = 0
|
||||
while non_working_week_days.include?(((cwday + days - 1) % 7) + 1)
|
||||
days += 1
|
||||
end
|
||||
date + days
|
||||
end
|
||||
|
||||
# Returns the index of non working week days (1=monday, 7=sunday)
|
||||
def non_working_week_days
|
||||
@non_working_week_days ||= begin
|
||||
days = Setting.non_working_week_days
|
||||
if days.is_a?(Array) && days.size < 7
|
||||
days.map(&:to_i)
|
||||
else
|
||||
[]
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
|
@ -493,6 +493,7 @@ html>body .tabular p {overflow:hidden;}
|
|||
}
|
||||
|
||||
.tabular label.inline{
|
||||
font-weight: normal;
|
||||
float:none;
|
||||
margin-left: 5px !important;
|
||||
width: auto;
|
||||
|
|
|
@ -346,7 +346,7 @@ class IssueNestedSetTest < ActiveSupport::TestCase
|
|||
c1 = Issue.generate!(:start_date => '2010-05-12', :due_date => '2010-05-18', :parent_issue_id => parent.id)
|
||||
c2 = Issue.generate!(:start_date => '2010-06-03', :due_date => '2010-06-10', :parent_issue_id => parent.id)
|
||||
parent.reload
|
||||
parent.reschedule_after(Date.parse('2010-06-02'))
|
||||
parent.reschedule_on!(Date.parse('2010-06-02'))
|
||||
c1.reload
|
||||
assert_equal [Date.parse('2010-06-02'), Date.parse('2010-06-08')], [c1.start_date, c1.due_date]
|
||||
c2.reload
|
||||
|
|
|
@ -1300,22 +1300,81 @@ class IssueTest < ActiveSupport::TestCase
|
|||
assert !closed_statuses.empty?
|
||||
end
|
||||
|
||||
def test_reschedule_an_issue_without_dates
|
||||
with_settings :non_working_week_days => [] do
|
||||
issue = Issue.new(:start_date => nil, :due_date => nil)
|
||||
issue.reschedule_on '2012-10-09'.to_date
|
||||
assert_equal '2012-10-09'.to_date, issue.start_date
|
||||
assert_equal '2012-10-09'.to_date, issue.due_date
|
||||
end
|
||||
|
||||
with_settings :non_working_week_days => %w(6 7) do
|
||||
issue = Issue.new(:start_date => nil, :due_date => nil)
|
||||
issue.reschedule_on '2012-10-09'.to_date
|
||||
assert_equal '2012-10-09'.to_date, issue.start_date
|
||||
assert_equal '2012-10-09'.to_date, issue.due_date
|
||||
|
||||
issue = Issue.new(:start_date => nil, :due_date => nil)
|
||||
issue.reschedule_on '2012-10-13'.to_date
|
||||
assert_equal '2012-10-15'.to_date, issue.start_date
|
||||
assert_equal '2012-10-15'.to_date, issue.due_date
|
||||
end
|
||||
end
|
||||
|
||||
def test_reschedule_an_issue_with_start_date
|
||||
with_settings :non_working_week_days => [] do
|
||||
issue = Issue.new(:start_date => '2012-10-09', :due_date => nil)
|
||||
issue.reschedule_on '2012-10-13'.to_date
|
||||
assert_equal '2012-10-13'.to_date, issue.start_date
|
||||
assert_equal '2012-10-13'.to_date, issue.due_date
|
||||
end
|
||||
|
||||
with_settings :non_working_week_days => %w(6 7) do
|
||||
issue = Issue.new(:start_date => '2012-10-09', :due_date => nil)
|
||||
issue.reschedule_on '2012-10-11'.to_date
|
||||
assert_equal '2012-10-11'.to_date, issue.start_date
|
||||
assert_equal '2012-10-11'.to_date, issue.due_date
|
||||
|
||||
issue = Issue.new(:start_date => '2012-10-09', :due_date => nil)
|
||||
issue.reschedule_on '2012-10-13'.to_date
|
||||
assert_equal '2012-10-15'.to_date, issue.start_date
|
||||
assert_equal '2012-10-15'.to_date, issue.due_date
|
||||
end
|
||||
end
|
||||
|
||||
def test_reschedule_an_issue_with_start_and_due_dates
|
||||
with_settings :non_working_week_days => [] do
|
||||
issue = Issue.new(:start_date => '2012-10-09', :due_date => '2012-10-15')
|
||||
issue.reschedule_on '2012-10-13'.to_date
|
||||
assert_equal '2012-10-13'.to_date, issue.start_date
|
||||
assert_equal '2012-10-19'.to_date, issue.due_date
|
||||
end
|
||||
|
||||
with_settings :non_working_week_days => %w(6 7) do
|
||||
issue = Issue.new(:start_date => '2012-10-09', :due_date => '2012-10-19') # 8 working days
|
||||
issue.reschedule_on '2012-10-11'.to_date
|
||||
assert_equal '2012-10-11'.to_date, issue.start_date
|
||||
assert_equal '2012-10-23'.to_date, issue.due_date
|
||||
|
||||
issue = Issue.new(:start_date => '2012-10-09', :due_date => '2012-10-19')
|
||||
issue.reschedule_on '2012-10-13'.to_date
|
||||
assert_equal '2012-10-15'.to_date, issue.start_date
|
||||
assert_equal '2012-10-25'.to_date, issue.due_date
|
||||
end
|
||||
end
|
||||
|
||||
def test_rescheduling_an_issue_should_reschedule_following_issue
|
||||
issue1 = Issue.create!(:project_id => 1, :tracker_id => 1,
|
||||
:author_id => 1, :status_id => 1,
|
||||
:subject => '-',
|
||||
:start_date => Date.today, :due_date => Date.today + 2)
|
||||
issue2 = Issue.create!(:project_id => 1, :tracker_id => 1,
|
||||
:author_id => 1, :status_id => 1,
|
||||
:subject => '-',
|
||||
:start_date => Date.today, :due_date => Date.today + 2)
|
||||
issue1 = Issue.generate!(:start_date => '2012-10-15', :due_date => '2012-10-17')
|
||||
issue2 = Issue.generate!(:start_date => '2012-10-15', :due_date => '2012-10-17')
|
||||
IssueRelation.create!(:issue_from => issue1, :issue_to => issue2,
|
||||
:relation_type => IssueRelation::TYPE_PRECEDES)
|
||||
assert_equal issue1.due_date + 1, issue2.reload.start_date
|
||||
assert_equal Date.parse('2012-10-18'), issue2.reload.start_date
|
||||
|
||||
issue1.due_date = Date.today + 5
|
||||
issue1.due_date = '2012-10-23'
|
||||
issue1.save!
|
||||
assert_equal issue1.due_date + 1, issue2.reload.start_date
|
||||
issue2.reload
|
||||
assert_equal Date.parse('2012-10-24'), issue2.start_date
|
||||
assert_equal Date.parse('2012-10-26'), issue2.due_date
|
||||
end
|
||||
|
||||
def test_rescheduling_a_stale_issue_should_not_raise_an_error
|
||||
|
@ -1326,7 +1385,7 @@ class IssueTest < ActiveSupport::TestCase
|
|||
|
||||
date = 10.days.from_now.to_date
|
||||
assert_nothing_raised do
|
||||
stale.reschedule_after(date)
|
||||
stale.reschedule_on!(date)
|
||||
end
|
||||
assert_equal date, stale.reload.start_date
|
||||
end
|
||||
|
|
|
@ -0,0 +1,76 @@
|
|||
# Redmine - project management software
|
||||
# Copyright (C) 2006-2012 Jean-Philippe Lang
|
||||
#
|
||||
# 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.
|
||||
|
||||
require File.expand_path('../../../../../test_helper', __FILE__)
|
||||
|
||||
class Redmine::Utils::DateCalculationTest < ActiveSupport::TestCase
|
||||
include Redmine::Utils::DateCalculation
|
||||
|
||||
def test_working_days_without_non_working_week_days
|
||||
with_settings :non_working_week_days => [] do
|
||||
assert_working_days 18, '2012-10-09', '2012-10-27'
|
||||
assert_working_days 6, '2012-10-09', '2012-10-15'
|
||||
assert_working_days 5, '2012-10-09', '2012-10-14'
|
||||
assert_working_days 3, '2012-10-09', '2012-10-12'
|
||||
assert_working_days 3, '2012-10-14', '2012-10-17'
|
||||
assert_working_days 16, '2012-10-14', '2012-10-30'
|
||||
end
|
||||
end
|
||||
|
||||
def test_working_days_with_non_working_week_days
|
||||
with_settings :non_working_week_days => %w(6 7) do
|
||||
assert_working_days 14, '2012-10-09', '2012-10-27'
|
||||
assert_working_days 4, '2012-10-09', '2012-10-15'
|
||||
assert_working_days 4, '2012-10-09', '2012-10-14'
|
||||
assert_working_days 3, '2012-10-09', '2012-10-12'
|
||||
assert_working_days 8, '2012-10-09', '2012-10-19'
|
||||
assert_working_days 8, '2012-10-11', '2012-10-23'
|
||||
assert_working_days 2, '2012-10-14', '2012-10-17'
|
||||
assert_working_days 11, '2012-10-14', '2012-10-30'
|
||||
end
|
||||
end
|
||||
|
||||
def test_add_working_days_without_non_working_week_days
|
||||
with_settings :non_working_week_days => [] do
|
||||
assert_add_working_days '2012-10-10', '2012-10-10', 0
|
||||
assert_add_working_days '2012-10-11', '2012-10-10', 1
|
||||
assert_add_working_days '2012-10-12', '2012-10-10', 2
|
||||
assert_add_working_days '2012-10-13', '2012-10-10', 3
|
||||
assert_add_working_days '2012-10-25', '2012-10-10', 15
|
||||
end
|
||||
end
|
||||
|
||||
def test_add_working_days_with_non_working_week_days
|
||||
with_settings :non_working_week_days => %w(6 7) do
|
||||
assert_add_working_days '2012-10-10', '2012-10-10', 0
|
||||
assert_add_working_days '2012-10-11', '2012-10-10', 1
|
||||
assert_add_working_days '2012-10-12', '2012-10-10', 2
|
||||
assert_add_working_days '2012-10-15', '2012-10-10', 3
|
||||
assert_add_working_days '2012-10-31', '2012-10-10', 15
|
||||
assert_add_working_days '2012-10-19', '2012-10-09', 8
|
||||
assert_add_working_days '2012-10-23', '2012-10-11', 8
|
||||
end
|
||||
end
|
||||
|
||||
def assert_working_days(expected_days, from, to)
|
||||
assert_equal expected_days, working_days(from.to_date, to.to_date)
|
||||
end
|
||||
|
||||
def assert_add_working_days(expected_date, from, working_days)
|
||||
assert_equal expected_date.to_date, add_working_days(from.to_date, working_days)
|
||||
end
|
||||
end
|
Loading…
Reference in New Issue