diff --git a/app/controllers/issues_controller.rb b/app/controllers/issues_controller.rb
index 93a087361..f8d982c12 100644
--- a/app/controllers/issues_controller.rb
+++ b/app/controllers/issues_controller.rb
@@ -407,6 +407,7 @@ private
@priorities = IssuePriority.active
@allowed_statuses = @issue.new_statuses_allowed_to(User.current, true)
+ @available_watchers = (@issue.project.users.sort + @issue.watcher_users).uniq
end
def check_for_default_issue_status
diff --git a/app/controllers/watchers_controller.rb b/app/controllers/watchers_controller.rb
index c51722b6a..4c45ff7b2 100644
--- a/app/controllers/watchers_controller.rb
+++ b/app/controllers/watchers_controller.rb
@@ -64,6 +64,23 @@ class WatchersController < ApplicationController
render :text => 'Watcher added.', :layout => true
end
+ def append
+ if params[:watcher].is_a?(Hash)
+ user_ids = params[:watcher][:user_ids] || [params[:watcher][:user_id]]
+ users = User.active.find_all_by_id(user_ids)
+ respond_to do |format|
+ format.js do
+ render :update do |page|
+ users.each do |user|
+ page.select("#issue_watcher_user_ids_#{user.id}").each(&:hide)
+ end
+ page.insert_html :bottom, 'watchers_inputs', :text => watchers_checkboxes(nil, users, true)
+ end
+ end
+ end
+ end
+ end
+
def destroy
@watched.set_watcher(User.find(params[:user_id]), false) if request.post?
respond_to do |format|
@@ -77,16 +94,23 @@ class WatchersController < ApplicationController
end
def autocomplete_for_user
- @users = User.active.like(params[:q]).find(:all, :limit => 100) - @watched.watcher_users
+ @users = User.active.like(params[:q]).find(:all, :limit => 100)
+ if @watched
+ @user -= @watched.watcher_users
+ end
render :layout => false
end
private
def find_project
- klass = Object.const_get(params[:object_type].camelcase)
- return false unless klass.respond_to?('watched_by')
- @watched = klass.find(params[:object_id])
- @project = @watched.project
+ if params[:object_type] && params[:object_id]
+ klass = Object.const_get(params[:object_type].camelcase)
+ return false unless klass.respond_to?('watched_by')
+ @watched = klass.find(params[:object_id])
+ @project = @watched.project
+ elsif params[:project_id]
+ @project = Project.visible.find(params[:project_id])
+ end
rescue
render_404
end
diff --git a/app/helpers/watchers_helper.rb b/app/helpers/watchers_helper.rb
index 74bb0e1ec..fb77a90ae 100644
--- a/app/helpers/watchers_helper.rb
+++ b/app/helpers/watchers_helper.rb
@@ -63,4 +63,12 @@ module WatchersHelper
end
(lis.empty? ? "" : "
").html_safe
end
+
+ def watchers_checkboxes(object, users, checked=nil)
+ users.map do |user|
+ c = checked.nil? ? object.watched_by?(user) : checked
+ tag = check_box_tag 'issue[watcher_user_ids][]', user.id, c, :id => nil
+ content_tag 'label', "#{tag} #{h(user)}", :id => "issue_watcher_user_ids_#{user.id}", :class => "floating"
+ end.join
+ end
end
diff --git a/app/views/attachments/_form.html.erb b/app/views/attachments/_form.html.erb
index 464d3a2d7..7eea17bb7 100644
--- a/app/views/attachments/_form.html.erb
+++ b/app/views/attachments/_form.html.erb
@@ -14,6 +14,5 @@
<%= link_to_function(image_tag('delete.png'), 'removeFileField(this)', :title => (l(:button_delete))) %>
-<%= link_to l(:label_add_another_file), '#', :onclick => 'addFileField(); return false;' %>
-(<%= l(:label_max_size) %>: <%= number_to_human_size(Setting.attachment_max_size.to_i.kilobytes) %>)
-
+<%= link_to l(:label_add_another_file), '#', :onclick => 'addFileField(); return false;', :class => 'add_attachment' %>
+(<%= l(:label_max_size) %>: <%= number_to_human_size(Setting.attachment_max_size.to_i.kilobytes) %>)
diff --git a/app/views/issues/new.html.erb b/app/views/issues/new.html.erb
index 11268214f..c95daa1e4 100644
--- a/app/views/issues/new.html.erb
+++ b/app/views/issues/new.html.erb
@@ -22,10 +22,15 @@
<% if @issue.safe_attribute? 'watcher_user_ids' -%>
- <% @issue.project.users.sort.each do |user| -%>
-
- <% end -%>
-
+
+ <%= watchers_checkboxes(@issue, @available_watchers) %>
+
+
+ <%= link_to_remote l(:label_search_for_watchers),
+ :url => {:controller => 'watchers', :action => 'new', :project_id => @issue.project},
+ :method => 'get' %>
+
+
<% end %>
diff --git a/app/views/watchers/_new.html.erb b/app/views/watchers/_new.html.erb
index 9b4227ad7..c2c133ee7 100644
--- a/app/views/watchers/_new.html.erb
+++ b/app/views/watchers/_new.html.erb
@@ -1,7 +1,7 @@
<%= l(:permission_add_issue_watchers) %>
<% form_remote_tag :url => {:controller => 'watchers',
- :action => 'create',
+ :action => (watched ? 'create' : 'append'),
:object_type => watched.class.name.underscore,
:object_id => watched},
:method => :post,
@@ -22,7 +22,7 @@
:with => 'q') %>
- <%= principals_check_box_tags 'watcher[user_ids][]', watched.addable_watcher_users %>
+ <%= principals_check_box_tags 'watcher[user_ids][]', (watched ? watched.addable_watcher_users : User.active.all(:limit => 100)) %>
diff --git a/config/locales/en.yml b/config/locales/en.yml
index 21e12ba17..6180a860e 100644
--- a/config/locales/en.yml
+++ b/config/locales/en.yml
@@ -844,6 +844,7 @@ en:
label_copy_attachments: Copy attachments
label_item_position: "%{position} of %{count}"
label_completed_versions: Completed versions
+ label_search_for_watchers: Search for watchers to add
button_login: Login
button_submit: Submit
diff --git a/config/routes.rb b/config/routes.rb
index 3e98e9043..25a7ca59b 100644
--- a/config/routes.rb
+++ b/config/routes.rb
@@ -124,6 +124,8 @@ ActionController::Routing::Routes.draw do |map|
:conditions => {:method => :get}
map.connect 'watchers', :controller=> 'watchers', :action => 'create',
:conditions => {:method => :post}
+ map.connect 'watchers/append', :controller=> 'watchers', :action => 'append',
+ :conditions => {:method => :post}
map.connect 'watchers/destroy', :controller=> 'watchers', :action => 'destroy',
:conditions => {:method => :post}
map.connect 'watchers/watch', :controller=> 'watchers', :action => 'watch',
diff --git a/public/stylesheets/application.css b/public/stylesheets/application.css
index 937a69a83..72090235e 100644
--- a/public/stylesheets/application.css
+++ b/public/stylesheets/application.css
@@ -265,6 +265,12 @@ div.projects h3 { background: url(../images/projects.png) no-repeat 0% 50%; padd
#watchers a.delete:hover {opacity: 1;}
#watchers img.gravatar {margin: 0 4px 2px 0;}
+span#watchers_inputs {overflow:auto; display:block;}
+span.search_for_watchers {display:block;}
+span.search_for_watchers, span.add_attachment {font-size:80%; line-height:2.5em;}
+span.search_for_watchers a, span.add_attachment a {padding-left:16px; background: url(../images/bullet_add.png) no-repeat 0 50%; }
+
+
.highlight { background-color: #FCFD8D;}
.highlight.token-1 { background-color: #faa;}
.highlight.token-2 { background-color: #afa;}
diff --git a/test/functional/issues_controller_test.rb b/test/functional/issues_controller_test.rb
index 199e9317e..e91acb20c 100644
--- a/test/functional/issues_controller_test.rb
+++ b/test/functional/issues_controller_test.rb
@@ -1680,6 +1680,21 @@ class IssuesControllerTest < ActionController::TestCase
:value => 'Value for field 2'}
end
+ def test_post_create_with_failure_should_preserve_watchers
+ assert !User.find(8).member_of?(Project.find(1))
+
+ @request.session[:user_id] = 2
+ post :create, :project_id => 1,
+ :issue => {:tracker_id => 1,
+ :watcher_user_ids => ['3', '8']}
+ assert_response :success
+ assert_template 'new'
+
+ assert_tag 'input', :attributes => {:name => 'issue[watcher_user_ids][]', :value => '2', :checked => nil}
+ assert_tag 'input', :attributes => {:name => 'issue[watcher_user_ids][]', :value => '3', :checked => 'checked'}
+ assert_tag 'input', :attributes => {:name => 'issue[watcher_user_ids][]', :value => '8', :checked => 'checked'}
+ end
+
def test_post_create_should_ignore_non_safe_attributes
@request.session[:user_id] = 2
assert_nothing_raised do
diff --git a/test/functional/watchers_controller_test.rb b/test/functional/watchers_controller_test.rb
index bfd93fe0d..03b1a977b 100644
--- a/test/functional/watchers_controller_test.rb
+++ b/test/functional/watchers_controller_test.rb
@@ -68,6 +68,13 @@ class WatchersControllerTest < ActionController::TestCase
assert_select_rjs :replace_html, 'ajax-modal'
end
+ def test_new_for_new_record
+ @request.session[:user_id] = 2
+ xhr :get, :new, :project_id => 1
+ assert_response :success
+ assert_select_rjs :replace_html, 'ajax-modal'
+ end
+
def test_create
@request.session[:user_id] = 2
assert_difference('Watcher.count') do
@@ -91,6 +98,18 @@ class WatchersControllerTest < ActionController::TestCase
assert Issue.find(2).watched_by?(User.find(7))
end
+ def test_append
+ @request.session[:user_id] = 2
+ assert_no_difference 'Watcher.count' do
+ xhr :post, :append, :watcher => {:user_ids => ['4', '7']}
+ assert_response :success
+ assert_select_rjs :insert_html, 'watchers_inputs' do
+ assert_select 'input[name=?][value=4]', 'issue[watcher_user_ids][]'
+ assert_select 'input[name=?][value=7]', 'issue[watcher_user_ids][]'
+ end
+ end
+ end
+
def test_remove_watcher
@request.session[:user_id] = 2
assert_difference('Watcher.count', -1) do
diff --git a/test/integration/routing/watchers_test.rb b/test/integration/routing/watchers_test.rb
index 441d2533c..00410289b 100644
--- a/test/integration/routing/watchers_test.rb
+++ b/test/integration/routing/watchers_test.rb
@@ -23,6 +23,10 @@ class RoutingWatchersTest < ActionController::IntegrationTest
{ :method => 'get', :path => "/watchers/new" },
{ :controller => 'watchers', :action => 'new' }
)
+ assert_routing(
+ { :method => 'post', :path => "/watchers/append" },
+ { :controller => 'watchers', :action => 'append' }
+ )
assert_routing(
{ :method => 'post', :path => "/watchers" },
{ :controller => 'watchers', :action => 'create' }