Files
redmine-budget-plugin/app/models/deliverable.rb

210 lines
5.6 KiB
Ruby

# A Deliverable is an item that is created as part of the project. These items
# contain a collection of issues.
class Deliverable < ActiveRecord::Base
unloadable
validates_presence_of :subject
belongs_to :project
has_many :issues, :dependent => :nullify
# Assign all the issues with +version_id+ to this Deliverable
def assign_issues_by_version(version_id)
version = Version.find_by_id(version_id)
return 0 if version.nil? || version.fixed_issues.blank?
version.fixed_issues.each do |issue|
issue.update_attribute(:deliverable_id, self.id)
end
return version.fixed_issues.size
end
# Change the Deliverable type to another type. Valid types are
#
# * FixedDeliverable
# * HourlyDeliverable
def change_type(to)
if [FixedDeliverable.name, HourlyDeliverable.name].include?(to)
self.type = to
self.save!
return Deliverable.find(self.id)
else
return self
end
end
# Adjusted score to show the status of the Deliverable. Will range from 100
# (everything done with no money spent) to -100 (nothing done, all the money spent)
def score
return self.progress - self.budget_ratio
end
# Amount spent. Virtual accessor that is overriden by subclasses.
def spent
0
end
# Percentage of the deliverable that is compelte based on the progress of the
# assigned issues.
def progress
return 0 unless self.issues.size > 0
total ||= self.issues.collect(&:estimated_hours).compact.sum || 0
return 0 unless total > 0
balance = 0.0
self.issues.each do |issue|
if use_issue_status_for_done_ratios?
balance += issue.status.default_done_ratio * issue.estimated_hours unless issue.estimated_hours.nil?
else
balance += issue.done_ratio * issue.estimated_hours unless issue.estimated_hours.nil?
end
end
return (balance / total).round
end
# Amount of the budget spent. Expressed as as a percentage whole number
def budget_ratio
return 0.0 if self.budget.nil? || self.budget == 0.0
return ((self.spent / self.budget) * 100).round
end
def overhead
return read_attribute(:overhead) unless read_attribute(:overhead).nil?
return ((read_attribute(:overhead_percent).to_f / 100.0) * self.labor_budget) unless read_attribute(:overhead_percent).nil?
return 0
end
# Setter for the overhead to take an Dollar amount or a %.
def overhead=(v)
return if v.nil?
if v.match(/%/)
# Clear amount since this is a %
write_attribute(:overhead, nil)
write_attribute(:overhead_percent, v.gsub(/[% ]/,''))
else
# Clear % since this is a dollar amount
write_attribute(:overhead_percent, nil)
# Take out $, commas, and spaces
write_attribute(:overhead, v.gsub(/[$, ]/,''))
end
end
# Setter for the materials to take an Dollar amount or a %.
def materials=(v)
return if v.nil?
if v.match(/%/)
# Clear amount since this is a %
write_attribute(:materials, nil)
write_attribute(:materials_percent, v.gsub(/[% ]/,''))
else
# Clear % since this is a dollar amount
write_attribute(:materials_percent, nil)
# Take out $, commas, and spaces
write_attribute(:materials, v.gsub(/[$, ]/,''))
end
end
# Setter for the profit to take an Dollar amount or a %.
def profit=(v)
if v.match(/%/)
# Clear amount since this is a %
write_attribute(:profit, nil)
write_attribute(:profit_percent, v.gsub(/[% ]/,''))
else
# Clear % since this is a dollar amount
write_attribute(:profit_percent, nil)
# Take out $, commas, and spaces
write_attribute(:profit, v.gsub(/[$, ]/,''))
end
end
# Wrap the budget getter so it returns 0 if budget is nil
def budget
raw_budget = read_attribute(:budget)
unless raw_budget.nil?
return raw_budget
else
return 0
end
end
# Amount of the budget remaining to be spent
def budget_remaining
return self.budget - self.spent
end
alias :left :budget_remaining
# Number of hours used.
def hours_used
return 0 unless self.issues.size > 0
return self.issues.collect(&:time_entries).flatten.collect(&:hours).sum
end
# Grouping of members and how much they spent
def members_spent
return MemberSpent.find_all_by_deliverable(self)
end
# Amount spent on members.
def spent_by_members
return self.members_spent.collect(&:spent).sum
end
# Amount spent over the total budget
def overruns
if self.left >= 0
return 0
else
return self.left * -1
end
end
# Budget of labor, without counting profit or overheads. Virtual accessor that is overriden by subclasses.
def labor_budget
0
end
# Helper method to return an Hash of the trackers and number of issues assigned to each tracker.
def issues_with_trackers
trackers = self.project.trackers
return { } if trackers.empty?
tracker_map = { }
trackers.each do |tracker|
tracker_map[tracker.name] = Issue.find_all_by_tracker_id_and_project_id_and_deliverable_id(tracker.id, self.project.id, self.id).size
end
return tracker_map
end
# Returns true if the deliverable can be edited by user, otherwise false
def editable_by?(user)
(user == user && user.allowed_to?(:manage_budget, project))
end
def fixed?
return self.class == FixedDeliverable
end
def hourly?
return self.class == HourlyDeliverable
end
def to_s
self.subject
end
private
def use_issue_status_for_done_ratios?
return defined?(Setting.issue_status_for_done_ratio?) && Setting.issue_status_for_done_ratio?
end
end