Merged r4006 from trunk.
git-svn-id: svn+ssh://rubyforge.org/var/svn/redmine/branches/1.0-stable@4018 e93f8b46-1217-0410-a6f0-8f06a7374b81
This commit is contained in:
parent
86e986f014
commit
27ef72c35e
|
@ -0,0 +1,33 @@
|
||||||
|
class ContextMenusController < ApplicationController
|
||||||
|
helper :watchers
|
||||||
|
|
||||||
|
def issues
|
||||||
|
@issues = Issue.find_all_by_id(params[:ids], :include => :project)
|
||||||
|
if (@issues.size == 1)
|
||||||
|
@issue = @issues.first
|
||||||
|
@allowed_statuses = @issue.new_statuses_allowed_to(User.current)
|
||||||
|
end
|
||||||
|
projects = @issues.collect(&:project).compact.uniq
|
||||||
|
@project = projects.first if projects.size == 1
|
||||||
|
|
||||||
|
@can = {:edit => (@project && User.current.allowed_to?(:edit_issues, @project)),
|
||||||
|
:log_time => (@project && User.current.allowed_to?(:log_time, @project)),
|
||||||
|
:update => (@project && (User.current.allowed_to?(:edit_issues, @project) || (User.current.allowed_to?(:change_status, @project) && @allowed_statuses && !@allowed_statuses.empty?))),
|
||||||
|
:move => (@project && User.current.allowed_to?(:move_issues, @project)),
|
||||||
|
:copy => (@issue && @project.trackers.include?(@issue.tracker) && User.current.allowed_to?(:add_issues, @project)),
|
||||||
|
:delete => (@project && User.current.allowed_to?(:delete_issues, @project))
|
||||||
|
}
|
||||||
|
if @project
|
||||||
|
@assignables = @project.assignable_users
|
||||||
|
@assignables << @issue.assigned_to if @issue && @issue.assigned_to && !@assignables.include?(@issue.assigned_to)
|
||||||
|
@trackers = @project.trackers
|
||||||
|
end
|
||||||
|
|
||||||
|
@priorities = IssuePriority.all.reverse
|
||||||
|
@statuses = IssueStatus.find(:all, :order => 'position')
|
||||||
|
@back = back_url
|
||||||
|
|
||||||
|
render :layout => false
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
|
@ -22,7 +22,7 @@ class IssuesController < ApplicationController
|
||||||
before_filter :find_issue, :only => [:show, :edit, :update]
|
before_filter :find_issue, :only => [:show, :edit, :update]
|
||||||
before_filter :find_issues, :only => [:bulk_edit, :move, :perform_move, :destroy]
|
before_filter :find_issues, :only => [:bulk_edit, :move, :perform_move, :destroy]
|
||||||
before_filter :find_project, :only => [:new, :create, :update_form]
|
before_filter :find_project, :only => [:new, :create, :update_form]
|
||||||
before_filter :authorize, :except => [:index, :changes, :context_menu]
|
before_filter :authorize, :except => [:index, :changes]
|
||||||
before_filter :find_optional_project, :only => [:index, :changes]
|
before_filter :find_optional_project, :only => [:index, :changes]
|
||||||
before_filter :check_for_default_issue_status, :only => [:new, :create]
|
before_filter :check_for_default_issue_status, :only => [:new, :create]
|
||||||
before_filter :build_new_issue_from_params, :only => [:new, :create]
|
before_filter :build_new_issue_from_params, :only => [:new, :create]
|
||||||
|
@ -258,35 +258,6 @@ class IssuesController < ApplicationController
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def context_menu
|
|
||||||
@issues = Issue.find_all_by_id(params[:ids], :include => :project)
|
|
||||||
if (@issues.size == 1)
|
|
||||||
@issue = @issues.first
|
|
||||||
@allowed_statuses = @issue.new_statuses_allowed_to(User.current)
|
|
||||||
end
|
|
||||||
projects = @issues.collect(&:project).compact.uniq
|
|
||||||
@project = projects.first if projects.size == 1
|
|
||||||
|
|
||||||
@can = {:edit => (@project && User.current.allowed_to?(:edit_issues, @project)),
|
|
||||||
:log_time => (@project && User.current.allowed_to?(:log_time, @project)),
|
|
||||||
:update => (@project && (User.current.allowed_to?(:edit_issues, @project) || (User.current.allowed_to?(:change_status, @project) && @allowed_statuses && !@allowed_statuses.empty?))),
|
|
||||||
:move => (@project && User.current.allowed_to?(:move_issues, @project)),
|
|
||||||
:copy => (@issue && @project.trackers.include?(@issue.tracker) && User.current.allowed_to?(:add_issues, @project)),
|
|
||||||
:delete => (@project && User.current.allowed_to?(:delete_issues, @project))
|
|
||||||
}
|
|
||||||
if @project
|
|
||||||
@assignables = @project.assignable_users
|
|
||||||
@assignables << @issue.assigned_to if @issue && @issue.assigned_to && !@assignables.include?(@issue.assigned_to)
|
|
||||||
@trackers = @project.trackers
|
|
||||||
end
|
|
||||||
|
|
||||||
@priorities = IssuePriority.all.reverse
|
|
||||||
@statuses = IssueStatus.find(:all, :order => 'position')
|
|
||||||
@back = back_url
|
|
||||||
|
|
||||||
render :layout => false
|
|
||||||
end
|
|
||||||
|
|
||||||
def update_form
|
def update_form
|
||||||
if params[:id].blank?
|
if params[:id].blank?
|
||||||
@issue = Issue.new
|
@issue = Issue.new
|
||||||
|
|
|
@ -81,4 +81,4 @@
|
||||||
<%= auto_discovery_link_tag(:atom, {:action => 'changes', :query_id => @query, :format => 'atom', :page => nil, :key => User.current.rss_key}, :title => l(:label_changes_details)) %>
|
<%= auto_discovery_link_tag(:atom, {:action => 'changes', :query_id => @query, :format => 'atom', :page => nil, :key => User.current.rss_key}, :title => l(:label_changes_details)) %>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
|
||||||
<%= context_menu :controller => 'issues', :action => 'context_menu' %>
|
<%= context_menu issues_context_menu_path %>
|
||||||
|
|
|
@ -130,4 +130,4 @@
|
||||||
<%= stylesheet_link_tag 'context_menu' %>
|
<%= stylesheet_link_tag 'context_menu' %>
|
||||||
<% end %>
|
<% end %>
|
||||||
<div id="context-menu" style="display: none;"></div>
|
<div id="context-menu" style="display: none;"></div>
|
||||||
<%= javascript_tag "new ContextMenu('#{url_for(:controller => 'issues', :action => 'context_menu')}')" %>
|
<%= javascript_tag "new ContextMenu('#{issues_context_menu_path}')" %>
|
||||||
|
|
|
@ -107,6 +107,7 @@ ActionController::Routing::Routes.draw do |map|
|
||||||
map.auto_complete_issues '/issues/auto_complete', :controller => 'auto_completes', :action => 'issues'
|
map.auto_complete_issues '/issues/auto_complete', :controller => 'auto_completes', :action => 'issues'
|
||||||
# TODO: would look nicer as /issues/:id/preview
|
# TODO: would look nicer as /issues/:id/preview
|
||||||
map.preview_issue '/issues/preview/:id', :controller => 'previews', :action => 'issue'
|
map.preview_issue '/issues/preview/:id', :controller => 'previews', :action => 'issue'
|
||||||
|
map.issues_context_menu '/issues/context_menu', :controller => 'context_menus', :action => 'issues'
|
||||||
|
|
||||||
map.with_options :controller => 'issues' do |issues_routes|
|
map.with_options :controller => 'issues' do |issues_routes|
|
||||||
issues_routes.with_options :conditions => {:method => :get} do |issues_views|
|
issues_routes.with_options :conditions => {:method => :get} do |issues_views|
|
||||||
|
|
|
@ -58,8 +58,9 @@ Redmine::AccessControl.map do |map|
|
||||||
map.permission :manage_categories, {:projects => :settings, :issue_categories => [:new, :edit, :destroy]}, :require => :member
|
map.permission :manage_categories, {:projects => :settings, :issue_categories => [:new, :edit, :destroy]}, :require => :member
|
||||||
# Issues
|
# Issues
|
||||||
map.permission :view_issues, {:projects => :roadmap,
|
map.permission :view_issues, {:projects => :roadmap,
|
||||||
:issues => [:index, :changes, :show, :context_menu],
|
:issues => [:index, :changes, :show],
|
||||||
:auto_complete => [:issues],
|
:auto_complete => [:issues],
|
||||||
|
:context_menus => [:issues],
|
||||||
:versions => [:show, :status_by],
|
:versions => [:show, :status_by],
|
||||||
:queries => :index,
|
:queries => :index,
|
||||||
:reports => [:issue_report, :issue_report_details]}
|
:reports => [:issue_report, :issue_report_details]}
|
||||||
|
|
|
@ -0,0 +1,89 @@
|
||||||
|
require File.dirname(__FILE__) + '/../test_helper'
|
||||||
|
|
||||||
|
class ContextMenusControllerTest < ActionController::TestCase
|
||||||
|
fixtures :all
|
||||||
|
|
||||||
|
def test_context_menu_one_issue
|
||||||
|
@request.session[:user_id] = 2
|
||||||
|
get :issues, :ids => [1]
|
||||||
|
assert_response :success
|
||||||
|
assert_template 'context_menu'
|
||||||
|
assert_tag :tag => 'a', :content => 'Edit',
|
||||||
|
:attributes => { :href => '/issues/1/edit',
|
||||||
|
:class => 'icon-edit' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Closed',
|
||||||
|
:attributes => { :href => '/issues/1/edit?issue%5Bstatus_id%5D=5',
|
||||||
|
:class => '' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Immediate',
|
||||||
|
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&issue%5Bpriority_id%5D=8',
|
||||||
|
:class => '' }
|
||||||
|
# Versions
|
||||||
|
assert_tag :tag => 'a', :content => '2.0',
|
||||||
|
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&issue%5Bfixed_version_id%5D=3',
|
||||||
|
:class => '' }
|
||||||
|
assert_tag :tag => 'a', :content => 'eCookbook Subproject 1 - 2.0',
|
||||||
|
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&issue%5Bfixed_version_id%5D=4',
|
||||||
|
:class => '' }
|
||||||
|
|
||||||
|
assert_tag :tag => 'a', :content => 'Dave Lopper',
|
||||||
|
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&issue%5Bassigned_to_id%5D=3',
|
||||||
|
:class => '' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Duplicate',
|
||||||
|
:attributes => { :href => '/projects/ecookbook/issues/1/copy',
|
||||||
|
:class => 'icon-duplicate' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Copy',
|
||||||
|
:attributes => { :href => '/issues/move/new?copy_options%5Bcopy%5D=t&ids%5B%5D=1',
|
||||||
|
:class => 'icon-copy' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Move',
|
||||||
|
:attributes => { :href => '/issues/move/new?ids%5B%5D=1',
|
||||||
|
:class => 'icon-move' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Delete',
|
||||||
|
:attributes => { :href => '/issues/destroy?ids%5B%5D=1',
|
||||||
|
:class => 'icon-del' }
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_context_menu_one_issue_by_anonymous
|
||||||
|
get :issues, :ids => [1]
|
||||||
|
assert_response :success
|
||||||
|
assert_template 'context_menu'
|
||||||
|
assert_tag :tag => 'a', :content => 'Delete',
|
||||||
|
:attributes => { :href => '#',
|
||||||
|
:class => 'icon-del disabled' }
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_context_menu_multiple_issues_of_same_project
|
||||||
|
@request.session[:user_id] = 2
|
||||||
|
get :issues, :ids => [1, 2]
|
||||||
|
assert_response :success
|
||||||
|
assert_template 'context_menu'
|
||||||
|
assert_tag :tag => 'a', :content => 'Edit',
|
||||||
|
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&ids%5B%5D=2',
|
||||||
|
:class => 'icon-edit' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Immediate',
|
||||||
|
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&ids%5B%5D=2&issue%5Bpriority_id%5D=8',
|
||||||
|
:class => '' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Dave Lopper',
|
||||||
|
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&ids%5B%5D=2&issue%5Bassigned_to_id%5D=3',
|
||||||
|
:class => '' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Copy',
|
||||||
|
:attributes => { :href => '/issues/move/new?copy_options%5Bcopy%5D=t&ids%5B%5D=1&ids%5B%5D=2',
|
||||||
|
:class => 'icon-copy' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Move',
|
||||||
|
:attributes => { :href => '/issues/move/new?ids%5B%5D=1&ids%5B%5D=2',
|
||||||
|
:class => 'icon-move' }
|
||||||
|
assert_tag :tag => 'a', :content => 'Delete',
|
||||||
|
:attributes => { :href => '/issues/destroy?ids%5B%5D=1&ids%5B%5D=2',
|
||||||
|
:class => 'icon-del' }
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_context_menu_multiple_issues_of_different_project
|
||||||
|
@request.session[:user_id] = 2
|
||||||
|
get :issues, :ids => [1, 2, 4]
|
||||||
|
assert_response :success
|
||||||
|
assert_template 'context_menu'
|
||||||
|
assert_tag :tag => 'a', :content => 'Delete',
|
||||||
|
:attributes => { :href => '#',
|
||||||
|
:class => 'icon-del disabled' }
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
|
@ -1024,89 +1024,6 @@ class IssuesControllerTest < ActionController::TestCase
|
||||||
assert_redirected_to :controller => 'issues', :action => 'index', :project_id => Project.find(1).identifier
|
assert_redirected_to :controller => 'issues', :action => 'index', :project_id => Project.find(1).identifier
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_context_menu_one_issue
|
|
||||||
@request.session[:user_id] = 2
|
|
||||||
get :context_menu, :ids => [1]
|
|
||||||
assert_response :success
|
|
||||||
assert_template 'context_menu'
|
|
||||||
assert_tag :tag => 'a', :content => 'Edit',
|
|
||||||
:attributes => { :href => '/issues/1/edit',
|
|
||||||
:class => 'icon-edit' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Closed',
|
|
||||||
:attributes => { :href => '/issues/1/edit?issue%5Bstatus_id%5D=5',
|
|
||||||
:class => '' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Immediate',
|
|
||||||
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&issue%5Bpriority_id%5D=8',
|
|
||||||
:class => '' }
|
|
||||||
# Versions
|
|
||||||
assert_tag :tag => 'a', :content => '2.0',
|
|
||||||
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&issue%5Bfixed_version_id%5D=3',
|
|
||||||
:class => '' }
|
|
||||||
assert_tag :tag => 'a', :content => 'eCookbook Subproject 1 - 2.0',
|
|
||||||
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&issue%5Bfixed_version_id%5D=4',
|
|
||||||
:class => '' }
|
|
||||||
|
|
||||||
assert_tag :tag => 'a', :content => 'Dave Lopper',
|
|
||||||
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&issue%5Bassigned_to_id%5D=3',
|
|
||||||
:class => '' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Duplicate',
|
|
||||||
:attributes => { :href => '/projects/ecookbook/issues/1/copy',
|
|
||||||
:class => 'icon-duplicate' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Copy',
|
|
||||||
:attributes => { :href => '/issues/move/new?copy_options%5Bcopy%5D=t&ids%5B%5D=1',
|
|
||||||
:class => 'icon-copy' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Move',
|
|
||||||
:attributes => { :href => '/issues/move/new?ids%5B%5D=1',
|
|
||||||
:class => 'icon-move' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Delete',
|
|
||||||
:attributes => { :href => '/issues/destroy?ids%5B%5D=1',
|
|
||||||
:class => 'icon-del' }
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_context_menu_one_issue_by_anonymous
|
|
||||||
get :context_menu, :ids => [1]
|
|
||||||
assert_response :success
|
|
||||||
assert_template 'context_menu'
|
|
||||||
assert_tag :tag => 'a', :content => 'Delete',
|
|
||||||
:attributes => { :href => '#',
|
|
||||||
:class => 'icon-del disabled' }
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_context_menu_multiple_issues_of_same_project
|
|
||||||
@request.session[:user_id] = 2
|
|
||||||
get :context_menu, :ids => [1, 2]
|
|
||||||
assert_response :success
|
|
||||||
assert_template 'context_menu'
|
|
||||||
assert_tag :tag => 'a', :content => 'Edit',
|
|
||||||
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&ids%5B%5D=2',
|
|
||||||
:class => 'icon-edit' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Immediate',
|
|
||||||
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&ids%5B%5D=2&issue%5Bpriority_id%5D=8',
|
|
||||||
:class => '' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Dave Lopper',
|
|
||||||
:attributes => { :href => '/issues/bulk_edit?ids%5B%5D=1&ids%5B%5D=2&issue%5Bassigned_to_id%5D=3',
|
|
||||||
:class => '' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Copy',
|
|
||||||
:attributes => { :href => '/issues/move/new?copy_options%5Bcopy%5D=t&ids%5B%5D=1&ids%5B%5D=2',
|
|
||||||
:class => 'icon-copy' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Move',
|
|
||||||
:attributes => { :href => '/issues/move/new?ids%5B%5D=1&ids%5B%5D=2',
|
|
||||||
:class => 'icon-move' }
|
|
||||||
assert_tag :tag => 'a', :content => 'Delete',
|
|
||||||
:attributes => { :href => '/issues/destroy?ids%5B%5D=1&ids%5B%5D=2',
|
|
||||||
:class => 'icon-del' }
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_context_menu_multiple_issues_of_different_project
|
|
||||||
@request.session[:user_id] = 2
|
|
||||||
get :context_menu, :ids => [1, 2, 4]
|
|
||||||
assert_response :success
|
|
||||||
assert_template 'context_menu'
|
|
||||||
assert_tag :tag => 'a', :content => 'Delete',
|
|
||||||
:attributes => { :href => '#',
|
|
||||||
:class => 'icon-del disabled' }
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_destroy_issue_with_no_time_entries
|
def test_destroy_issue_with_no_time_entries
|
||||||
assert_nil TimeEntry.find_by_issue_id(2)
|
assert_nil TimeEntry.find_by_issue_id(2)
|
||||||
@request.session[:user_id] = 2
|
@request.session[:user_id] = 2
|
||||||
|
|
|
@ -104,6 +104,8 @@ class RoutingTest < ActionController::IntegrationTest
|
||||||
|
|
||||||
should_route :get, "/issues/preview/123", :controller => 'previews', :action => 'issue', :id => '123'
|
should_route :get, "/issues/preview/123", :controller => 'previews', :action => 'issue', :id => '123'
|
||||||
should_route :post, "/issues/preview/123", :controller => 'previews', :action => 'issue', :id => '123'
|
should_route :post, "/issues/preview/123", :controller => 'previews', :action => 'issue', :id => '123'
|
||||||
|
should_route :get, "/issues/context_menu", :controller => 'context_menus', :action => 'issues'
|
||||||
|
should_route :post, "/issues/context_menu", :controller => 'context_menus', :action => 'issues'
|
||||||
end
|
end
|
||||||
|
|
||||||
context "issue categories" do
|
context "issue categories" do
|
||||||
|
|
Loading…
Reference in New Issue