123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257 |
- # Redmine - project management software
- # Copyright (C) 2006-2011 Jean-Philippe Lang
- #
- # 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
- menu_item :overview
- menu_item :roadmap, :only => :roadmap
- menu_item :settings, :only => :settings
-
- 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]
- before_filter :require_admin, :only => [ :copy, :archive, :unarchive, :destroy ]
- accept_rss_auth :index
- accept_api_auth :index, :show, :create, :update, :destroy
-
- after_filter :only => [:create, :edit, :update, :archive, :unarchive, :destroy] do |controller|
- if controller.request.post?
- controller.send :expire_action, :controller => 'welcome', :action => 'robots'
- end
- end
-
- helper :sort
- include SortHelper
- helper :custom_fields
- include CustomFieldsHelper
- helper :issues
- helper :queries
- include QueriesHelper
- helper :repositories
- include RepositoriesHelper
- include ProjectsHelper
-
- # Lists visible projects
- def index
- respond_to do |format|
- format.html {
- @projects = Project.visible.find(:all, :order => 'lft')
- }
- format.api {
- @offset, @limit = api_offset_and_limit
- @project_count = Project.visible.count
- @projects = Project.visible.all(:offset => @offset, :limit => @limit, :order => 'lft')
- }
- format.atom {
- 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)}")
- }
- end
- end
-
- def new
- @issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position")
- @trackers = Tracker.all
- @project = Project.new
- @project.safe_attributes = params[:project]
- end
-
- def create
- @issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position")
- @trackers = Tracker.all
- @project = Project.new
- @project.safe_attributes = params[:project]
-
- 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(params[:continue] ?
- {:controller => 'projects', :action => 'new', :project => {:parent_id => @project.parent_id}.reject {|k,v| v.nil?}} :
- {:controller => 'projects', :action => 'settings', :id => @project}
- )
- }
- format.api { render :action => 'show', :status => :created, :location => url_for(:controller => 'projects', :action => 'show', :id => @project.id) }
- end
- else
- respond_to do |format|
- format.html { render :action => 'new' }
- format.api { render_validation_errors(@project) }
- end
- end
-
- end
-
- 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')
- @source_project = Project.find(params[:id])
- if request.get?
- @project = Project.copy_from(@source_project)
- if @project
- @project.identifier = Project.next_identifier if Setting.sequential_project_identifiers?
- else
- redirect_to :controller => 'admin', :action => 'projects'
- end
- else
- Mailer.with_deliveries(params[:notifications] == '1') do
- @project = Project.new
- @project.safe_attributes = params[:project]
- 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)
- redirect_to :controller => 'projects', :action => 'settings', :id => @project
- 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
- redirect_to :controller => 'projects', :action => 'settings', :id => @project
- end
- end
- end
- rescue ActiveRecord::RecordNotFound
- redirect_to :controller => 'admin', :action => 'projects'
- end
-
- # Show @project
- def show
- if params[:jump]
- # try to redirect to the requested menu item
- redirect_to_project_menu_item(@project, params[:jump]) && return
- end
-
- @users_by_role = @project.users_by_role
- @subprojects = @project.children.visible.all
- @news = @project.news.find(:all, :limit => 5, :include => [ :author, :project ], :order => "#{News.table_name}.created_on DESC")
- @trackers = @project.rolled_up_trackers
-
- cond = @project.project_condition(Setting.display_subprojects_issues?)
-
- @open_issues_by_tracker = Issue.visible.count(:group => :tracker,
- :include => [:project, :status, :tracker],
- :conditions => ["(#{cond}) AND #{IssueStatus.table_name}.is_closed=?", false])
- @total_issues_by_tracker = Issue.visible.count(:group => :tracker,
- :include => [:project, :status, :tracker],
- :conditions => cond)
-
- if User.current.allowed_to?(:view_time_entries, @project)
- @total_hours = TimeEntry.visible.sum(:hours, :include => :project, :conditions => cond).to_f
- end
-
- @key = User.current.rss_key
-
- respond_to do |format|
- format.html
- format.api
- end
- end
-
- def settings
- @issue_custom_fields = IssueCustomField.find(:all, :order => "#{CustomField.table_name}.position")
- @issue_category ||= IssueCategory.new
- @member ||= @project.members.new
- @trackers = Tracker.all
- @wiki ||= @project.wiki
- end
-
- def edit
- end
-
- def update
- @project.safe_attributes = params[:project]
- 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
- }
- format.api { head :ok }
- end
- else
- respond_to do |format|
- format.html {
- settings
- render :action => 'settings'
- }
- format.api { render_validation_errors(@project) }
- end
- end
- end
-
- def modules
- @project.enabled_module_names = params[:enabled_module_names]
- flash[:notice] = l(:notice_successful_update)
- redirect_to :action => 'settings', :id => @project, :tab => 'modules'
- end
-
- def archive
- if request.post?
- unless @project.archive
- flash[:error] = l(:error_can_not_archive_project)
- end
- end
- redirect_to(url_for(:controller => 'admin', :action => 'projects', :status => params[:status]))
- end
-
- def unarchive
- @project.unarchive if request.post? && !@project.active?
- redirect_to(url_for(:controller => 'admin', :action => 'projects', :status => params[:status]))
- end
-
- # Delete @project
- def destroy
- @project_to_destroy = @project
- if api_request? || params[:confirm]
- @project_to_destroy.destroy
- respond_to do |format|
- format.html { redirect_to :controller => 'admin', :action => 'projects' }
- format.api { head :ok }
- end
- end
- # hide project in layout
- @project = nil
- end
-
- private
-
- # 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
- end
|