# frozen_string_literal: true class ProjectsController < ApplicationController include RenamingUtil include TeamsHelper include InputSanitizeHelper include ProjectsHelper include CardsViewHelper include ExperimentsHelper attr_reader :current_folder helper_method :current_folder before_action :switch_team_with_param, only: :index before_action :load_vars, only: %i(show edit update notifications sidebar experiments_cards view_type) before_action :load_current_folder, only: %i(index cards new show) before_action :check_view_permissions, only: %i(show notifications sidebar experiments_cards view_type) before_action :check_create_permissions, only: %i(new create) before_action :check_manage_permissions, only: :edit before_action :set_inline_name_editing, only: %i(show) before_action :load_exp_sort_var, only: :show before_action :reset_invalid_view_state, only: %i(index cards show) before_action :set_folder_inline_name_editing, only: %i(index cards) layout 'fluid' def index if current_team view_state = current_team.current_view_state(current_user) @current_sort = view_state.state.dig('projects', projects_view_mode, 'sort') || 'atoz' @current_view_type = view_state.state.dig('projects', 'view_type') end end def cards overview_service = ProjectsOverviewService.new(current_team, current_user, current_folder, params) title = params[:view_mode] == 'archived' ? t('projects.index.head_title_archived') : t('projects.index.head_title') if filters_included? render json: { toolbar_html: render_to_string(partial: 'projects/index/toolbar.html.erb'), cards_html: render_to_string( partial: 'projects/index/team_projects_grouped_by_folder.html.erb', locals: { projects_by_folder: overview_service.grouped_by_folder_project_cards } ) } else if current_folder breadcrumbs_html = render_to_string(partial: 'projects/index/breadcrumbs.html.erb', locals: { target_folder: current_folder, folder_page: true }) projects_cards_url = project_folder_cards_url(current_folder) title = if @inline_editable_title_config.present? render_to_string(partial: 'shared/inline_editing', locals: { initial_value: current_folder&.name, config: @inline_editable_title_config }) else current_folder.name end else breadcrumbs_html = '' projects_cards_url = cards_projects_url end render json: { projects_cards_url: projects_cards_url, breadcrumbs_html: breadcrumbs_html, title: title, toolbar_html: render_to_string(partial: 'projects/index/toolbar.html.erb'), cards_html: render_to_string( partial: 'projects/index/team_projects.html.erb', locals: { cards: overview_service.project_and_folder_cards } ) } end end def sidebar @current_sort = @project.current_view_state(current_user).state.dig('experiments', params[:view_mode], 'sort') render json: { html: render_to_string( partial: 'shared/sidebar/experiments.html.erb', locals: { project: @project, view_mode: experiments_view_mode(@project) } ) } end def new @project = current_team.projects.new(project_folder: current_folder) respond_to do |format| format.json do render json: { html: render_to_string( partial: 'projects/index/modals/new_project.html.erb' ) } end end end def create @project = current_team.projects.new(project_params) @project.created_by = current_user @project.last_modified_by = current_user if @project.save # Create user-project association user_project = UserProject.new( role: :owner, user: current_user, project: @project ) user_project.save log_activity(:create_project) message = t('projects.create.success_flash', name: escape_input(@project.name)) respond_to do |format| format.json do render json: { message: message }, status: :ok end end else respond_to do |format| format.json do render json: @project.errors, status: :unprocessable_entity end end end end def edit render json: { html: render_to_string(partial: 'projects/index/modals/edit_project_contents.html.erb', locals: { project: @project }) } end def update return_error = false flash_error = t('projects.update.error_flash', name: escape_input(@project.name)) # Check archive permissions if archiving/restoring if project_params.include? :archived if (project_params[:archived] == 'true' && !can_archive_project?(@project)) || (project_params[:archived] == 'false' && !can_restore_project?(@project)) return_error = true is_archive = project_params[:archived] == 'true' ? 'archive' : 'restore' flash_error = t("projects.#{is_archive}.error_flash", name: escape_input(@project.name)) end elsif !can_manage_project?(@project) render_403 && return end message_renamed = nil message_visibility = nil if (project_params.include? :name) && (project_params[:name] != @project.name) message_renamed = true end if (project_params.include? :visibility) && (project_params[:visibility] != @project.visibility) message_visibility = if project_params[:visibility] == 'visible' t('projects.activity.visibility_visible') else t('projects.activity.visibility_hidden') end end @project.last_modified_by = current_user if !return_error && @project.update(project_params) # Add activities if needed log_activity(:change_project_visibility, @project, visibility: message_visibility) if message_visibility.present? log_activity(:rename_project) if message_renamed.present? log_activity(:archive_project) if project_params[:archived] == 'true' log_activity(:restore_project) if project_params[:archived] == 'false' flash_success = t('projects.update.success_flash', name: escape_input(@project.name)) if project_params[:archived] == 'true' flash_success = t('projects.archive.success_flash', name: escape_input(@project.name)) elsif project_params[:archived] == 'false' flash_success = t('projects.restore.success_flash', name: escape_input(@project.name)) end respond_to do |format| format.html do # Redirect URL for archive view is different as for other views. if project_params[:archived] == 'false' # The project should be restored unless @project.archived @project.restore(current_user) end elsif @project.archived # The project should be archived @project.archive(current_user) end redirect_to projects_path flash[:success] = flash_success end format.json do render json: { status: :ok, message: flash_success } end end else return_error = true end if return_error respond_to do |format| format.html do flash[:error] = flash_error # Redirect URL for archive view is different as for other views. if URI(request.referer).path == projects_archive_path redirect_to projects_archive_path else redirect_to projects_path end end format.json do render json: { message: flash_error, errors: @project.errors }, status: :unprocessable_entity end end end end def archive_group projects = current_team.projects.active.where(id: params[:projects_ids]) counter = 0 projects.each do |project| next unless can_archive_project?(project) project.transaction do project.archive!(current_user) log_activity(:archive_project, project) counter += 1 rescue StandardError => e Rails.logger.error e.message raise ActiveRecord::Rollback end end if counter.positive? render json: { message: t('projects.archive_group.success_flash', number: counter) } else render json: { message: t('projects.archive_group.error_flash') }, status: :unprocessable_entity end end def restore_group projects = current_team.projects.archived.where(id: params[:projects_ids]) counter = 0 projects.each do |project| next unless can_restore_project?(project) project.transaction do project.restore!(current_user) log_activity(:restore_project, project) counter += 1 rescue StandardError => e Rails.logger.error e.message raise ActiveRecord::Rollback end end if counter.positive? render json: { message: t('projects.restore_group.success_flash', number: counter) } else render json: { message: t('projects.restore_group.error_flash') }, status: :unprocessable_entity end end def show # This is the "info" view current_team_switch(@project.team) view_state = @project.current_view_state(current_user) @current_sort = view_state.state.dig('experiments', experiments_view_mode(@project), 'sort') || 'atoz' @current_view_type = view_state.state.dig('experiments', 'view_type') end def experiments_cards overview_service = ExperimentsOverviewService.new(@project, current_user, params) render json: { cards_html: render_to_string( partial: 'projects/show/experiments_list.html.erb', locals: { cards: overview_service.experiments, filters_included: filters_included? } ) } end def notifications @modules = @project .assigned_modules(current_user) .order(due_date: :desc) respond_to do |format| #format.html format.json { render :json => { :html => render_to_string({ :partial => "notifications.html.erb" }) } } end end def users_filter users = current_team.users.search(false, params[:query]).map do |u| { value: u.id, label: sanitize_input(u.name), params: { avatar_url: avatar_path(u, :icon_small) } } end render json: users, status: :ok end def view_type view_state = @project.current_view_state(current_user) view_state.state['experiments']['view_type'] = view_type_params view_state.save! render json: { cards_view_type_class: cards_view_type_class(view_type_params) }, status: :ok end private def project_params params.require(:project) .permit( :name, :team_id, :visibility, :archived, :project_folder_id, :default_public_user_role_id ) end def view_type_params params.require(:project).require(:view_type) end def load_vars @project = Project.find_by(id: params[:id]) render_404 unless @project end def load_current_folder if current_team && params[:project_folder_id].present? @current_folder = current_team.project_folders.find_by(id: params[:project_folder_id]) elsif @project&.project_folder @current_folder = @project&.project_folder end end def check_view_permissions render_403 unless can_read_project?(@project) end def check_create_permissions render_403 unless can_create_projects?(current_team) end def check_manage_permissions render_403 unless can_manage_project?(@project) end def set_inline_name_editing return unless can_manage_project?(@project) @inline_editable_title_config = { name: 'title', params_group: 'project', item_id: @project.id, field_to_udpate: 'name', path_to_update: project_path(@project) } end def set_folder_inline_name_editing return if !can_update_team?(current_team) || @current_folder.nil? @inline_editable_title_config = { name: 'title', params_group: 'project_folder', item_id: @current_folder.id, field_to_udpate: 'name', path_to_update: project_folder_path(@current_folder) } end def load_exp_sort_var if params[:sort] @project.experiments_order = params[:sort].to_s @project.save end @current_sort = @project.experiments_order || 'new' end def filters_included? %i(search created_on_from created_on_to updated_on_from updated_on_to members archived_on_from archived_on_to folders_search) .any? { |param_name| params.dig(param_name).present? } end def reset_invalid_view_state view_state = if action_name == 'show' @project.current_view_state(current_user) else current_team.current_view_state(current_user) end view_state.destroy unless view_state.valid? end def log_activity(type_of, project = nil, message_items = {}) project ||= @project message_items = { project: project.id }.merge(message_items) Activities::CreateActivityService .call(activity_type: type_of, owner: current_user, subject: project, team: project.team, project: project, message_items: message_items) end end