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/modules/costs/app/controllers/cost_objects_controller.rb

282 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_action :find_cost_object, only: [:show, :edit, :update, :copy]
before_action :find_cost_objects, only: :destroy
before_action :find_project, only: [
:new, :create,
:update_material_budget_item, :update_labor_budget_item
]
before_action :find_optional_project, only: :index
before_action :authorize_global, only: :index
before_action :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 PaginationHelper
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
@cost_objects = CostObject
.visible(current_user)
.order(sort_clause)
.includes(:author)
.where(project_id: @project.id)
.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
@cost_object.attach_files(permitted_params.attachments.to_h)
if @cost_object.save
render_attachment_warning_if_needed(@cost_object)
flash[:notice] = t(: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 params[:cost_object][:existing_material_budget_item_attributes].nil?
@cost_object.existing_material_budget_item_attributes = ({})
end
if params[:cost_object][:existing_labor_budget_item_attributes].nil?
@cost_object.existing_labor_budget_item_attributes = ({})
end
@cost_object.attach_files(permitted_params.attachments.to_h)
if @cost_object.save
render_attachment_warning_if_needed(@cost_object)
flash[:notice] = t(: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] = t(:notice_locking_conflict)
end
def destroy
@cost_objects.each(&:destroy)
flash[:notice] = t(:notice_successful_delete)
redirect_to action: 'index', project_id: @project
end
def update_material_budget_item
@element_id = params[:element_id]
cost_type = CostType.where(id: params[:cost_type_id]).first
if cost_type && params[:units].present?
volume = BigDecimal(Rate.clean_currency(params[:units])) rescue 0.0
@costs = (volume * cost_type.rate_at(params[:fixed_date]).rate rescue 0.0)
@unit = volume == 1.0 ? cost_type.unit : cost_type.unit_plural
else
@costs = 0.0
@unit = cost_type.try(:unit_plural) || ''
end
response = { "#{@element_id}_unit_name" => h(@unit) }
if current_user.allowed_to?(:view_cost_rates, @project)
response["#{@element_id}_costs"] = number_to_currency(@costs)
end
respond_to do |format|
format.json do
render json: response
end
end
end
def update_labor_budget_item
@element_id = params[:element_id]
user = User.where(id: params[:user_id]).first
if user && params[:hours]
hours = params[:hours].to_s.to_hours
@costs = hours * user.rate_at(params[:fixed_date], @project).rate rescue 0.0
else
@costs = 0.0
end
response = { "#{@element_id}_unit_name" => h(@unit) }
if current_user.allowed_to?(:view_hourly_rates, @project)
response["#{@element_id}_costs"] = number_to_currency(@costs)
end
respond_to do |format|
format.json do
render json: response
end
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