Bulk watch/unwatch issues from the context menu (#7159).
git-svn-id: svn+ssh://rubyforge.org/var/svn/redmine/trunk@11339 e93f8b46-1217-0410-a6f0-8f06a7374b81
This commit is contained in:
parent
7303cc416c
commit
856ef810b4
|
@ -16,23 +16,19 @@
|
||||||
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
|
||||||
class WatchersController < ApplicationController
|
class WatchersController < ApplicationController
|
||||||
before_filter :find_project
|
before_filter :require_login, :find_watchables, :only => [:watch, :unwatch]
|
||||||
before_filter :require_login, :check_project_privacy, :only => [:watch, :unwatch]
|
|
||||||
before_filter :authorize, :only => [:new, :destroy]
|
|
||||||
accept_api_auth :create, :destroy
|
|
||||||
|
|
||||||
def watch
|
def watch
|
||||||
if @watched.respond_to?(:visible?) && !@watched.visible?(User.current)
|
set_watcher(@watchables, User.current, true)
|
||||||
render_403
|
|
||||||
else
|
|
||||||
set_watcher(User.current, true)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def unwatch
|
def unwatch
|
||||||
set_watcher(User.current, false)
|
set_watcher(@watchables, User.current, false)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
before_filter :find_project, :authorize, :only => [:new, :create, :append, :destroy, :autocomplete_for_user]
|
||||||
|
accept_api_auth :create, :destroy
|
||||||
|
|
||||||
def new
|
def new
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -77,7 +73,8 @@ class WatchersController < ApplicationController
|
||||||
render :layout => false
|
render :layout => false
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def find_project
|
def find_project
|
||||||
if params[:object_type] && params[:object_id]
|
if params[:object_type] && params[:object_id]
|
||||||
klass = Object.const_get(params[:object_type].camelcase)
|
klass = Object.const_get(params[:object_type].camelcase)
|
||||||
|
@ -91,11 +88,22 @@ private
|
||||||
render_404
|
render_404
|
||||||
end
|
end
|
||||||
|
|
||||||
def set_watcher(user, watching)
|
def find_watchables
|
||||||
@watched.set_watcher(user, watching)
|
klass = Object.const_get(params[:object_type].camelcase) rescue nil
|
||||||
|
if klass && klass.respond_to?('watched_by')
|
||||||
|
@watchables = klass.find_all_by_id(Array.wrap(params[:object_id]))
|
||||||
|
raise Unauthorized if @watchables.any? {|w| w.respond_to?(:visible?) && !w.visible?}
|
||||||
|
end
|
||||||
|
render_404 unless @watchables.present?
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_watcher(watchables, user, watching)
|
||||||
|
watchables.each do |watchable|
|
||||||
|
watchable.set_watcher(user, watching)
|
||||||
|
end
|
||||||
respond_to do |format|
|
respond_to do |format|
|
||||||
format.html { redirect_to_referer_or {render :text => (watching ? 'Watcher added.' : 'Watcher removed.'), :layout => true}}
|
format.html { redirect_to_referer_or {render :text => (watching ? 'Watcher added.' : 'Watcher removed.'), :layout => true}}
|
||||||
format.js { render :partial => 'set_watcher', :locals => {:user => user, :watched => @watched} }
|
format.js { render :partial => 'set_watcher', :locals => {:user => user, :watched => watchables} }
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -24,21 +24,28 @@ module WatchersHelper
|
||||||
watcher_link(object, user)
|
watcher_link(object, user)
|
||||||
end
|
end
|
||||||
|
|
||||||
def watcher_link(object, user)
|
def watcher_link(objects, user)
|
||||||
return '' unless user && user.logged? && object.respond_to?('watched_by?')
|
return '' unless user && user.logged?
|
||||||
watched = object.watched_by?(user)
|
objects = Array.wrap(objects)
|
||||||
css = [watcher_css(object), watched ? 'icon icon-fav' : 'icon icon-fav-off'].join(' ')
|
|
||||||
url = {:controller => 'watchers',
|
watched = objects.any? {|object| object.watched_by?(user)}
|
||||||
:action => (watched ? 'unwatch' : 'watch'),
|
css = [watcher_css(objects), watched ? 'icon icon-fav' : 'icon icon-fav-off'].join(' ')
|
||||||
:object_type => object.class.to_s.underscore,
|
text = watched ? l(:button_unwatch) : l(:button_watch)
|
||||||
:object_id => object.id}
|
url = {
|
||||||
link_to((watched ? l(:button_unwatch) : l(:button_watch)), url,
|
:controller => 'watchers',
|
||||||
:remote => true, :method => 'post', :class => css)
|
:action => (watched ? 'unwatch' : 'watch'),
|
||||||
|
:object_type => objects.first.class.to_s.underscore,
|
||||||
|
:object_id => (objects.size == 1 ? objects.first.id : objects.map(&:id).sort)
|
||||||
|
}
|
||||||
|
|
||||||
|
link_to text, url, :remote => true, :method => 'post', :class => css
|
||||||
end
|
end
|
||||||
|
|
||||||
# Returns the css class used to identify watch links for a given +object+
|
# Returns the css class used to identify watch links for a given +object+
|
||||||
def watcher_css(object)
|
def watcher_css(objects)
|
||||||
"#{object.class.to_s.underscore}-#{object.id}-watcher"
|
objects = Array.wrap(objects)
|
||||||
|
id = (objects.size == 1 ? objects.first.id : 'bulk')
|
||||||
|
"#{objects.first.class.to_s.underscore}-#{id}-watcher"
|
||||||
end
|
end
|
||||||
|
|
||||||
# Returns a comma separated list of users watching the given object
|
# Returns a comma separated list of users watching the given object
|
||||||
|
|
|
@ -117,10 +117,11 @@
|
||||||
</li>
|
</li>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
|
||||||
|
<% if User.current.logged? %>
|
||||||
|
<li><%= watcher_link(@issues, User.current) %></li>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
<% if @issue.present? %>
|
<% if @issue.present? %>
|
||||||
<% if User.current.logged? %>
|
|
||||||
<li><%= watcher_link(@issue, User.current) %></li>
|
|
||||||
<% end %>
|
|
||||||
<% if @can[:log_time] -%>
|
<% if @can[:log_time] -%>
|
||||||
<li><%= context_menu_link l(:button_log_time), new_issue_time_entry_path(@issue),
|
<li><%= context_menu_link l(:button_log_time), new_issue_time_entry_path(@issue),
|
||||||
:class => 'icon-time-add' %></li>
|
:class => 'icon-time-add' %></li>
|
||||||
|
|
|
@ -2,8 +2,9 @@
|
||||||
|
|
||||||
<%= form_tag({:controller => 'watchers',
|
<%= form_tag({:controller => 'watchers',
|
||||||
:action => (watched ? 'create' : 'append'),
|
:action => (watched ? 'create' : 'append'),
|
||||||
:object_type => watched.class.name.underscore,
|
:object_type => (watched && watched.class.name.underscore),
|
||||||
:object_id => watched},
|
:object_id => watched,
|
||||||
|
:project_id => @project},
|
||||||
:remote => true,
|
:remote => true,
|
||||||
:method => :post,
|
:method => :post,
|
||||||
:id => 'new-watcher-form') do %>
|
:id => 'new-watcher-form') do %>
|
||||||
|
@ -11,8 +12,9 @@
|
||||||
<p><%= label_tag 'user_search', l(:label_user_search) %><%= text_field_tag 'user_search', nil %></p>
|
<p><%= label_tag 'user_search', l(:label_user_search) %><%= text_field_tag 'user_search', nil %></p>
|
||||||
<%= javascript_tag "observeSearchfield('user_search', 'users_for_watcher', '#{ escape_javascript url_for(:controller => 'watchers',
|
<%= javascript_tag "observeSearchfield('user_search', 'users_for_watcher', '#{ escape_javascript url_for(:controller => 'watchers',
|
||||||
:action => 'autocomplete_for_user',
|
:action => 'autocomplete_for_user',
|
||||||
:object_type => watched.class.name.underscore,
|
:object_type => (watched && watched.class.name.underscore),
|
||||||
:object_id => watched) }')" %>
|
:object_id => watched,
|
||||||
|
:project_id => @project) }')" %>
|
||||||
|
|
||||||
<div id="users_for_watcher">
|
<div id="users_for_watcher">
|
||||||
<%= principals_check_box_tags 'watcher[user_ids][]', (watched ? watched.addable_watcher_users : User.active.all(:limit => 100)) %>
|
<%= principals_check_box_tags 'watcher[user_ids][]', (watched ? watched.addable_watcher_users : User.active.all(:limit => 100)) %>
|
||||||
|
|
|
@ -127,7 +127,7 @@ Redmine::AccessControl.map do |map|
|
||||||
map.permission :save_queries, {:queries => [:new, :create, :edit, :update, :destroy]}, :require => :loggedin
|
map.permission :save_queries, {:queries => [:new, :create, :edit, :update, :destroy]}, :require => :loggedin
|
||||||
# Watchers
|
# Watchers
|
||||||
map.permission :view_issue_watchers, {}, :read => true
|
map.permission :view_issue_watchers, {}, :read => true
|
||||||
map.permission :add_issue_watchers, {:watchers => :new}
|
map.permission :add_issue_watchers, {:watchers => [:new, :create, :append, :autocomplete_for_user]}
|
||||||
map.permission :delete_issue_watchers, {:watchers => :destroy}
|
map.permission :delete_issue_watchers, {:watchers => :destroy}
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -25,7 +25,7 @@ class WatchersControllerTest < ActionController::TestCase
|
||||||
User.current = nil
|
User.current = nil
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_watch
|
def test_watch_a_single_object
|
||||||
@request.session[:user_id] = 3
|
@request.session[:user_id] = 3
|
||||||
assert_difference('Watcher.count') do
|
assert_difference('Watcher.count') do
|
||||||
xhr :post, :watch, :object_type => 'issue', :object_id => '1'
|
xhr :post, :watch, :object_type => 'issue', :object_id => '1'
|
||||||
|
@ -35,6 +35,27 @@ class WatchersControllerTest < ActionController::TestCase
|
||||||
assert Issue.find(1).watched_by?(User.find(3))
|
assert Issue.find(1).watched_by?(User.find(3))
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_watch_a_collection_with_a_single_object
|
||||||
|
@request.session[:user_id] = 3
|
||||||
|
assert_difference('Watcher.count') do
|
||||||
|
xhr :post, :watch, :object_type => 'issue', :object_id => ['1']
|
||||||
|
assert_response :success
|
||||||
|
assert_include '$(".issue-1-watcher")', response.body
|
||||||
|
end
|
||||||
|
assert Issue.find(1).watched_by?(User.find(3))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_watch_a_collection_with_multiple_objects
|
||||||
|
@request.session[:user_id] = 3
|
||||||
|
assert_difference('Watcher.count', 2) do
|
||||||
|
xhr :post, :watch, :object_type => 'issue', :object_id => ['1', '3']
|
||||||
|
assert_response :success
|
||||||
|
assert_include '$(".issue-bulk-watcher")', response.body
|
||||||
|
end
|
||||||
|
assert Issue.find(1).watched_by?(User.find(3))
|
||||||
|
assert Issue.find(3).watched_by?(User.find(3))
|
||||||
|
end
|
||||||
|
|
||||||
def test_watch_should_be_denied_without_permission
|
def test_watch_should_be_denied_without_permission
|
||||||
Role.find(2).remove_permission! :view_issues
|
Role.find(2).remove_permission! :view_issues
|
||||||
@request.session[:user_id] = 3
|
@request.session[:user_id] = 3
|
||||||
|
@ -70,6 +91,20 @@ class WatchersControllerTest < ActionController::TestCase
|
||||||
assert !Issue.find(1).watched_by?(User.find(3))
|
assert !Issue.find(1).watched_by?(User.find(3))
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_unwatch_a_collection_with_multiple_objects
|
||||||
|
@request.session[:user_id] = 3
|
||||||
|
Watcher.create!(:user_id => 3, :watchable => Issue.find(1))
|
||||||
|
Watcher.create!(:user_id => 3, :watchable => Issue.find(3))
|
||||||
|
|
||||||
|
assert_difference('Watcher.count', -2) do
|
||||||
|
xhr :post, :unwatch, :object_type => 'issue', :object_id => ['1', '3']
|
||||||
|
assert_response :success
|
||||||
|
assert_include '$(".issue-bulk-watcher")', response.body
|
||||||
|
end
|
||||||
|
assert !Issue.find(1).watched_by?(User.find(3))
|
||||||
|
assert !Issue.find(3).watched_by?(User.find(3))
|
||||||
|
end
|
||||||
|
|
||||||
def test_new
|
def test_new
|
||||||
@request.session[:user_id] = 2
|
@request.session[:user_id] = 2
|
||||||
xhr :get, :new, :object_type => 'issue', :object_id => '2'
|
xhr :get, :new, :object_type => 'issue', :object_id => '2'
|
||||||
|
@ -77,7 +112,7 @@ class WatchersControllerTest < ActionController::TestCase
|
||||||
assert_match /ajax-modal/, response.body
|
assert_match /ajax-modal/, response.body
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_new_for_new_record_with_id
|
def test_new_for_new_record_with_project_id
|
||||||
@request.session[:user_id] = 2
|
@request.session[:user_id] = 2
|
||||||
xhr :get, :new, :project_id => 1
|
xhr :get, :new, :project_id => 1
|
||||||
assert_response :success
|
assert_response :success
|
||||||
|
@ -85,7 +120,7 @@ class WatchersControllerTest < ActionController::TestCase
|
||||||
assert_match /ajax-modal/, response.body
|
assert_match /ajax-modal/, response.body
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_new_for_new_record_with_identifier
|
def test_new_for_new_record_with_project_identifier
|
||||||
@request.session[:user_id] = 2
|
@request.session[:user_id] = 2
|
||||||
xhr :get, :new, :project_id => 'ecookbook'
|
xhr :get, :new, :project_id => 'ecookbook'
|
||||||
assert_response :success
|
assert_response :success
|
||||||
|
@ -117,7 +152,8 @@ class WatchersControllerTest < ActionController::TestCase
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_autocomplete_on_watchable_creation
|
def test_autocomplete_on_watchable_creation
|
||||||
xhr :get, :autocomplete_for_user, :q => 'mi'
|
@request.session[:user_id] = 2
|
||||||
|
xhr :get, :autocomplete_for_user, :q => 'mi', :project_id => 'ecookbook'
|
||||||
assert_response :success
|
assert_response :success
|
||||||
assert_select 'input', :count => 4
|
assert_select 'input', :count => 4
|
||||||
assert_select 'input[name=?][value=1]', 'watcher[user_ids][]'
|
assert_select 'input[name=?][value=1]', 'watcher[user_ids][]'
|
||||||
|
@ -127,7 +163,8 @@ class WatchersControllerTest < ActionController::TestCase
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_autocomplete_on_watchable_update
|
def test_autocomplete_on_watchable_update
|
||||||
xhr :get, :autocomplete_for_user, :q => 'mi', :object_id => '2' , :object_type => 'issue'
|
@request.session[:user_id] = 2
|
||||||
|
xhr :get, :autocomplete_for_user, :q => 'mi', :object_id => '2' , :object_type => 'issue', :project_id => 'ecookbook'
|
||||||
assert_response :success
|
assert_response :success
|
||||||
assert_select 'input', :count => 3
|
assert_select 'input', :count => 3
|
||||||
assert_select 'input[name=?][value=2]', 'watcher[user_ids][]'
|
assert_select 'input[name=?][value=2]', 'watcher[user_ids][]'
|
||||||
|
@ -139,7 +176,7 @@ class WatchersControllerTest < ActionController::TestCase
|
||||||
def test_append
|
def test_append
|
||||||
@request.session[:user_id] = 2
|
@request.session[:user_id] = 2
|
||||||
assert_no_difference 'Watcher.count' do
|
assert_no_difference 'Watcher.count' do
|
||||||
xhr :post, :append, :watcher => {:user_ids => ['4', '7']}
|
xhr :post, :append, :watcher => {:user_ids => ['4', '7']}, :project_id => 'ecookbook'
|
||||||
assert_response :success
|
assert_response :success
|
||||||
assert_include 'watchers_inputs', response.body
|
assert_include 'watchers_inputs', response.body
|
||||||
assert_include 'issue[watcher_user_ids][]', response.body
|
assert_include 'issue[watcher_user_ids][]', response.body
|
||||||
|
|
|
@ -0,0 +1,69 @@
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
require File.expand_path('../../../test_helper', __FILE__)
|
||||||
|
|
||||||
|
class WatchersHelperTest < ActionView::TestCase
|
||||||
|
include WatchersHelper
|
||||||
|
include Redmine::I18n
|
||||||
|
|
||||||
|
fixtures :users, :issues
|
||||||
|
|
||||||
|
def setup
|
||||||
|
super
|
||||||
|
set_language_if_valid('en')
|
||||||
|
User.current = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
test '#watcher_link with a non-watched object' do
|
||||||
|
expected = link_to(
|
||||||
|
"Watch",
|
||||||
|
"/watchers/watch?object_id=1&object_type=issue",
|
||||||
|
:remote => true, :method => 'post', :class => "issue-1-watcher icon icon-fav-off"
|
||||||
|
)
|
||||||
|
assert_equal expected, watcher_link(Issue.find(1), User.find(1))
|
||||||
|
end
|
||||||
|
|
||||||
|
test '#watcher_link with a single objet array' do
|
||||||
|
expected = link_to(
|
||||||
|
"Watch",
|
||||||
|
"/watchers/watch?object_id=1&object_type=issue",
|
||||||
|
:remote => true, :method => 'post', :class => "issue-1-watcher icon icon-fav-off"
|
||||||
|
)
|
||||||
|
assert_equal expected, watcher_link([Issue.find(1)], User.find(1))
|
||||||
|
end
|
||||||
|
|
||||||
|
test '#watcher_link with a multiple objets array' do
|
||||||
|
expected = link_to(
|
||||||
|
"Watch",
|
||||||
|
"/watchers/watch?object_id%5B%5D=1&object_id%5B%5D=3&object_type=issue",
|
||||||
|
:remote => true, :method => 'post', :class => "issue-bulk-watcher icon icon-fav-off"
|
||||||
|
)
|
||||||
|
assert_equal expected, watcher_link([Issue.find(1), Issue.find(3)], User.find(1))
|
||||||
|
end
|
||||||
|
|
||||||
|
test '#watcher_link with a watched object' do
|
||||||
|
Watcher.create!(:watchable => Issue.find(1), :user => User.find(1))
|
||||||
|
|
||||||
|
expected = link_to(
|
||||||
|
"Unwatch",
|
||||||
|
"/watchers/unwatch?object_id=1&object_type=issue",
|
||||||
|
:remote => true, :method => 'post', :class => "issue-1-watcher icon icon-fav"
|
||||||
|
)
|
||||||
|
assert_equal expected, watcher_link(Issue.find(1), User.find(1))
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in New Issue