issue_status.rb
113 lines
| 4.3 KiB
| text/x-ruby
|
RubyLexer
|
r6384 | # Redmine - project management software | ||
|
r13490 | # Copyright (C) 2006-2015 Jean-Philippe Lang | ||
|
r330 | # | ||
# 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. | ||||
|
r6384 | # | ||
|
r330 | # 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. | ||||
|
r6384 | # | ||
|
r330 | # 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. | ||||
class IssueStatus < ActiveRecord::Base | ||||
|
r6384 | before_destroy :check_integrity | ||
|
r9794 | has_many :workflows, :class_name => 'WorkflowTransition', :foreign_key => "old_status_id" | ||
|
r13134 | has_many :workflow_transitions_as_new_status, :class_name => 'WorkflowTransition', :foreign_key => "new_status_id" | ||
|
r330 | acts_as_list | ||
|
r6384 | |||
|
r13178 | after_update :handle_is_closed_change | ||
|
r9794 | before_destroy :delete_workflow_rules | ||
|
r330 | |||
validates_presence_of :name | ||||
validates_uniqueness_of :name | ||||
|
r590 | validates_length_of :name, :maximum => 30 | ||
|
r3043 | validates_inclusion_of :default_done_ratio, :in => 0..100, :allow_nil => true | ||
|
r13100 | attr_protected :id | ||
|
r6384 | |||
|
r13144 | scope :sorted, lambda { order(:position) } | ||
|
r10660 | scope :named, lambda {|arg| where("LOWER(#{table_name}.name) = LOWER(?)", arg.to_s.strip)} | ||
|
r330 | |||
|
r3037 | # Update all the +Issues+ setting their done_ratio to the value of their +IssueStatus+ | ||
def self.update_issue_done_ratios | ||||
if Issue.use_status_for_done_ratio? | ||||
|
r12450 | IssueStatus.where("default_done_ratio >= 0").each do |status| | ||
|
r12236 | Issue.where({:status_id => status.id}).update_all({:done_ratio => status.default_done_ratio}) | ||
|
r3037 | end | ||
end | ||||
return Issue.use_status_for_done_ratio? | ||||
end | ||||
|
r330 | |||
# Returns an array of all statuses the given role can switch to | ||||
|
r4775 | def new_statuses_allowed_to(roles, tracker, author=false, assignee=false) | ||
|
r14076 | self.class.new_statuses_allowed(self, roles, tracker, author, assignee) | ||
|
r411 | end | ||
|
r14076 | alias :find_new_statuses_allowed_to :new_statuses_allowed_to | ||
|
r6384 | |||
|
r14076 | def self.new_statuses_allowed(status, roles, tracker, author=false, assignee=false) | ||
|
r6180 | if roles.present? && tracker | ||
|
r14076 | status_id = status.try(:id) || 0 | ||
|
r13134 | scope = IssueStatus. | ||
joins(:workflow_transitions_as_new_status). | ||||
|
r14076 | where(:workflows => {:old_status_id => status_id, :role_id => roles.map(&:id), :tracker_id => tracker.id}) | ||
|
r13134 | |||
unless author && assignee | ||||
if author || assignee | ||||
scope = scope.where("author = ? OR assignee = ?", author, assignee) | ||||
else | ||||
scope = scope.where("author = ? AND assignee = ?", false, false) | ||||
end | ||||
end | ||||
scope.uniq.to_a.sort | ||||
|
r2627 | else | ||
[] | ||||
end | ||||
|
r330 | end | ||
|
r771 | |||
|
r1030 | def <=>(status) | ||
position <=> status.position | ||||
end | ||||
|
r6384 | |||
|
r771 | def to_s; name end | ||
|
r9798 | private | ||
|
r13178 | # Updates issues closed_on attribute when an existing status is set as closed. | ||
def handle_is_closed_change | ||||
if is_closed_changed? && is_closed == true | ||||
# First we update issues that have a journal for when the current status was set, | ||||
# a subselect is used to update all issues with a single query | ||||
subselect = "SELECT MAX(j.created_on) FROM #{Journal.table_name} j" + | ||||
" JOIN #{JournalDetail.table_name} d ON d.journal_id = j.id" + | ||||
" WHERE j.journalized_type = 'Issue' AND j.journalized_id = #{Issue.table_name}.id" + | ||||
" AND d.property = 'attr' AND d.prop_key = 'status_id' AND d.value = :status_id" | ||||
|
r13259 | Issue.where(:status_id => id, :closed_on => nil). | ||
update_all(["closed_on = (#{subselect})", {:status_id => id.to_s}]) | ||||
|
r13178 | |||
# Then we update issues that don't have a journal which means the | ||||
# current status was set on creation | ||||
Issue.where(:status_id => id, :closed_on => nil).update_all("closed_on = created_on") | ||||
end | ||||
end | ||||
|
r330 | def check_integrity | ||
|
r13153 | if Issue.where(:status_id => id).any? | ||
raise "This status is used by some issues" | ||||
elsif Tracker.where(:default_status_id => id).any? | ||||
raise "This status is used as the default status by some trackers" | ||||
end | ||||
|
r2 | end | ||
|
r6384 | |||
|
r3767 | # Deletes associated workflows | ||
|
r9794 | def delete_workflow_rules | ||
WorkflowRule.delete_all(["old_status_id = :id OR new_status_id = :id", {:id => id}]) | ||||
|
r3767 | end | ||
|
r2 | end | ||