projects_controller.rb
270 lines
| 9.7 KiB
| text/x-ruby
|
RubyLexer
|
r2861 | # Redmine - project management software | ||
# Copyright (C) 2006-2009 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. | ||||
# | ||||
# 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 ProjectsController < ApplicationController | ||||
|
r1062 | menu_item :overview | ||
menu_item :roadmap, :only => :roadmap | ||||
menu_item :settings, :only => :settings | ||||
|
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 ] | ||
|
r4330 | accept_key_auth :index, :show, :create, :update, :destroy | ||
|
r3956 | |||
after_filter :only => [:create, :edit, :update, :archive, :unarchive, :destroy] do |controller| | ||||
|
r2317 | if controller.request.post? | ||
controller.send :expire_action, :controller => 'welcome', :action => 'robots.txt' | ||||
end | ||||
end | ||||
|
r3956 | |||
# TODO: convert to PUT only | ||||
verify :method => [:post, :put], :only => :update, :render => {:nothing => true, :status => :method_not_allowed } | ||||
|
r330 | helper :sort | ||
include SortHelper | ||||
helper :custom_fields | ||||
include CustomFieldsHelper | ||||
|
r783 | helper :issues | ||
|
r330 | helper :queries | ||
include QueriesHelper | ||||
|
r556 | helper :repositories | ||
include RepositoriesHelper | ||||
|
r660 | include ProjectsHelper | ||
|
r330 | |||
|
r718 | # Lists visible projects | ||
|
r1450 | def index | ||
|
r1451 | respond_to do |format| | ||
format.html { | ||||
|
r2302 | @projects = Project.visible.find(:all, :order => 'lft') | ||
|
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 | ||
|
r718 | |||
|
r3955 | def new | ||
|
r1578 | @issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position") | ||
|
r907 | @trackers = Tracker.all | ||
|
r330 | @project = Project.new(params[:project]) | ||
|
r3953 | end | ||
def create | ||||
@issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position") | ||||
@trackers = Tracker.all | ||||
|
r4378 | @project = Project.new | ||
@project.safe_attributes = params[:project] | ||||
|
r3953 | |||
|
r4346 | @project.enabled_module_names = params[:enabled_modules] if params[:enabled_modules] | ||
|
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| | ||||
format.html { | ||||
flash[:notice] = l(:notice_successful_create) | ||||
redirect_to :controller => 'projects', :action => 'settings', :id => @project | ||||
} | ||||
|
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 | ||
|
r330 | end | ||
|
r2608 | |||
def copy | ||||
@issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position") | ||||
@trackers = Tracker.all | ||||
@root_projects = Project.find(:all, | ||||
:conditions => "parent_id IS NULL AND status = #{Project::STATUS_ACTIVE}", | ||||
:order => 'name') | ||||
|
r2861 | @source_project = Project.find(params[:id]) | ||
|
r2608 | if request.get? | ||
|
r2861 | @project = Project.copy_from(@source_project) | ||
|
r2608 | if @project | ||
@project.identifier = Project.next_identifier if Setting.sequential_project_identifiers? | ||||
else | ||||
redirect_to :controller => 'admin', :action => 'projects' | ||||
end | ||||
else | ||||
|
r3494 | Mailer.with_deliveries(params[:notifications] == '1') do | ||
|
r4378 | @project = Project.new | ||
@project.safe_attributes = params[:project] | ||||
|
r3494 | @project.enabled_module_names = params[:enabled_modules] | ||
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) | ||||
|
r4071 | redirect_to :controller => 'projects', :action => 'settings' | ||
|
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 | ||||
|
r4071 | redirect_to :controller => 'projects', :action => 'settings' | ||
|
r3494 | end | ||
|
r3128 | end | ||
|
r2878 | end | ||
rescue ActiveRecord::RecordNotFound | ||||
redirect_to :controller => 'admin', :action => 'projects' | ||||
|
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 | ||||
|
r2635 | @users_by_role = @project.users_by_role | ||
|
r2302 | @subprojects = @project.children.visible | ||
|
r334 | @news = @project.news.find(:all, :limit => 5, :include => [ :author, :project ], :order => "#{News.table_name}.created_on DESC") | ||
|
r1164 | @trackers = @project.rolled_up_trackers | ||
|
r1283 | |||
cond = @project.project_condition(Setting.display_subprojects_issues?) | ||||
|
r2340 | |||
@open_issues_by_tracker = Issue.visible.count(:group => :tracker, | ||||
|
r1164 | :include => [:project, :status, :tracker], | ||
|
r1283 | :conditions => ["(#{cond}) AND #{IssueStatus.table_name}.is_closed=?", false]) | ||
|
r2340 | @total_issues_by_tracker = Issue.visible.count(:group => :tracker, | ||
|
r1164 | :include => [:project, :status, :tracker], | ||
|
r1283 | :conditions => cond) | ||
|
r2340 | |||
|
r1162 | TimeEntry.visible_by(User.current) do | ||
@total_hours = TimeEntry.sum(:hours, | ||||
:include => :project, | ||||
|
r1283 | :conditions => cond).to_f | ||
|
r1162 | end | ||
|
r663 | @key = User.current.rss_key | ||
|
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 | ||||
|
r907 | @trackers = Tracker.all | ||
|
r714 | @repository ||= @project.repository | ||
@wiki ||= @project.wiki | ||||
|
r330 | end | ||
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| | ||||
format.html { | ||||
flash[:notice] = l(:notice_successful_update) | ||||
redirect_to :action => 'settings', :id => @project | ||||
} | ||||
|
r4342 | format.api { head :ok } | ||
|
r3956 | end | ||
|
r3199 | else | ||
|
r3956 | respond_to do |format| | ||
format.html { | ||||
settings | ||||
render :action => 'settings' | ||||
} | ||||
|
r4342 | format.api { render_validation_errors(@project) } | ||
|
r330 | end | ||
|
r5 | end | ||
|
r330 | end | ||
|
r714 | |||
def modules | ||||
@project.enabled_module_names = params[:enabled_modules] | ||||
|
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 | ||
def unarchive | ||||
@project.unarchive if request.post? && !@project.active? | ||||
|
r2706 | redirect_to(url_for(:controller => 'admin', :action => 'projects', :status => params[:status])) | ||
|
r546 | end | ||
|
r10 | # Delete @project | ||
|
r330 | def destroy | ||
|
r546 | @project_to_destroy = @project | ||
|
r3199 | if request.get? | ||
# display confirmation view | ||||
else | ||||
|
r4342 | if api_request? || params[:confirm] | ||
|
r3199 | @project_to_destroy.destroy | ||
respond_to do |format| | ||||
format.html { redirect_to :controller => 'admin', :action => 'projects' } | ||||
|
r4342 | format.api { head :ok } | ||
|
r3199 | end | ||
end | ||||
|
r330 | end | ||
|
r546 | # hide project in layout | ||
@project = nil | ||||
|
r330 | end | ||
private | ||||
|
r1213 | def find_optional_project | ||
return true unless params[:id] | ||||
@project = Project.find(params[:id]) | ||||
authorize | ||||
rescue ActiveRecord::RecordNotFound | ||||
render_404 | ||||
end | ||||
|
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 | ||