OpenProject is the leading open source project management software.
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.
openproject/app/controllers/cost_objects_controller.rb

270 lines
8.5 KiB

#-- copyright
# OpenProject Costs Plugin
#
# Copyright (C) 2009 - 2014 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.
#
# 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.
#++
class CostObjectsController < ApplicationController
before_filter :find_cost_object, only: [:show, :edit, :update, :copy]
before_filter :find_cost_objects, only: :destroy
before_filter :find_project, only: [
:new, :create,
:update_material_budget_item, :update_labor_budget_item
]
before_filter :find_optional_project, only: :index
before_filter :authorize_global, only: :index
before_filter :authorize, except: [
# unrestricted actions
:index,
:update_material_budget_item, :update_labor_budget_item
]
helper :sort
include SortHelper
helper :projects
include ProjectsHelper
helper :attachments
include AttachmentsHelper
helper :costlog
include CostlogHelper
helper :cost_objects
include CostObjectsHelper
include WorkPackage::PdfExporter
include PaginationHelper
menu_item :new_budget, only: [:new]
menu_item :show_all, only: [:index]
def index
respond_to do |format|
format.html do end
format.csv { limit = Setting.work_packages_export_limit.to_i }
end
sort_columns = { 'id' => "#{CostObject.table_name}.id",
'subject' => "#{CostObject.table_name}.subject",
'fixed_date' => "#{CostObject.table_name}.fixed_date"
}
sort_init 'id', 'desc'
sort_update sort_columns
allowed_condition = Project.allowed_to(User.current,
:view_cost_objects,
project: @project)
@cost_objects = CostObject.order(sort_clause)
.includes(:project, :author)
.merge(allowed_condition)
.page(page_param)
.per_page(per_page_param)
respond_to do |format|
format.html do render action: 'index', layout: !request.xhr? end
format.csv { send_data(cost_objects_to_csv(@cost_objects), type: 'text/csv; header=present', filename: 'export.csv') }
end
end
def show
@edit_allowed = User.current.allowed_to?(:edit_cost_objects, @project)
respond_to do |format|
format.html { render action: 'show', layout: !request.xhr? }
end
end
def new
# FIXME: I forcibly create a VariableCostObject for now. Following Ticket #5360
@cost_object ||= VariableCostObject.new
@cost_object.project_id = @project.id
@cost_object.fixed_date ||= Date.today
render layout: !request.xhr?
end
def copy
source = CostObject.find(params[:id].to_i)
if source
@cost_object = create_cost_object(source.kind)
@cost_object.copy_from(source)
end
# FIXME: I forcibly create a VariableCostObject for now. Following Ticket #5360
@cost_object ||= VariableCostObject.new
@cost_object.fixed_date ||= Date.today
render action: :new, layout: !request.xhr?
end
def create
if params[:cost_object]
@cost_object = create_cost_object(params[:cost_object].delete(:kind))
end
# FIXME: I forcibly create a VariableCostObject for now. Following Ticket #5360
@cost_object ||= VariableCostObject.new
@cost_object.project_id = @project.id
# fixed_date must be set before material_budget_items and labor_budget_items
if params[:cost_object] && params[:cost_object][:fixed_date]
@cost_object.fixed_date = params[:cost_object].delete(:fixed_date)
else
@cost_object.fixed_date = Date.today
end
@cost_object.attributes = permitted_params.cost_object
if @cost_object.save
Attachment.attach_files(@cost_object, params[:attachments])
render_attachment_warning_if_needed(@cost_object)
flash[:notice] = l(:notice_successful_create)
redirect_to(params[:continue] ? { action: 'new' } :
{ action: 'show', id: @cost_object })
return
else
render action: 'new', layout: !request.xhr?
end
end
def edit
# TODO: This method used to be responsible for both edit and update
# Please remove code where necessary
# check whether this method is needed at all
@cost_object.attributes = permitted_params.cost_object if params[:cost_object]
end
def update
# TODO: This was simply copied over from edit in order to have
# something as a starting point for separating the two
# Please go ahead and start removing code where necessary
# TODO: use better way to prevent mass assignment errors
params[:cost_object].delete(:kind)
@cost_object.attributes = permitted_params.cost_object if params[:cost_object]
if @cost_object.save
Attachment.attach_files(@cost_object, params[:attachments])
render_attachment_warning_if_needed(@cost_object)
flash[:notice] = l(:notice_successful_update)
redirect_to(params[:back_to] || { action: 'show', id: @cost_object })
else
render action: 'edit'
end
rescue ActiveRecord::StaleObjectError
# Optimistic locking exception
flash.now[:error] = l(:notice_locking_conflict)
end
def destroy
@cost_objects.each(&:destroy)
flash[:notice] = l(:notice_successful_delete)
redirect_to action: 'index', project_id: @project
end
def update_material_budget_item
element_id = params[:element_id] if params.has_key? :element_id
cost_type = CostType.find(params[:cost_type_id]) if params.has_key? :cost_type_id
units = BigDecimal.new(Rate.clean_currency(params[:units]))
costs = (units * cost_type.rate_at(params[:fixed_date]).rate rescue 0.0)
if request.xhr?
render :update do |page|
if User.current.allowed_to? :view_cost_rates, @project
page.replace_html "#{element_id}_costs", number_to_currency(costs)
end
page.replace_html "#{element_id}_unit_name", h(units == 1.0 ? cost_type.unit : cost_type.unit_plural)
end
end
rescue ActiveRecord::RecordNotFound
render_404
end
def update_labor_budget_item
element_id = params[:element_id] if params.has_key? :element_id
user = User.find(params[:user_id])
hours = params[:hours].to_hours
costs = hours * user.rate_at(params[:fixed_date], @project).rate rescue 0.0
if request.xhr?
render :update do |page|
if User.current.allowed_to?(:view_hourly_rates, @project, for: user)
page.replace_html "#{element_id}_costs", number_to_currency(costs)
end
end
end
rescue ActiveRecord::RecordNotFound
render :update do |page|
page.replace_html "#{element_id}_costs", number_to_currency(0.0)
end
end
private
def create_cost_object(kind)
case kind
when FixedCostObject.name
FixedCostObject.new
when VariableCostObject.name
VariableCostObject.new
else
CostObject.new
end
end
def find_cost_object
# This function comes directly from issues_controller.rb (Redmine 0.8.4)
@cost_object = CostObject.includes(:project, :author).find_by(id: params[:id])
@project = @cost_object.project if @cost_object
rescue ActiveRecord::RecordNotFound
render_404
end
def find_cost_objects
# This function comes directly from issues_controller.rb (Redmine 0.8.4)
@cost_objects = CostObject.where(id: params[:id] || params[:ids])
raise ActiveRecord::RecordNotFound if @cost_objects.empty?
projects = @cost_objects.map(&:project).compact.uniq
if projects.size == 1
@project = projects.first
else
# TODO: let users bulk edit/move/destroy cost_objects from different projects
render_error 'Can not bulk edit/move/destroy cost objects from different projects' and return false
end
rescue ActiveRecord::RecordNotFound
render_404
end
def find_project
@project = Project.find(params[:project_id])
rescue ActiveRecord::RecordNotFound
render_404
end
def find_optional_project
@project = Project.find(params[:project_id]) unless params[:project_id].blank?
rescue ActiveRecord::RecordNotFound
render_404
end
end