projects_controller.rb
261 lines
| 8.9 KiB
| text/x-ruby
|
RubyLexer
|
r2861 | # Redmine - project management software | ||
|
r9453 | # Copyright (C) 2006-2012 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. | ||||
|
r6773 | # | ||
|
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. | ||||
|
r6773 | # | ||
|
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 ProjectsController < ApplicationController | ||||
|
r1062 | menu_item :overview | ||
menu_item :roadmap, :only => :roadmap | ||||
menu_item :settings, :only => :settings | ||||
|
r6773 | |||
|
r3955 | before_filter :find_project, :except => [ :index, :list, :new, :create, :copy ] | ||
before_filter :authorize, :except => [ :index, :list, :new, :create, :copy, :archive, :unarchive, :destroy] | ||||
before_filter :authorize_global, :only => [:new, :create] | ||||
|
r2651 | before_filter :require_admin, :only => [ :copy, :archive, :unarchive, :destroy ] | ||
|
r6077 | accept_rss_auth :index | ||
accept_api_auth :index, :show, :create, :update, :destroy | ||||
|
r3956 | |||
after_filter :only => [:create, :edit, :update, :archive, :unarchive, :destroy] do |controller| | ||||
|
r2317 | if controller.request.post? | ||
|
r8072 | controller.send :expire_action, :controller => 'welcome', :action => 'robots' | ||
|
r2317 | end | ||
end | ||||
|
r3956 | |||
|
r330 | helper :sort | ||
include SortHelper | ||||
helper :custom_fields | ||||
|
r6773 | include CustomFieldsHelper | ||
|
r783 | helper :issues | ||
|
r330 | helper :queries | ||
include QueriesHelper | ||||
|
r556 | helper :repositories | ||
include RepositoriesHelper | ||||
|
r660 | include ProjectsHelper | ||
|
r6773 | |||
|
r718 | # Lists visible projects | ||
|
r1450 | def index | ||
|
r1451 | respond_to do |format| | ||
|
r6773 | format.html { | ||
|
r9700 | scope = Project | ||
unless params[:closed] | ||||
scope = scope.active | ||||
end | ||||
@projects = scope.visible.order('lft').all | ||||
|
r1451 | } | ||
|
r4342 | format.api { | ||
|
r4458 | @offset, @limit = api_offset_and_limit | ||
@project_count = Project.visible.count | ||||
@projects = Project.visible.all(:offset => @offset, :limit => @limit, :order => 'lft') | ||||
|
r3199 | } | ||
|
r1451 | format.atom { | ||
|
r2302 | projects = Project.visible.find(:all, :order => 'created_on DESC', | ||
:limit => Setting.feeds_limit.to_i) | ||||
render_feed(projects, :title => "#{Setting.app_title}: #{l(:label_project_latest)}") | ||||
|
r1451 | } | ||
end | ||||
|
r330 | end | ||
|
r6773 | |||
|
r3955 | def new | ||
|
r1578 | @issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position") | ||
|
r9531 | @trackers = Tracker.sorted.all | ||
|
r9015 | @project = Project.new | ||
@project.safe_attributes = params[:project] | ||||
|
r3953 | end | ||
def create | ||||
@issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position") | ||||
|
r9531 | @trackers = Tracker.sorted.all | ||
|
r4378 | @project = Project.new | ||
@project.safe_attributes = params[:project] | ||||
|
r3953 | |||
if validate_parent_id && @project.save | ||||
@project.set_allowed_parent!(params[:project]['parent_id']) if params[:project].has_key?('parent_id') | ||||
# Add current user as a project member if he is not admin | ||||
unless User.current.admin? | ||||
r = Role.givable.find_by_id(Setting.new_project_user_role_id.to_i) || Role.givable.first | ||||
m = Member.new(:user => User.current, :roles => [r]) | ||||
@project.members << m | ||||
end | ||||
respond_to do |format| | ||||
|
r6773 | format.html { | ||
|
r3953 | flash[:notice] = l(:notice_successful_create) | ||
|
r6183 | redirect_to(params[:continue] ? | ||
{:controller => 'projects', :action => 'new', :project => {:parent_id => @project.parent_id}.reject {|k,v| v.nil?}} : | ||||
{:controller => 'projects', :action => 'settings', :id => @project} | ||||
) | ||||
|
r3953 | } | ||
|
r4352 | format.api { render :action => 'show', :status => :created, :location => url_for(:controller => 'projects', :action => 'show', :id => @project.id) } | ||
|
r3953 | end | ||
|
r330 | else | ||
|
r3953 | respond_to do |format| | ||
|
r3955 | format.html { render :action => 'new' } | ||
|
r4342 | format.api { render_validation_errors(@project) } | ||
|
r2651 | end | ||
|
r3953 | end | ||
|
r6773 | |||
|
r330 | end | ||
|
r6773 | |||
|
r2608 | def copy | ||
@issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position") | ||||
|
r9531 | @trackers = Tracker.sorted.all | ||
|
r2861 | @source_project = Project.find(params[:id]) | ||
|
r2608 | if request.get? | ||
|
r2861 | @project = Project.copy_from(@source_project) | ||
|
r10676 | @project.identifier = Project.next_identifier if Setting.sequential_project_identifiers? | ||
|
r2608 | else | ||
|
r3494 | Mailer.with_deliveries(params[:notifications] == '1') do | ||
|
r4378 | @project = Project.new | ||
@project.safe_attributes = params[:project] | ||||
|
r3494 | if validate_parent_id && @project.copy(@source_project, :only => params[:only]) | ||
@project.set_allowed_parent!(params[:project]['parent_id']) if params[:project].has_key?('parent_id') | ||||
flash[:notice] = l(:notice_successful_create) | ||||
|
r4496 | redirect_to :controller => 'projects', :action => 'settings', :id => @project | ||
|
r3494 | elsif !@project.new_record? | ||
# Project was created | ||||
# But some objects were not copied due to validation failures | ||||
# (eg. issues from disabled trackers) | ||||
# TODO: inform about that | ||||
|
r4496 | redirect_to :controller => 'projects', :action => 'settings', :id => @project | ||
|
r3494 | end | ||
|
r3128 | end | ||
|
r2878 | end | ||
rescue ActiveRecord::RecordNotFound | ||||
|
r10676 | # source_project not found | ||
render_404 | ||||
|
r2608 | end | ||
|
r330 | |||
# Show @project | ||||
def show | ||||
|
r2208 | if params[:jump] | ||
# try to redirect to the requested menu item | ||||
redirect_to_project_menu_item(@project, params[:jump]) && return | ||||
end | ||||
|
r6773 | |||
|
r2635 | @users_by_role = @project.users_by_role | ||
|
r5174 | @subprojects = @project.children.visible.all | ||
|
r334 | @news = @project.news.find(:all, :limit => 5, :include => [ :author, :project ], :order => "#{News.table_name}.created_on DESC") | ||
|
r1164 | @trackers = @project.rolled_up_trackers | ||
|
r6773 | |||
|
r1283 | cond = @project.project_condition(Setting.display_subprojects_issues?) | ||
|
r6773 | |||
|
r9722 | @open_issues_by_tracker = Issue.visible.open.where(cond).count(:group => :tracker) | ||
@total_issues_by_tracker = Issue.visible.where(cond).count(:group => :tracker) | ||||
|
r6773 | |||
|
r5029 | if User.current.allowed_to?(:view_time_entries, @project) | ||
@total_hours = TimeEntry.visible.sum(:hours, :include => :project, :conditions => cond).to_f | ||||
|
r1162 | end | ||
|
r6773 | |||
|
r663 | @key = User.current.rss_key | ||
|
r6773 | |||
|
r3199 | respond_to do |format| | ||
format.html | ||||
|
r4352 | format.api | ||
|
r3199 | end | ||
|
r5 | end | ||
|
r330 | |||
def settings | ||||
|
r1578 | @issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position") | ||
|
r330 | @issue_category ||= IssueCategory.new | ||
@member ||= @project.members.new | ||||
|
r9531 | @trackers = Tracker.sorted.all | ||
|
r714 | @wiki ||= @project.wiki | ||
|
r330 | end | ||
|
r6773 | |||
|
r330 | def edit | ||
|
r3956 | end | ||
def update | ||||
|
r4378 | @project.safe_attributes = params[:project] | ||
|
r3956 | if validate_parent_id && @project.save | ||
@project.set_allowed_parent!(params[:project]['parent_id']) if params[:project].has_key?('parent_id') | ||||
respond_to do |format| | ||||
|
r6773 | format.html { | ||
|
r3956 | flash[:notice] = l(:notice_successful_update) | ||
redirect_to :action => 'settings', :id => @project | ||||
} | ||||
|
r9792 | format.api { render_api_ok } | ||
|
r3956 | end | ||
|
r3199 | else | ||
|
r3956 | respond_to do |format| | ||
|
r6773 | format.html { | ||
|
r3956 | settings | ||
render :action => 'settings' | ||||
} | ||||
|
r4342 | format.api { render_validation_errors(@project) } | ||
|
r330 | end | ||
|
r5 | end | ||
|
r330 | end | ||
|
r4527 | |||
|
r714 | def modules | ||
|
r4527 | @project.enabled_module_names = params[:enabled_module_names] | ||
|
r3451 | flash[:notice] = l(:notice_successful_update) | ||
|
r714 | redirect_to :action => 'settings', :id => @project, :tab => 'modules' | ||
end | ||||
|
r330 | |||
|
r546 | def archive | ||
|
r3009 | if request.post? | ||
unless @project.archive | ||||
flash[:error] = l(:error_can_not_archive_project) | ||||
end | ||||
end | ||||
|
r2706 | redirect_to(url_for(:controller => 'admin', :action => 'projects', :status => params[:status])) | ||
|
r546 | end | ||
|
r6773 | |||
|
r546 | def unarchive | ||
@project.unarchive if request.post? && !@project.active? | ||||
|
r2706 | redirect_to(url_for(:controller => 'admin', :action => 'projects', :status => params[:status])) | ||
|
r546 | end | ||
|
r6773 | |||
|
r9700 | def close | ||
@project.close | ||||
redirect_to project_path(@project) | ||||
end | ||||
def reopen | ||||
@project.reopen | ||||
redirect_to project_path(@project) | ||||
end | ||||
|
r10 | # Delete @project | ||
|
r330 | def destroy | ||
|
r546 | @project_to_destroy = @project | ||
|
r8032 | if api_request? || params[:confirm] | ||
@project_to_destroy.destroy | ||||
respond_to do |format| | ||||
format.html { redirect_to :controller => 'admin', :action => 'projects' } | ||||
|
r9792 | format.api { render_api_ok } | ||
|
r3199 | end | ||
|
r330 | end | ||
|
r546 | # hide project in layout | ||
@project = nil | ||||
|
r330 | end | ||
|
r8826 | private | ||
|
r1213 | |||
|
r3124 | # Validates parent_id param according to user's permissions | ||
# TODO: move it to Project model in a validation that depends on User.current | ||||
def validate_parent_id | ||||
return true if User.current.admin? | ||||
parent_id = params[:project] && params[:project][:parent_id] | ||||
if parent_id || @project.new_record? | ||||
parent = parent_id.blank? ? nil : Project.find_by_id(parent_id.to_i) | ||||
unless @project.allowed_parents.include?(parent) | ||||
@project.errors.add :parent_id, :invalid | ||||
return false | ||||
end | ||||
end | ||||
true | ||||
end | ||||
|
r2 | end | ||