# Redmine - project management software # Copyright (C) 2006-2013 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 VersionsController < ApplicationController layout "base_projects" menu_item :roadmap model_object Version before_filter :find_model_object, :except => [:index, :new, :create, :close_completed,:judge_version_title] #before_filter :find_model_object_contest, :except => [:index, :new, :create] before_filter :find_project_from_association, :except => [:index, :new, :create, :close_completed, :judge_version_title] before_filter :find_project_by_project_id, :only => [:index, :new, :create, :close_completed,:judge_version_title] before_filter :authorize, :except => [:judge_version_title] accept_api_auth :index, :show, :create, :update, :destroy helper :custom_fields helper :projects helper :project_score def index # 顶部导航 @project_menu_type = 7 type = params[:type] case type when nil,"1" @versions = @project.versions.reorder('updated_on desc') when "2" @versions = @project.versions.where(:status => 'open').reorder('updated_on desc') when "3" @versions = @project.versions.where(:status => 'locked').reorder('updated_on desc') when "4" @versions = @project.versions.where(:status => 'closed').reorder('updated_on desc') end @versions_count = Version.where(:project_id => @project.id).count @versions_open_count = Version.where(:project_id => @project.id, :status => "open").count @versions_locked_count = Version.where(:project_id => @project.id, :status => "locked").count @versions_closed_count = Version.where(:project_id => @project.id, :status => "closed").count @versions_count = version_type_count(type, @versions_count, @versions_open_count, @versions_locked_count, @versions_closed_count) @limit = 10 @is_remote = true @version_pages = Paginator.new @versions_count, @limit, params['page'] || 1 @offset ||= @version_pages.offset @versions = paginateHelper @versions, @limit respond_to do |format| format.html format.js format.api # format.html { # @trackers = @project.trackers.sorted.all # retrieve_selected_tracker_ids(@trackers, @trackers.select {|t| t.is_in_roadmap?}) # @with_subprojects = params[:with_subprojects].nil? ? Setting.display_subprojects_issues? : (params[:with_subprojects] == '1') # project_ids = @with_subprojects ? @project.self_and_descendants.collect(&:id) : [@project.id] # # @versions = @project.shared_versions || [] # @versions += @project.rolled_up_versions.visible if @with_subprojects # #added by young # @versions = @versions.uniq.reverse#Modified by young # unless params[:completed] # @completed_versions = @versions.select {|version| version.closed? || version.completed? } # @versions -= @completed_versions # end # @offset, @limit = api_offset_and_limit({:limit => 4}) # @versions_count = @versions.count # @versions_pages = Paginator.new @versions_count, @limit, params['page'] # @offset ||= @versions_pages.offset # @versions = @versions.slice(@offset, @limit) # #end by young # # @issues_by_version = {} # if @selected_tracker_ids.any? && @versions.any? # issues = Issue.visible.all( # :include => [:project, :status, :tracker, :priority, :fixed_version], # :conditions => {:tracker_id => @selected_tracker_ids, :project_id => project_ids, :fixed_version_id => @versions.map(&:id)}, # :order => "#{Project.table_name}.lft, #{Tracker.table_name}.position, #{Issue.table_name}.id" # ) # @issues_by_version = issues.group_by(&:fixed_version) # end # @versions.reject! {|version| !project_ids.include?(version.project_id) && @issues_by_version[version].blank?} # } # format.api { # @versions = @project.shared_versions.all # } end end # 统计各种类型数量 def version_type_count type, all_count, opened_count, locked_count, closed_count case type when nil, "1" all_count when "2" opened_count when "3" locked_count when "4" closed_count end end def show # 顶部导航 @project_menu_type = 7 respond_to do |format| @version_issues = @version.fixed_issues.visible. includes(:status, :tracker, :priority). reorder("#{Tracker.table_name}.position, #{Issue.table_name}.id"). all @issue_count = @version_issues.count @limit = 20 @issue_pages = Paginator.new @issue_count, @limit, params['page'] || 1 # @offset ||= @issue_pages.offset @issues = paginateHelper @version_issues, @limit @version_issue_assigned_name = @version_issues.sort_by{ |i| Issue.where(:project_id => @project.id , :assigned_to_id => i.assigned_to_id, :fixed_version_id => @version.id).count }.reverse.group_by(&:assigned_to_id) format.html { # @issues = @version.fixed_issues.visible. # includes(:status, :tracker, :priority). # reorder("#{Tracker.table_name}.position, #{Issue.table_name}.id"). # all } format.api format.xls { @issues = @version.fixed_issues.visible.includes(:status, :tracker, :priority).reorder("#{Tracker.table_name}.position, #{Issue.table_name}.id").all filename = "#{@version.name.to_s}_#{l(:label_issue_list_xls)}.xls" send_data(issue_list_xls(@issues), :type => 'application/octet-stream', :filename => filename_for_content_disposition(filename)) } end end def new # @version = @project.versions.build # @version.safe_attributes = params[:version] # respond_to do |format| @is_setting = params[:is_setting] @is_create = params[:is_create] @is_issue = params[:is_issue] @issue_project_id = params[:issue_project_id] #@@issue = Issue.find(params[:issue].to_i) format.js end end def create # 项目配置中新建 @is_setting = params[:is_setting] @is_issue = params[:is_issue] @is_create = params[:is_create] @issue_project_id = params[:issue_project_id] # @issue = current_issue @version = @project.versions.build if params[:version] attributes = params[:version].dup attributes.delete('sharing') unless attributes.nil? || @version.allowed_sharings.include?(attributes['sharing']) @version.safe_attributes = attributes @version.user_id = User.current.id end if request.post? if @version.save respond_to do |format| format.html{ if @is_create redirect_to project_versions_path(@project) elsif @is_issue redirect_to new_project_issue_path(@project) end } format.js format.api do render :action => 'show', :status => :created, :location => version_url(@version) end end else respond_to do |format| # format.html { flash[:error] = @version.errors.full_messages.flatten.to_s # redirect_to settings_project_url(@project, :tab => 'versions') } format.js { @error = @version.errors.full_messages.flatten.to_s } format.api { render_validation_errors(@version) } end end end end def edit @is_setting = params[:is_setting] @is_create = params[:is_create] @is_index = params[:is_index] @version_id = params[:id] end def update @is_setting = params[:is_setting] @is_create = params[:is_create] @is_index = params[:is_index] @flag = params[:flag] if request.put? && params[:version] # 处理里程碑里面的更新 if @flag.to_i == 1 @version.update_attribute(:status, params[:status]) if @version.save respond_to do |format| format.html { flash[:notice] = l(:notice_successful_update) redirect_to settings_project_path(@project, :tab => 'versions') } format.js format.api { render_api_ok } end else respond_to do |format| format.html { render :action => 'edit' } format.api { render_validation_errors(@version) } end end else attributes = params[:version].dup attributes.delete('sharing') unless @version.allowed_sharings.include?(attributes['sharing']) @version.safe_attributes = attributes @is_setting = params[:is_setting] if @version.save # 为了再setting里面局部刷新 if @is_setting @versions = @version.project.shared_versions.sort end respond_to do |format| format.html { if @is_create redirect_to version_path(@version) elsif @is_index redirect_to project_versions_path(@project) end } format.js format.api { render_api_ok } end else respond_to do |format| format.html { render :action => 'edit' } format.js format.api { render_validation_errors(@version) } end end end end end def close_completed if request.put? @project.close_completed_versions end redirect_to settings_project_url(@project, :tab => 'versions') end def close_completed_contest if request.put? @contest.close_completed_versions end redirect_to settings_contest_url(@contest, :tab => 'versions') end # 判断里程碑是否重名 # 项目内的里程碑不能重名,项目之间的里程碑能重名 def judge_version_title begin version = Version.where(:name => params[:version_name], :project_id => @project.id).first if version.blank? || version.id == params[:version_id].to_i result = {:result => true} else result = {:result => false} end rescue Exception => e puts e end render :json => result end def destroy begin project = @version.project issues = Issue.where(:fixed_version_id => @version.id) @version.destroy issues.update_all(:fixed_version_id, nil) rescue Exception => e puts e end respond_to do |format| format.js format.html { redirect_to project_versions_path(project)} format.api { render_api_ok } end end def status_by respond_to do |format| format.html { render :action => 'show' } format.js end end private def issue_list_xls issues xls_report = StringIO.new book = Spreadsheet::Workbook.new sheet1 = book.create_worksheet :name => "issues" blue = Spreadsheet::Format.new :color => :blue, :weight => :bold, :size => 10 sheet1.row(0).default_format = blue sheet1.row(0).concat([l(:issue_xls_id),l(:issue_xls_tracker_id),l(:issue_xls_title),l(:issue_xls_description),l(:issue_xls_status),l(:issue_xls_assign),l(:issue_xls_priority),l(:issue_xls_author),l(:issue_xls_created_at),l(:milestone),l(:issue_xls_start),l(:issue_xls_due),l(:issue_xls_ratio)]) count_row = 1 issues.each do |issue| sheet1[count_row,0] = issue.id sheet1[count_row,1] = issue_tracker_change(issue.tracker_id) sheet1[count_row,2] = issue.subject sheet1[count_row,3] = (issue.description.gsub(/<\/?.*?>/,"")).html_safe sheet1[count_row,4] = issue_status_change(issue.status_id) sheet1[count_row,5] = issue.assigned_to.try(:show_name) sheet1[count_row,6] = issue_priority_change(issue.priority_id) sheet1[count_row,7] = issue.author.show_name sheet1[count_row,8] = issue.created_on.nil? ? issue.created_on : issue.created_on.strftime('%Y-%m-%d %H:%M:%S') sheet1[count_row,9] = issue.fixed_version.try(:name) sheet1[count_row,10] = issue.start_date.nil? ? issue.start_date : issue.start_date.strftime('%Y-%m-%d') sheet1[count_row,11] = issue.due_date.nil? ? issue.due_date : issue.due_date.strftime('%Y-%m-%d') sheet1[count_row,12] = issue_ratio_change(issue.done_ratio, issue.status_id) count_row += 1 end book.write xls_report xls_report.string end def retrieve_selected_tracker_ids(selectable_trackers, default_trackers=nil) if ids = params[:tracker_ids] @selected_tracker_ids = (ids.is_a? Array) ? ids.collect { |id| id.to_i.to_s } : ids.split('/').collect { |id| id.to_i.to_s } else @selected_tracker_ids = (selectable_trackers).collect {|t| t.id.to_s } end end end