# frozen_string_literal: true # Redmine - project management software # Copyright (C) 2006-2023 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, :bulk_destroy] before_action :authorize, :except => [:index, :autocomplete, :list, :new, :create, :copy, :archive, :unarchive, :destroy, :bulk_destroy] before_action :authorize_global, :only => [:new, :create] before_action :require_admin, :only => [:copy, :archive, :unarchive, :bulk_destroy] accept_atom_auth :index accept_api_auth :index, :show, :create, :update, :destroy, :archive, :unarchive, :close, :reopen require_sudo_mode :destroy, :bulk_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.atom_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 DestroyProjectJob.schedule(@project_to_destroy) flash[:notice] = l(:notice_successful_delete) 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 # Delete selected projects def bulk_destroy @projects = Project.where(id: params[:ids]). where.not(status: Project::STATUS_SCHEDULED_FOR_DELETION).to_a if @projects.empty? render_404 return end if params[:confirm] == I18n.t(:general_text_Yes) DestroyProjectsJob.schedule @projects flash[:notice] = l(:notice_successful_delete) redirect_to admin_projects_path end 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] 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