123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342 |
- # frozen_string_literal: true
-
- # Redmine - project management software
- # Copyright (C) 2006-2022 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 :settings, :only => :settings
- menu_item :projects, :only => [:index, :new, :copy, :create]
-
- before_action :find_project,
- :except => [:index, :autocomplete, :list, :new, :create, :copy]
- before_action :authorize,
- :except => [:index, :autocomplete, :list, :new, :create, :copy,
- :archive, :unarchive,
- :destroy]
- before_action :authorize_global, :only => [:new, :create]
- before_action :require_admin, :only => [:copy, :archive, :unarchive]
- accept_rss_auth :index
- accept_api_auth :index, :show, :create, :update, :destroy, :archive, :unarchive, :close, :reopen
- require_sudo_mode :destroy
-
- helper :custom_fields
- helper :issues
- helper :queries
- include QueriesHelper
- helper :projects_queries
- include ProjectsQueriesHelper
- helper :repositories
- helper :members
- helper :trackers
-
- # Lists visible projects
- def index
- # try to redirect to the requested menu item
- if params[:jump] && redirect_to_menu_item(params[:jump])
- return
- end
-
- retrieve_default_query
- retrieve_project_query
- scope = project_scope
-
- respond_to do |format|
- format.html do
- # TODO: see what to do with the board view and pagination
- if @query.display_type == 'board'
- @entries = scope.to_a
- else
- @entry_count = scope.count
- @entry_pages = Paginator.new @entry_count, per_page_option, params['page']
- @entries = scope.offset(@entry_pages.offset).limit(@entry_pages.per_page).to_a
- end
- end
- format.api do
- @offset, @limit = api_offset_and_limit
- @project_count = scope.count
- @projects = scope.offset(@offset).limit(@limit).to_a
- end
- format.atom do
- projects = scope.reorder(:created_on => :desc).limit(Setting.feeds_limit.to_i).to_a
- render_feed(projects, :title => "#{Setting.app_title}: #{l(:label_project_latest)}")
- end
- format.csv do
- # Export all entries
- @entries = scope.to_a
- send_data(query_to_csv(@entries, @query, params), :type => 'text/csv; header=present', :filename => 'projects.csv')
- end
- end
- end
-
- def autocomplete
- respond_to do |format|
- format.js do
- if params[:q].present?
- @projects = Project.visible.like(params[:q]).to_a
- else
- @projects = User.current.projects.to_a
- end
- end
- end
- end
-
- def new
- @issue_custom_fields = IssueCustomField.sorted.to_a
- @trackers = Tracker.sorted.to_a
- @project = Project.new
- @project.safe_attributes = params[:project]
- end
-
- def create
- @issue_custom_fields = IssueCustomField.sorted.to_a
- @trackers = Tracker.sorted.to_a
- @project = Project.new
- @project.safe_attributes = params[:project]
-
- if @project.save
- unless User.current.admin?
- @project.add_default_member(User.current)
- end
- respond_to do |format|
- format.html do
- flash[:notice] = l(:notice_successful_create)
- if params[:continue]
- attrs = {:parent_id => @project.parent_id}.reject {|k,v| v.nil?}
- redirect_to new_project_path(attrs)
- else
- redirect_to settings_project_path(@project)
- end
- end
- format.api do
- render(
- :action => 'show',
- :status => :created,
- :location => url_for(:controller => 'projects',
- :action => 'show', :id => @project.id)
- )
- end
- 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.sorted.to_a
- @trackers = Tracker.sorted.to_a
- @source_project = Project.find(params[:id])
- if request.get?
- @project = Project.copy_from(@source_project)
- @project.identifier = Project.next_identifier if Setting.sequential_project_identifiers?
- else
- Mailer.with_deliveries(params[:notifications] == '1') do
- @project = Project.new
- @project.safe_attributes = params[:project]
- if @project.copy(@source_project, :only => params[:only])
- flash[:notice] = l(:notice_successful_create)
- redirect_to settings_project_path(@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 settings_project_path(@project)
- end
- end
- end
- rescue ActiveRecord::RecordNotFound
- # source_project not found
- render_404
- end
-
- # Show @project
- def show
- # try to redirect to the requested menu item
- if params[:jump] && redirect_to_project_menu_item(@project, params[:jump])
- return
- end
-
- respond_to do |format|
- format.html do
- @principals_by_role = @project.principals_by_role
- @subprojects = @project.children.visible.to_a
- @news = @project.news.limit(5).includes(:author, :project).reorder("#{News.table_name}.created_on DESC").to_a
- with_subprojects = Setting.display_subprojects_issues?
- @trackers = @project.rolled_up_trackers(with_subprojects).visible
-
- cond = @project.project_condition(with_subprojects)
-
- @open_issues_by_tracker = Issue.visible.open.where(cond).group(:tracker).count
- @total_issues_by_tracker = Issue.visible.where(cond).group(:tracker).count
-
- if User.current.allowed_to_view_all_time_entries?(@project)
- @total_hours = TimeEntry.visible.where(cond).sum(:hours).to_f
- @total_estimated_hours = Issue.visible.where(cond).sum(:estimated_hours).to_f
- end
-
- @key = User.current.rss_key
- end
- format.api
- end
- end
-
- def settings
- @issue_custom_fields = IssueCustomField.sorted.to_a
- @issue_category ||= IssueCategory.new
- @member ||= @project.members.new
- @trackers = Tracker.sorted.to_a
-
- @version_status = params[:version_status] || 'open'
- @version_name = params[:version_name]
- @versions = @project.shared_versions.status(@version_status).like(@version_name).sorted
- end
-
- def edit
- end
-
- def update
- @project.safe_attributes = params[:project]
- if @project.save
- respond_to do |format|
- format.html do
- flash[:notice] = l(:notice_successful_update)
- redirect_to settings_project_path(@project, params[:tab])
- end
- format.api {render_api_ok}
- end
- else
- respond_to do |format|
- format.html do
- settings
- render :action => 'settings'
- end
- format.api {render_validation_errors(@project)}
- end
- end
- end
-
- def archive
- unless @project.archive
- error = l(:error_can_not_archive_project)
- end
- respond_to do |format|
- format.html do
- flash[:error] = error if error
- redirect_to_referer_or admin_projects_path(:status => params[:status])
- end
- format.api do
- if error
- render_api_errors error
- else
- render_api_ok
- end
- end
- end
- end
-
- def unarchive
- unless @project.active?
- @project.unarchive
- end
- respond_to do |format|
- format.html{ redirect_to_referer_or admin_projects_path(:status => params[:status]) }
- format.api{ render_api_ok }
- end
- end
-
- def bookmark
- jump_box = Redmine::ProjectJumpBox.new User.current
- if request.delete?
- jump_box.delete_project_bookmark @project
- elsif request.post?
- jump_box.bookmark_project @project
- end
- respond_to do |format|
- format.js
- format.html {redirect_to project_path(@project)}
- end
- end
-
- def close
- @project.close
- respond_to do |format|
- format.html { redirect_to project_path(@project) }
- format.api { render_api_ok }
- end
- end
-
- def reopen
- @project.reopen
- respond_to do |format|
- format.html { redirect_to project_path(@project) }
- format.api { render_api_ok }
- end
- end
-
- # Delete @project
- def destroy
- unless @project.deletable?
- deny_access
- return
- end
-
- @project_to_destroy = @project
- if api_request? || params[:confirm] == @project_to_destroy.identifier
- @project_to_destroy.destroy
- respond_to do |format|
- format.html do
- redirect_to(
- User.current.admin? ? admin_projects_path : projects_path
- )
- end
- format.api {render_api_ok}
- end
- end
- # hide project in layout
- @project = nil
- end
-
- private
-
- # Returns the ProjectEntry scope for index
- def project_scope(options={})
- @query.results_scope(options)
- end
-
- def retrieve_project_query
- retrieve_query(ProjectQuery, false, :defaults => @default_columns_names)
- end
-
- def retrieve_default_query
- return if params[:query_id].present?
- return if api_request?
- return if params[:set_filter] && (params.key?(:op) || params.key?(:f))
-
- if params[:without_default].present?
- params[:set_filter] = 1
- return
- end
-
- if default_query = ProjectQuery.default
- params[:query_id] = default_query.id
- end
- end
- end
|